mirror of
https://github.com/immich-app/immich
synced 2025-10-17 18:19:27 +00:00
* Add new cli * Remove old readme * Add documentation to readme file * Add github workflow tests for cli * Fix typo in docs * Add usage info to readme * Add package-lock.json * Fix tsconfig * Cleanup * Fix lint * Cleanup package.json * Fix accidental server change * Remove rootdir from cli * Remove tsbuildinfo * Add prettierignore * Make CLI use internal openapi specs * Add ignore and dry-run features * Sort paths alphabetically * Don't remove substring * Remove shorthand for delete * Remove unused import * Remove chokidar * Set correct openapi cli generator script * Add progress bar * Rename target to asset * Add deletion progress bar * Ignore require statement * Use read streams instead of readfile * Fix github feedback * Fix upload requires * More github comments * Cleanup messages * Cleaner pattern concats * Github comments --------- Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
71 lines
2.1 KiB
TypeScript
71 lines
2.1 KiB
TypeScript
import * as fs from 'fs';
|
|
import * as mime from 'mime-types';
|
|
import { basename } from 'node:path';
|
|
import * as path from 'path';
|
|
import crypto from 'crypto';
|
|
import { AssetTypeEnum } from 'src/api/open-api';
|
|
|
|
export class CrawledAsset {
|
|
public path: string;
|
|
|
|
public assetType?: AssetTypeEnum;
|
|
public assetData?: fs.ReadStream;
|
|
public deviceAssetId?: string;
|
|
public fileCreatedAt?: string;
|
|
public fileModifiedAt?: string;
|
|
public fileExtension?: string;
|
|
public sidecarData?: Buffer;
|
|
public sidecarPath?: string;
|
|
public fileSize!: number;
|
|
public skipped = false;
|
|
|
|
constructor(path: string) {
|
|
this.path = path;
|
|
}
|
|
|
|
async readData() {
|
|
this.assetData = fs.createReadStream(this.path);
|
|
}
|
|
|
|
async process() {
|
|
const stats = await fs.promises.stat(this.path);
|
|
this.deviceAssetId = `${basename(this.path)}-${stats.size}`.replace(/\s+/g, '');
|
|
|
|
// TODO: Determine file type from extension only
|
|
const mimeType = mime.lookup(this.path);
|
|
if (!mimeType) {
|
|
throw Error('Cannot determine mime type of asset: ' + this.path);
|
|
}
|
|
this.assetType = mimeType.split('/')[0].toUpperCase() as AssetTypeEnum;
|
|
this.fileCreatedAt = stats.ctime.toISOString();
|
|
this.fileModifiedAt = stats.mtime.toISOString();
|
|
this.fileExtension = path.extname(this.path);
|
|
this.fileSize = stats.size;
|
|
|
|
// TODO: doesn't xmp replace the file extension? Will need investigation
|
|
const sideCarPath = `${this.path}.xmp`;
|
|
try {
|
|
fs.accessSync(sideCarPath, fs.constants.R_OK);
|
|
this.sidecarData = await fs.promises.readFile(sideCarPath);
|
|
this.sidecarPath = sideCarPath;
|
|
} catch (error) {}
|
|
}
|
|
|
|
async delete(): Promise<void> {
|
|
return fs.promises.unlink(this.path);
|
|
}
|
|
|
|
public async hash(): Promise<string> {
|
|
const sha1 = (filePath: string) => {
|
|
const hash = crypto.createHash('sha1');
|
|
return new Promise<string>((resolve, reject) => {
|
|
const rs = fs.createReadStream(filePath);
|
|
rs.on('error', reject);
|
|
rs.on('data', (chunk) => hash.update(chunk));
|
|
rs.on('end', () => resolve(hash.digest('hex')));
|
|
});
|
|
};
|
|
|
|
return await sha1(this.path);
|
|
}
|
|
}
|