@sync-in/server 1.8.0 → 1.8.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (55) hide show
  1. package/CHANGELOG.md +11 -2
  2. package/package.json +10 -9
  3. package/server/app.bootstrap.js +4 -2
  4. package/server/app.bootstrap.js.map +1 -1
  5. package/server/applications/files/files.controller.js +12 -4
  6. package/server/applications/files/files.controller.js.map +1 -1
  7. package/server/applications/files/files.controller.spec.js +18 -4
  8. package/server/applications/files/files.controller.spec.js.map +1 -1
  9. package/server/applications/files/services/files-manager.service.js.map +1 -1
  10. package/server/applications/files/services/files-methods.service.js +1 -6
  11. package/server/applications/files/services/files-methods.service.js.map +1 -1
  12. package/server/applications/files/utils/doc-textify/adapters/pdf.js +5 -16
  13. package/server/applications/files/utils/doc-textify/adapters/pdf.js.map +1 -1
  14. package/server/applications/users/services/users-manager.service.js +1 -2
  15. package/server/applications/users/services/users-manager.service.js.map +1 -1
  16. package/server/applications/users/users.gateway.js +6 -0
  17. package/server/applications/users/users.gateway.js.map +1 -1
  18. package/server/common/image.js +62 -58
  19. package/server/common/image.js.map +1 -1
  20. package/static/3rdpartylicenses.txt +0 -26
  21. package/static/chunk-2VMSXRCB.js +12 -0
  22. package/static/{chunk-X5XGK6T7.js → chunk-3OHSRRKH.js} +1 -1
  23. package/static/{chunk-H6NE33VX.js → chunk-3R4WKOHQ.js} +1 -1
  24. package/static/{chunk-WLMNXRBS.js → chunk-3R74L4UU.js} +1 -1
  25. package/static/{chunk-ZPF2DSQV.js → chunk-5UKZLU5H.js} +1 -1
  26. package/static/{chunk-KFJIQIGR.js → chunk-BQV4FRM6.js} +1 -1
  27. package/static/{chunk-MFLIJH6T.js → chunk-CETH7UYS.js} +1 -1
  28. package/static/{chunk-C36MW4ME.js → chunk-DIT6W7VM.js} +9 -9
  29. package/static/{chunk-YW57T2PF.js → chunk-IQSKQXC3.js} +1 -1
  30. package/static/{chunk-QZU2S5CV.js → chunk-ITUFI2BJ.js} +1 -1
  31. package/static/{chunk-PGZZP5W3.js → chunk-LCTZJ537.js} +1 -1
  32. package/static/{chunk-LVM4QB22.js → chunk-LK2UCQJ6.js} +1 -1
  33. package/static/{chunk-ZTCRGJ6Y.js → chunk-LP5TBXEN.js} +1 -1
  34. package/static/{chunk-BIKLW4YS.js → chunk-LVSNIS5P.js} +1 -1
  35. package/static/{chunk-VM4YX6Q7.js → chunk-MTVSJTIW.js} +1 -1
  36. package/static/{chunk-V3AT2BKP.js → chunk-O6FYXVHI.js} +1 -1
  37. package/static/{chunk-SHIVUDP3.js → chunk-PNR6M34W.js} +1 -1
  38. package/static/{chunk-M3XVNQZQ.js → chunk-QMRBZHE4.js} +1 -1
  39. package/static/{chunk-NO2LTNW3.js → chunk-QSJRY3TF.js} +1 -1
  40. package/static/{chunk-373XVRXW.js → chunk-QUUIRSYT.js} +1 -1
  41. package/static/{chunk-UEQCWMXD.js → chunk-RFH46UW3.js} +1 -1
  42. package/static/{chunk-AY2SZ3G6.js → chunk-RSXHRKM5.js} +1 -1
  43. package/static/{chunk-UJPPR4MX.js → chunk-RV3VZJPZ.js} +1 -1
  44. package/static/{chunk-JSWCNGXJ.js → chunk-S7HNXVRB.js} +1 -1
  45. package/static/{chunk-VKK5BSLX.js → chunk-SJR5R3Y4.js} +1 -1
  46. package/static/{chunk-UG5DMXYO.js → chunk-V3LHHZYN.js} +1 -1
  47. package/static/{chunk-HAS5ZOTR.js → chunk-VJTXJ43D.js} +1 -1
  48. package/static/{chunk-HNQRZALS.js → chunk-VQQKMY2C.js} +1 -1
  49. package/static/{chunk-MSUHTBB2.js → chunk-WSSU2HXE.js} +1 -1
  50. package/static/{chunk-GUGNR5TF.js → chunk-XDZGW64M.js} +2 -2
  51. package/static/{chunk-TPYBFZS5.js → chunk-XTRDKGKG.js} +1 -1
  52. package/static/{chunk-YEKR5OPO.js → chunk-YLWTEC3X.js} +1 -1
  53. package/static/index.html +1 -1
  54. package/static/{main-VOL6OMJ5.js → main-4H5BJY3J.js} +2 -2
  55. package/static/chunk-WJW7CT6G.js +0 -27
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-manager.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { HttpService } from '@nestjs/axios'\nimport { HttpStatus, Injectable, Logger } from '@nestjs/common'\nimport archiver, { Archiver, ArchiverError } from 'archiver'\nimport { AxiosResponse } from 'axios'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport { pipeline } from 'node:stream/promises'\nimport { extract as extractTar } from 'tar'\nimport { FastifyAuthenticatedRequest } from '../../../authentication/interfaces/auth-request.interface'\nimport { generateThumbnail } from '../../../common/image'\nimport { HTTP_METHOD } from '../../applications.constants'\nimport { SPACE_OPERATION } from '../../spaces/constants/spaces'\nimport { FastifySpaceRequest } from '../../spaces/interfaces/space-request.interface'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { realTrashPathFromSpace } from '../../spaces/utils/paths'\nimport { canAccessToSpace, haveSpaceEnvPermissions } from '../../spaces/utils/permissions'\nimport { UserModel } from '../../users/models/user.model'\nimport { DEPTH, LOCK_DEPTH } from '../../webdav/constants/webdav'\nimport { tarGzExtension } from '../constants/compress'\nimport { COMPRESSION_EXTENSION, DEFAULT_HIGH_WATER_MARK } from '../constants/files'\nimport { FILE_OPERATION } from '../constants/operations'\nimport { DOCUMENT_TYPE, SAMPLE_PATH_WITHOUT_EXT } from '../constants/samples'\nimport { CompressFileDto } from '../dto/file-operations.dto'\nimport { FileTaskEvent } from '../events/file-task-event'\nimport { FileDBProps } from '../interfaces/file-db-props.interface'\nimport { FileLock } from '../interfaces/file-lock.interface'\nimport { FileError } from '../models/file-error'\nimport { LockConflict } from '../models/file-lock-error'\nimport {\n checkFileName,\n copyFiles,\n createEmptyFile,\n dirName,\n dirSize,\n fileName,\n fileSize,\n getMimeType,\n isPathExists,\n isPathIsDir,\n makeDir,\n moveFiles,\n removeFiles,\n touchFile,\n uniqueDatedFilePath,\n uniqueFilePathFromDir,\n writeFromStream,\n writeFromStreamAndChecksum\n} from '../utils/files'\nimport { SendFile } from '../utils/send-file'\nimport { extractZip } from '../utils/unzip-file'\nimport { regExpPrivateIP } from '../utils/url-file'\nimport { FilesLockManager } from './files-lock-manager.service'\nimport { FilesQueries } from './files-queries.service'\n\n@Injectable()\nexport class FilesManager {\n /* Spaces permissions are checked in the space guard, except for the copy/move destination */\n private logger = new Logger(FilesManager.name)\n\n constructor(\n private readonly http: HttpService,\n private readonly filesQueries: FilesQueries,\n private readonly filesLockManager: FilesLockManager\n ) {}\n\n sendFileFromSpace(space: SpaceEnv, asAttachment = false, downloadName = ''): SendFile {\n return new SendFile(space.realPath, asAttachment, downloadName)\n }\n\n async saveStream(\n user: UserModel,\n space: SpaceEnv,\n req: FastifyAuthenticatedRequest,\n options: {\n checksumAlg: string\n tmpPath?: string\n }\n ): Promise<string>\n async saveStream(user: UserModel, space: SpaceEnv, req: FastifyAuthenticatedRequest, options?: any): Promise<boolean>\n async saveStream(\n user: UserModel,\n space: SpaceEnv,\n req: FastifyAuthenticatedRequest,\n options?: { dav?: { depth: LOCK_DEPTH; lockTokens: string[] }; checksumAlg?: string; tmpPath?: string }\n ): Promise<boolean | string> {\n // if tmpPath is used, we lock the final destination during the transfer\n // space.realPath is replaced by tmpPath (if allowed), if the move operation failed we remove the tmp file\n const fExists = await isPathExists(space.realPath)\n const fTmpExists = options?.tmpPath ? await isPathExists(options.tmpPath) : false\n if (fExists && req.method === HTTP_METHOD.POST) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'Resource already exists')\n }\n if (fExists && (await isPathIsDir(space.realPath))) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'The location is a directory')\n }\n if (options?.tmpPath) {\n // ensure tmpPath parent dir exists\n await makeDir(dirName(options.tmpPath), true)\n } else if (!(await isPathExists(dirName(space.realPath)))) {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n /* File Lock */\n let fileLock: FileLock | undefined\n if (options?.dav) {\n // check locks\n await this.filesLockManager.checkConflicts(space.dbFile, options?.dav?.depth || DEPTH.RESOURCE, {\n userId: user.id,\n lockTokens: options.dav?.lockTokens\n })\n } else {\n // create lock if there is no webdav context\n const [ok, lock] = await this.filesLockManager.create(user, space.dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n fileLock = lock\n }\n try {\n // check range\n let startRange = 0\n if ((fExists || fTmpExists) && req.headers['content-range']) {\n // with PUT method, some webdav clients use the `content-range` header,\n // which is normally reserved for a response to a request containing the `range` header.\n // However, for more compatibility let's accept it\n const match = /\\d+/.exec(req.headers['content-range'])\n if (!match.length) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Content-range : header is malformed')\n }\n startRange = parseInt(match[0], 10)\n const size = await fileSize(options?.tmpPath || space.realPath)\n if (startRange !== size) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Content-range : start offset does not match the current file size')\n }\n }\n // todo: check file in db to update\n // todo : versioning here\n let checksum: string\n if (options?.checksumAlg) {\n checksum = await writeFromStreamAndChecksum(options?.tmpPath || space.realPath, req.raw, startRange, options.checksumAlg)\n } else {\n await writeFromStream(options?.tmpPath || space.realPath, req.raw, startRange)\n }\n if (options?.tmpPath) {\n try {\n // ensure parent path exists\n await makeDir(path.dirname(space.realPath), true)\n // move the uploaded file to destination\n await moveFiles(options.tmpPath, space.realPath, true)\n } catch (e) {\n // cleanup tmp file\n await removeFiles(options.tmpPath)\n this.logger.error(`${this.saveStream.name} - unable to move ${options.tmpPath} -> ${space.realPath} : ${e}`)\n throw new FileError(HttpStatus.INTERNAL_SERVER_ERROR, 'Unable to move tmp file to dst file')\n }\n }\n if (options?.checksumAlg) {\n return checksum\n }\n return fExists\n } finally {\n if (fileLock) {\n try {\n await this.filesLockManager.removeLock(fileLock.key)\n } catch (e) {\n this.logger.warn(`Failed to remove lock ${fileLock.key}: ${e}`)\n }\n }\n }\n }\n\n async saveMultipart(user: UserModel, space: SpaceEnv, req: FastifySpaceRequest) {\n /* Accepted methods:\n POST: create new resource\n PUT: create or update new resource (even if parent path does not exist)\n */\n const realParentPath = dirName(space.realPath)\n\n if (req.method === HTTP_METHOD.POST) {\n if (await isPathExists(space.realPath)) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Resource already exists')\n }\n if (!(await isPathExists(realParentPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Parent must exists')\n }\n if (!(await isPathIsDir(realParentPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Parent must be a directory')\n }\n }\n\n const basePath = realParentPath + path.sep\n\n for await (const part of req.files()) {\n // part.filename may contain a path like foo/bar.txt.\n const dstFile = path.resolve(basePath, part.filename)\n // prevent path traversal\n if (!dstFile.startsWith(basePath)) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Location is not allowed')\n }\n // make dir in space\n const dstDir = dirName(dstFile)\n if (!(await isPathExists(dstDir))) {\n await makeDir(dstDir, true)\n }\n // create lock\n const dbFile = { ...space.dbFile, path: path.join(dirName(space.dbFile.path), part.filename) }\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, DEPTH.RESOURCE)\n if (!ok) throw new LockConflict(fileLock, 'Conflicting lock')\n // do\n try {\n await pipeline(part.file, fs.createWriteStream(dstFile, { highWaterMark: DEFAULT_HIGH_WATER_MARK }))\n } finally {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n }\n\n async touch(user: UserModel, space: SpaceEnv, mtime: number, checkLocks = true): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (checkLocks) {\n await this.filesLockManager.checkConflicts(space.dbFile, DEPTH.RESOURCE, { userId: user.id })\n }\n // todo: update mtime in last files ( & in db file ?)\n await touchFile(space.realPath, mtime)\n }\n\n async mkFile(user: UserModel, space: SpaceEnv, overwrite = false, checkLocks = true, checkDocument = false): Promise<void> {\n checkFileName(space.realPath)\n if (!overwrite && (await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Resource already exists')\n }\n if (checkLocks) {\n await this.filesLockManager.checkConflicts(space.dbFile, DEPTH.RESOURCE, { userId: user.id })\n }\n // use sample documents when possible\n const fileExtension = path.extname(space.realPath)\n if (checkDocument && fileExtension !== '.txt' && Object.values(DOCUMENT_TYPE).indexOf(fileExtension) > -1) {\n const srcSample = path.join(__dirname, `${SAMPLE_PATH_WITHOUT_EXT}${fileExtension}`)\n return copyFiles(srcSample, space.realPath, false, false, false)\n } else {\n return createEmptyFile(space.realPath)\n }\n }\n\n async mkDir(user: UserModel, space: SpaceEnv, recursive = false, dav?: { depth: LOCK_DEPTH; lockTokens: string[] }): Promise<void> {\n checkFileName(space.realPath)\n if (!recursive) {\n if (await isPathExists(space.realPath)) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'Resource already exists')\n } else if (!(await isPathExists(dirName(space.realPath)))) {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n }\n await this.filesLockManager.checkConflicts(space.dbFile, dav?.depth || DEPTH.RESOURCE, { userId: user.id, lockTokens: dav?.lockTokens })\n await makeDir(space.realPath, recursive)\n }\n\n async copyMove(\n user: UserModel,\n srcSpace: SpaceEnv,\n dstSpace: SpaceEnv,\n isMove: boolean,\n overwrite = false,\n mkdirDstParentPath = false,\n dav?: { depth: LOCK_DEPTH; lockTokens: string[] }\n ): Promise<void> {\n // checks\n if (!canAccessToSpace(user, dstSpace)) {\n this.logger.warn(`${this.copyMove.name} - is not allowed to access to this space repository : ${dstSpace.repository}`)\n throw new FileError(HttpStatus.FORBIDDEN, 'You are not allowed to access to this repository')\n }\n if (!haveSpaceEnvPermissions(dstSpace, SPACE_OPERATION.ADD)) {\n this.logger.warn(`${this.copyMove.name} - is not allowed to copy/move on this space : *${dstSpace.alias}* (${dstSpace.id}) : ${dstSpace.url}`)\n throw new FileError(HttpStatus.FORBIDDEN, 'You are not allowed to copy/move on the destination')\n }\n if (dstSpace.quotaIsExceeded) {\n this.logger.warn(`${this.copyMove.name} - quota is exceeded for *${dstSpace.alias}* (${dstSpace.id})`)\n throw new FileError(HttpStatus.INSUFFICIENT_STORAGE, 'Quota is exceeded')\n }\n if (!(await isPathExists(srcSpace.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (!(await isPathExists(dirName(dstSpace.realPath)))) {\n if (mkdirDstParentPath) {\n try {\n await makeDir(dirName(dstSpace.realPath), true)\n } catch (e) {\n this.logger.error(`${this.copyMove.name} - Cannot create parent directory for destination ${dstSpace.realPath} : ${e}`)\n throw new FileError(HttpStatus.INTERNAL_SERVER_ERROR, 'Cannot create parent directory for destination')\n }\n } else {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n }\n if (srcSpace.realPath === dstSpace.realPath) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Cannot copy/move source onto itself')\n }\n if (`${dstSpace.realPath}/`.startsWith(`${srcSpace.realPath}/`)) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Cannot copy/move source below itself')\n }\n if (dirName(srcSpace.url) === dirName(dstSpace.url) && dirName(srcSpace.realPath) !== dirName(dstSpace.realPath)) {\n /* Handle renaming a space file with the same name as a space root :\n srcSpace.url = '/space/sync-in/code2.ts' (a space file)\n srcSpace.realPath = '/home/sync-in/spaces/sync-in/code2.ts\n dstSpace.url = '/space/sync-in/code.ts' (a space root)\n dstSpace.realPath = '/home/sync-in/users/jo/files/code2.ts !!\n */\n throw new FileError(HttpStatus.BAD_REQUEST, 'An anchored file already has this name')\n }\n if (!overwrite && (await isPathExists(dstSpace.realPath))) {\n /* Handle case-sensitive (in renaming context):\n srcSpace.url = '/space/sync-in/code.ts'\n dstSpace.url = '/space/sync-in/code.TS'\n The destination exists because it's the same file, bypass this\n */\n if (!(isMove && srcSpace.realPath.toLowerCase() === dstSpace.realPath.toLowerCase())) {\n throw new FileError(dav ? HttpStatus.PRECONDITION_FAILED : HttpStatus.BAD_REQUEST, 'The destination already exists')\n }\n }\n\n const isDir = await isPathIsDir(srcSpace.realPath)\n\n if (dstSpace.storageQuota) {\n /* Skip validation when moving to the same space; for copy operations, run all checks. */\n if (!isMove || (isMove && srcSpace.id !== dstSpace.id)) {\n const size = isDir ? (await dirSize(srcSpace.realPath))[0] : await fileSize(srcSpace.realPath)\n if (dstSpace.willExceedQuota(size)) {\n this.logger.warn(`${this.copyMove.name} - storage quota will be exceeded for *${dstSpace.alias}* (${dstSpace.id})`)\n throw new FileError(HttpStatus.INSUFFICIENT_STORAGE, 'Storage quota will be exceeded')\n }\n }\n }\n\n // check lock conflicts on source and destination\n let recursive: boolean\n let depth: LOCK_DEPTH\n if (dav?.depth) {\n recursive = dav.depth === DEPTH.INFINITY\n depth = dav.depth\n } else {\n recursive = isDir\n depth = recursive ? DEPTH.INFINITY : DEPTH.RESOURCE\n }\n if (isMove) {\n // check source\n await this.filesLockManager.checkConflicts(srcSpace.dbFile, depth, { userId: user.id, lockTokens: dav?.lockTokens })\n }\n // check destination\n await this.filesLockManager.checkConflicts(dstSpace.dbFile, depth, { userId: user.id, lockTokens: dav?.lockTokens })\n\n // overwrite\n if (overwrite && (await isPathExists(dstSpace.realPath))) {\n // todo : versioning here\n await this.delete(user, dstSpace)\n }\n\n // send it to task watcher\n if (srcSpace.task?.cacheKey) {\n if (!isDir) srcSpace.task.props.totalSize = await fileSize(srcSpace.realPath)\n FileTaskEvent.emit('startWatch', srcSpace, isMove ? FILE_OPERATION.MOVE : FILE_OPERATION.COPY, dstSpace.realPath)\n }\n\n // do\n if (isMove) {\n await moveFiles(srcSpace.realPath, dstSpace.realPath, overwrite)\n return this.filesQueries.moveFiles(srcSpace.dbFile, dstSpace.dbFile, isDir)\n }\n return copyFiles(srcSpace.realPath, dstSpace.realPath, overwrite, recursive)\n }\n\n async delete(user: UserModel, space: SpaceEnv, dav?: { lockTokens: string[] }): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n // check lock conflicts\n const isDir = await isPathIsDir(space.realPath)\n await this.filesLockManager.checkConflicts(space.dbFile, isDir ? DEPTH.INFINITY : DEPTH.RESOURCE, {\n userId: user.id,\n lockTokens: dav?.lockTokens\n })\n // file system deletion\n let forceDeleteInDB = false\n if (space.inTrashRepository) {\n await removeFiles(space.realPath)\n } else {\n const baseTrashPath = realTrashPathFromSpace(user, space)\n if (baseTrashPath) {\n const name = fileName(space.realPath)\n const trashDir = path.join(baseTrashPath, dirName(space.dbFile.path))\n const trashFile = path.join(trashDir, name)\n if (!(await isPathExists(trashDir))) {\n await makeDir(trashDir, true)\n }\n if (await isPathExists(trashFile)) {\n // if a resource already exists in the trash, rename it with the date\n const dstTrash = await uniqueDatedFilePath(trashFile)\n // move the resource on fs\n await moveFiles(trashFile, dstTrash.path)\n // move the resource in db\n const trashFileDB: FileDBProps = { ...space.dbFile, inTrash: true }\n const dstTrashFileDB: FileDBProps = { ...trashFileDB, path: path.join(dirName(trashFileDB.path), fileName(dstTrash.path)) }\n this.filesQueries\n .moveFiles(trashFileDB, dstTrashFileDB, dstTrash.isDir)\n .catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n await moveFiles(space.realPath, trashFile, true)\n } else {\n this.logger.log(`Unable to find trash path for space - *${space.alias}* (${space.id}) : delete permanently : ${space.realPath}`)\n // todo: define a default trash for external paths\n forceDeleteInDB = true\n await removeFiles(space.realPath)\n }\n }\n // remove locks, these locks have already been checked in the `checkConflicts` function\n if (isDir) {\n this.filesLockManager.removeChildLocks(user, space.dbFile).catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n for (const lock of await this.filesLockManager.getLocksByPath(space.dbFile)) {\n this.filesLockManager.removeLock(lock.key).catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n // delete or move to trash the files in db\n return this.filesQueries.deleteFiles(space.dbFile, isDir, forceDeleteInDB)\n }\n\n async downloadFromUrl(user: UserModel, space: SpaceEnv, url: string): Promise<void> {\n this.logger.log(`${this.downloadFromUrl.name} : ${url}`)\n // create lock\n const rPath = await uniqueFilePathFromDir(space.realPath)\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(space.realPath))\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(fileLock, 'Conflicting lock')\n }\n // tasking\n if (space.task.cacheKey) {\n let headRes: AxiosResponse\n\n try {\n headRes = await this.http.axiosRef({ method: HTTP_METHOD.HEAD, url: url, maxRedirects: 1 })\n } catch (e) {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n this.logger.error(`${this.downloadFromUrl.name} - ${url} : ${e}`)\n throw new FileError(HttpStatus.BAD_REQUEST, 'Unable to download file')\n }\n\n if (regExpPrivateIP.test(headRes.request.socket.remoteAddress)) {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n // prevent SSRF attack\n throw new FileError(HttpStatus.FORBIDDEN, 'Access to internal IP addresses is forbidden')\n }\n\n // attempt to retrieve the Content-Length header\n try {\n if ('content-length' in headRes.headers) {\n space.task.props.totalSize = parseInt(headRes.headers['content-length'], 10) || null\n }\n } catch (e) {\n this.logger.debug(`${this.downloadFromUrl.name} - content-length : ${e}`)\n }\n FileTaskEvent.emit('startWatch', space, FILE_OPERATION.DOWNLOAD, rPath)\n }\n // do\n try {\n const getRes = await this.http.axiosRef({ method: HTTP_METHOD.GET, url: url, responseType: 'stream', maxRedirects: 1 })\n if (regExpPrivateIP.test(getRes.request.socket.remoteAddress)) {\n // Prevent SSRF attacks and perform a DNS-rebinding check if a HEAD request has already been made\n throw new FileError(HttpStatus.FORBIDDEN, 'Access to internal IP addresses is forbidden')\n }\n await writeFromStream(rPath, getRes.data)\n } finally {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n\n async compress(user: UserModel, space: SpaceEnv, dto: CompressFileDto): Promise<void> {\n // This method is currently used only by files-methods.service, which handles input sanitization.\n // If it is used in other services in the future, make sure to refactor accordingly to sanitize inputs properly.\n const srcPath = dirName(space.realPath)\n // todo: a guest link tasksPath should be in specific directory (guest link has no home)\n const archiveExt = dto.name.endsWith(dto.extension) ? '' : `.${dto.extension}`\n const dstPath = await uniqueFilePathFromDir(path.join(dto.compressInDirectory ? srcPath : user.tasksPath, `${dto.name}${archiveExt}`))\n const archive: Archiver = archiver('tar', {\n gzip: dto.extension === tarGzExtension,\n gzipOptions: {\n level: 9\n }\n })\n // create lock\n let fileLock: FileLock\n if (dto.compressInDirectory) {\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(dstPath))\n const [ok, lock] = await this.filesLockManager.create(user, dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n fileLock = lock\n }\n if (space.task?.cacheKey) {\n space.task.props.compressInDirectory = dto.compressInDirectory\n FileTaskEvent.emit('startWatch', space, FILE_OPERATION.COMPRESS, dstPath)\n }\n // do\n try {\n archive.on('error', (error: ArchiverError) => {\n throw error\n })\n const dstStream = fs.createWriteStream(dstPath, { highWaterMark: DEFAULT_HIGH_WATER_MARK })\n archive.pipe(dstStream)\n for (const f of dto.files) {\n if (await isPathIsDir(f.path)) {\n archive.directory(f.path, dto.files.length > 1 ? fileName(f.path) : false)\n } else {\n archive.file(f.path, {\n name: f.rootAlias ? f.name : fileName(f.path)\n })\n }\n }\n await archive.finalize()\n } finally {\n if (fileLock) {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n }\n\n async decompress(user: UserModel, space: SpaceEnv): Promise<void> {\n // checks\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n const extension = path.extname(space.realPath)\n if (!COMPRESSION_EXTENSION.has(extension)) {\n throw new FileError(HttpStatus.BAD_REQUEST, `${extension} is not supported`)\n }\n // make destination folder\n const dstPath = await uniqueFilePathFromDir(path.join(dirName(space.realPath), path.basename(space.realPath, extension)))\n await makeDir(dstPath)\n // create lock\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(dstPath))\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, DEPTH.INFINITY)\n if (!ok) {\n throw new LockConflict(fileLock, 'Conflicting lock')\n }\n // tasking\n if (space.task?.cacheKey) FileTaskEvent.emit('startWatch', space, FILE_OPERATION.DECOMPRESS, dstPath)\n // do\n try {\n if (extension === '.zip') {\n await extractZip(space.realPath, dstPath)\n } else {\n await extractTar({\n file: space.realPath,\n cwd: dstPath,\n gzip: COMPRESSION_EXTENSION.get(extension) === tarGzExtension,\n preserveOwner: false\n })\n }\n } finally {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n\n async generateThumbnail(space: SpaceEnv, size: number): Promise<Buffer> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (getMimeType(space.realPath, false).indexOf('image') === -1) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'File is not an image')\n }\n try {\n return generateThumbnail(space.realPath, size)\n } catch (e) {\n this.logger.warn(e)\n throw new FileError(HttpStatus.BAD_REQUEST, 'File is not an image')\n }\n }\n}\n"],"names":["FilesManager","sendFileFromSpace","space","asAttachment","downloadName","SendFile","realPath","saveStream","user","req","options","fExists","isPathExists","fTmpExists","tmpPath","method","HTTP_METHOD","POST","FileError","HttpStatus","METHOD_NOT_ALLOWED","isPathIsDir","makeDir","dirName","CONFLICT","fileLock","dav","filesLockManager","checkConflicts","dbFile","depth","DEPTH","RESOURCE","userId","id","lockTokens","ok","lock","create","LockConflict","startRange","headers","match","exec","length","BAD_REQUEST","parseInt","size","fileSize","checksum","checksumAlg","writeFromStreamAndChecksum","raw","writeFromStream","path","dirname","moveFiles","e","removeFiles","logger","error","name","INTERNAL_SERVER_ERROR","removeLock","key","warn","saveMultipart","realParentPath","basePath","sep","part","files","dstFile","resolve","filename","startsWith","FORBIDDEN","dstDir","join","pipeline","file","fs","createWriteStream","highWaterMark","DEFAULT_HIGH_WATER_MARK","touch","mtime","checkLocks","NOT_FOUND","touchFile","mkFile","overwrite","checkDocument","checkFileName","fileExtension","extname","Object","values","DOCUMENT_TYPE","indexOf","srcSample","__dirname","SAMPLE_PATH_WITHOUT_EXT","copyFiles","createEmptyFile","mkDir","recursive","copyMove","srcSpace","dstSpace","isMove","mkdirDstParentPath","canAccessToSpace","repository","haveSpaceEnvPermissions","SPACE_OPERATION","ADD","alias","url","quotaIsExceeded","INSUFFICIENT_STORAGE","toLowerCase","PRECONDITION_FAILED","isDir","storageQuota","dirSize","willExceedQuota","INFINITY","delete","task","cacheKey","props","totalSize","FileTaskEvent","emit","FILE_OPERATION","MOVE","COPY","filesQueries","forceDeleteInDB","inTrashRepository","baseTrashPath","realTrashPathFromSpace","fileName","trashDir","trashFile","dstTrash","uniqueDatedFilePath","trashFileDB","inTrash","dstTrashFileDB","catch","log","removeChildLocks","getLocksByPath","deleteFiles","downloadFromUrl","rPath","uniqueFilePathFromDir","headRes","http","axiosRef","HEAD","maxRedirects","regExpPrivateIP","test","request","socket","remoteAddress","debug","DOWNLOAD","getRes","GET","responseType","data","compress","dto","srcPath","archiveExt","endsWith","extension","dstPath","compressInDirectory","tasksPath","archive","archiver","gzip","tarGzExtension","gzipOptions","level","COMPRESS","on","dstStream","pipe","f","directory","rootAlias","finalize","decompress","COMPRESSION_EXTENSION","has","basename","DECOMPRESS","extractZip","extractTar","cwd","get","preserveOwner","generateThumbnail","getMimeType","Logger"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-manager.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { HttpService } from '@nestjs/axios'\nimport { HttpStatus, Injectable, Logger } from '@nestjs/common'\nimport archiver, { Archiver, ArchiverError } from 'archiver'\nimport { AxiosResponse } from 'axios'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport { Readable } from 'node:stream'\nimport { pipeline } from 'node:stream/promises'\nimport { extract as extractTar } from 'tar'\nimport { FastifyAuthenticatedRequest } from '../../../authentication/interfaces/auth-request.interface'\nimport { generateThumbnail } from '../../../common/image'\nimport { HTTP_METHOD } from '../../applications.constants'\nimport { SPACE_OPERATION } from '../../spaces/constants/spaces'\nimport { FastifySpaceRequest } from '../../spaces/interfaces/space-request.interface'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { realTrashPathFromSpace } from '../../spaces/utils/paths'\nimport { canAccessToSpace, haveSpaceEnvPermissions } from '../../spaces/utils/permissions'\nimport { UserModel } from '../../users/models/user.model'\nimport { DEPTH, LOCK_DEPTH } from '../../webdav/constants/webdav'\nimport { tarGzExtension } from '../constants/compress'\nimport { COMPRESSION_EXTENSION, DEFAULT_HIGH_WATER_MARK } from '../constants/files'\nimport { FILE_OPERATION } from '../constants/operations'\nimport { DOCUMENT_TYPE, SAMPLE_PATH_WITHOUT_EXT } from '../constants/samples'\nimport { CompressFileDto } from '../dto/file-operations.dto'\nimport { FileTaskEvent } from '../events/file-task-event'\nimport { FileDBProps } from '../interfaces/file-db-props.interface'\nimport { FileLock } from '../interfaces/file-lock.interface'\nimport { FileError } from '../models/file-error'\nimport { LockConflict } from '../models/file-lock-error'\nimport {\n checkFileName,\n copyFiles,\n createEmptyFile,\n dirName,\n dirSize,\n fileName,\n fileSize,\n getMimeType,\n isPathExists,\n isPathIsDir,\n makeDir,\n moveFiles,\n removeFiles,\n touchFile,\n uniqueDatedFilePath,\n uniqueFilePathFromDir,\n writeFromStream,\n writeFromStreamAndChecksum\n} from '../utils/files'\nimport { SendFile } from '../utils/send-file'\nimport { extractZip } from '../utils/unzip-file'\nimport { regExpPrivateIP } from '../utils/url-file'\nimport { FilesLockManager } from './files-lock-manager.service'\nimport { FilesQueries } from './files-queries.service'\n\n@Injectable()\nexport class FilesManager {\n /* Spaces permissions are checked in the space guard, except for the copy/move destination */\n private logger = new Logger(FilesManager.name)\n\n constructor(\n private readonly http: HttpService,\n private readonly filesQueries: FilesQueries,\n private readonly filesLockManager: FilesLockManager\n ) {}\n\n sendFileFromSpace(space: SpaceEnv, asAttachment = false, downloadName = ''): SendFile {\n return new SendFile(space.realPath, asAttachment, downloadName)\n }\n\n async saveStream(\n user: UserModel,\n space: SpaceEnv,\n req: FastifyAuthenticatedRequest,\n options: {\n checksumAlg: string\n tmpPath?: string\n }\n ): Promise<string>\n async saveStream(user: UserModel, space: SpaceEnv, req: FastifyAuthenticatedRequest, options?: any): Promise<boolean>\n async saveStream(\n user: UserModel,\n space: SpaceEnv,\n req: FastifyAuthenticatedRequest,\n options?: { dav?: { depth: LOCK_DEPTH; lockTokens: string[] }; checksumAlg?: string; tmpPath?: string }\n ): Promise<boolean | string> {\n // if tmpPath is used, we lock the final destination during the transfer\n // space.realPath is replaced by tmpPath (if allowed), if the move operation failed we remove the tmp file\n const fExists = await isPathExists(space.realPath)\n const fTmpExists = options?.tmpPath ? await isPathExists(options.tmpPath) : false\n if (fExists && req.method === HTTP_METHOD.POST) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'Resource already exists')\n }\n if (fExists && (await isPathIsDir(space.realPath))) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'The location is a directory')\n }\n if (options?.tmpPath) {\n // ensure tmpPath parent dir exists\n await makeDir(dirName(options.tmpPath), true)\n } else if (!(await isPathExists(dirName(space.realPath)))) {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n /* File Lock */\n let fileLock: FileLock | undefined\n if (options?.dav) {\n // check locks\n await this.filesLockManager.checkConflicts(space.dbFile, options?.dav?.depth || DEPTH.RESOURCE, {\n userId: user.id,\n lockTokens: options.dav?.lockTokens\n })\n } else {\n // create lock if there is no webdav context\n const [ok, lock] = await this.filesLockManager.create(user, space.dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n fileLock = lock\n }\n try {\n // check range\n let startRange = 0\n if ((fExists || fTmpExists) && req.headers['content-range']) {\n // with PUT method, some webdav clients use the `content-range` header,\n // which is normally reserved for a response to a request containing the `range` header.\n // However, for more compatibility let's accept it\n const match = /\\d+/.exec(req.headers['content-range'])\n if (!match.length) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Content-range : header is malformed')\n }\n startRange = parseInt(match[0], 10)\n const size = await fileSize(options?.tmpPath || space.realPath)\n if (startRange !== size) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Content-range : start offset does not match the current file size')\n }\n }\n // todo: check file in db to update\n // todo : versioning here\n let checksum: string\n if (options?.checksumAlg) {\n checksum = await writeFromStreamAndChecksum(options?.tmpPath || space.realPath, req.raw, startRange, options.checksumAlg)\n } else {\n await writeFromStream(options?.tmpPath || space.realPath, req.raw, startRange)\n }\n if (options?.tmpPath) {\n try {\n // ensure parent path exists\n await makeDir(path.dirname(space.realPath), true)\n // move the uploaded file to destination\n await moveFiles(options.tmpPath, space.realPath, true)\n } catch (e) {\n // cleanup tmp file\n await removeFiles(options.tmpPath)\n this.logger.error(`${this.saveStream.name} - unable to move ${options.tmpPath} -> ${space.realPath} : ${e}`)\n throw new FileError(HttpStatus.INTERNAL_SERVER_ERROR, 'Unable to move tmp file to dst file')\n }\n }\n if (options?.checksumAlg) {\n return checksum\n }\n return fExists\n } finally {\n if (fileLock) {\n try {\n await this.filesLockManager.removeLock(fileLock.key)\n } catch (e) {\n this.logger.warn(`Failed to remove lock ${fileLock.key}: ${e}`)\n }\n }\n }\n }\n\n async saveMultipart(user: UserModel, space: SpaceEnv, req: FastifySpaceRequest) {\n /* Accepted methods:\n POST: create new resource\n PUT: create or update new resource (even if parent path does not exist)\n */\n const realParentPath = dirName(space.realPath)\n\n if (req.method === HTTP_METHOD.POST) {\n if (await isPathExists(space.realPath)) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Resource already exists')\n }\n if (!(await isPathExists(realParentPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Parent must exists')\n }\n if (!(await isPathIsDir(realParentPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Parent must be a directory')\n }\n }\n\n const basePath = realParentPath + path.sep\n\n for await (const part of req.files()) {\n // part.filename may contain a path like foo/bar.txt.\n const dstFile = path.resolve(basePath, part.filename)\n // prevent path traversal\n if (!dstFile.startsWith(basePath)) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Location is not allowed')\n }\n // make dir in space\n const dstDir = dirName(dstFile)\n if (!(await isPathExists(dstDir))) {\n await makeDir(dstDir, true)\n }\n // create lock\n const dbFile = { ...space.dbFile, path: path.join(dirName(space.dbFile.path), part.filename) }\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, DEPTH.RESOURCE)\n if (!ok) throw new LockConflict(fileLock, 'Conflicting lock')\n // do\n try {\n await pipeline(part.file, fs.createWriteStream(dstFile, { highWaterMark: DEFAULT_HIGH_WATER_MARK }))\n } finally {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n }\n\n async touch(user: UserModel, space: SpaceEnv, mtime: number, checkLocks = true): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (checkLocks) {\n await this.filesLockManager.checkConflicts(space.dbFile, DEPTH.RESOURCE, { userId: user.id })\n }\n // todo: update mtime in last files ( & in db file ?)\n await touchFile(space.realPath, mtime)\n }\n\n async mkFile(user: UserModel, space: SpaceEnv, overwrite = false, checkLocks = true, checkDocument = false): Promise<void> {\n checkFileName(space.realPath)\n if (!overwrite && (await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'Resource already exists')\n }\n if (checkLocks) {\n await this.filesLockManager.checkConflicts(space.dbFile, DEPTH.RESOURCE, { userId: user.id })\n }\n // use sample documents when possible\n const fileExtension = path.extname(space.realPath)\n if (checkDocument && fileExtension !== '.txt' && Object.values(DOCUMENT_TYPE).indexOf(fileExtension) > -1) {\n const srcSample = path.join(__dirname, `${SAMPLE_PATH_WITHOUT_EXT}${fileExtension}`)\n return copyFiles(srcSample, space.realPath, false, false, false)\n } else {\n return createEmptyFile(space.realPath)\n }\n }\n\n async mkDir(user: UserModel, space: SpaceEnv, recursive = false, dav?: { depth: LOCK_DEPTH; lockTokens: string[] }): Promise<void> {\n checkFileName(space.realPath)\n if (!recursive) {\n if (await isPathExists(space.realPath)) {\n throw new FileError(HttpStatus.METHOD_NOT_ALLOWED, 'Resource already exists')\n } else if (!(await isPathExists(dirName(space.realPath)))) {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n }\n await this.filesLockManager.checkConflicts(space.dbFile, dav?.depth || DEPTH.RESOURCE, { userId: user.id, lockTokens: dav?.lockTokens })\n await makeDir(space.realPath, recursive)\n }\n\n async copyMove(\n user: UserModel,\n srcSpace: SpaceEnv,\n dstSpace: SpaceEnv,\n isMove: boolean,\n overwrite = false,\n mkdirDstParentPath = false,\n dav?: { depth: LOCK_DEPTH; lockTokens: string[] }\n ): Promise<void> {\n // checks\n if (!canAccessToSpace(user, dstSpace)) {\n this.logger.warn(`${this.copyMove.name} - is not allowed to access to this space repository : ${dstSpace.repository}`)\n throw new FileError(HttpStatus.FORBIDDEN, 'You are not allowed to access to this repository')\n }\n if (!haveSpaceEnvPermissions(dstSpace, SPACE_OPERATION.ADD)) {\n this.logger.warn(`${this.copyMove.name} - is not allowed to copy/move on this space : *${dstSpace.alias}* (${dstSpace.id}) : ${dstSpace.url}`)\n throw new FileError(HttpStatus.FORBIDDEN, 'You are not allowed to copy/move on the destination')\n }\n if (dstSpace.quotaIsExceeded) {\n this.logger.warn(`${this.copyMove.name} - quota is exceeded for *${dstSpace.alias}* (${dstSpace.id})`)\n throw new FileError(HttpStatus.INSUFFICIENT_STORAGE, 'Quota is exceeded')\n }\n if (!(await isPathExists(srcSpace.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (!(await isPathExists(dirName(dstSpace.realPath)))) {\n if (mkdirDstParentPath) {\n try {\n await makeDir(dirName(dstSpace.realPath), true)\n } catch (e) {\n this.logger.error(`${this.copyMove.name} - Cannot create parent directory for destination ${dstSpace.realPath} : ${e}`)\n throw new FileError(HttpStatus.INTERNAL_SERVER_ERROR, 'Cannot create parent directory for destination')\n }\n } else {\n throw new FileError(HttpStatus.CONFLICT, 'Parent must exists')\n }\n }\n if (srcSpace.realPath === dstSpace.realPath) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Cannot copy/move source onto itself')\n }\n if (`${dstSpace.realPath}/`.startsWith(`${srcSpace.realPath}/`)) {\n throw new FileError(HttpStatus.FORBIDDEN, 'Cannot copy/move source below itself')\n }\n if (dirName(srcSpace.url) === dirName(dstSpace.url) && dirName(srcSpace.realPath) !== dirName(dstSpace.realPath)) {\n /* Handle renaming a space file with the same name as a space root :\n srcSpace.url = '/space/sync-in/code2.ts' (a space file)\n srcSpace.realPath = '/home/sync-in/spaces/sync-in/code2.ts\n dstSpace.url = '/space/sync-in/code.ts' (a space root)\n dstSpace.realPath = '/home/sync-in/users/jo/files/code2.ts !!\n */\n throw new FileError(HttpStatus.BAD_REQUEST, 'An anchored file already has this name')\n }\n if (!overwrite && (await isPathExists(dstSpace.realPath))) {\n /* Handle case-sensitive (in renaming context):\n srcSpace.url = '/space/sync-in/code.ts'\n dstSpace.url = '/space/sync-in/code.TS'\n The destination exists because it's the same file, bypass this\n */\n if (!(isMove && srcSpace.realPath.toLowerCase() === dstSpace.realPath.toLowerCase())) {\n throw new FileError(dav ? HttpStatus.PRECONDITION_FAILED : HttpStatus.BAD_REQUEST, 'The destination already exists')\n }\n }\n\n const isDir = await isPathIsDir(srcSpace.realPath)\n\n if (dstSpace.storageQuota) {\n /* Skip validation when moving to the same space; for copy operations, run all checks. */\n if (!isMove || (isMove && srcSpace.id !== dstSpace.id)) {\n const size = isDir ? (await dirSize(srcSpace.realPath))[0] : await fileSize(srcSpace.realPath)\n if (dstSpace.willExceedQuota(size)) {\n this.logger.warn(`${this.copyMove.name} - storage quota will be exceeded for *${dstSpace.alias}* (${dstSpace.id})`)\n throw new FileError(HttpStatus.INSUFFICIENT_STORAGE, 'Storage quota will be exceeded')\n }\n }\n }\n\n // check lock conflicts on source and destination\n let recursive: boolean\n let depth: LOCK_DEPTH\n if (dav?.depth) {\n recursive = dav.depth === DEPTH.INFINITY\n depth = dav.depth\n } else {\n recursive = isDir\n depth = recursive ? DEPTH.INFINITY : DEPTH.RESOURCE\n }\n if (isMove) {\n // check source\n await this.filesLockManager.checkConflicts(srcSpace.dbFile, depth, { userId: user.id, lockTokens: dav?.lockTokens })\n }\n // check destination\n await this.filesLockManager.checkConflicts(dstSpace.dbFile, depth, { userId: user.id, lockTokens: dav?.lockTokens })\n\n // overwrite\n if (overwrite && (await isPathExists(dstSpace.realPath))) {\n // todo : versioning here\n await this.delete(user, dstSpace)\n }\n\n // send it to task watcher\n if (srcSpace.task?.cacheKey) {\n if (!isDir) srcSpace.task.props.totalSize = await fileSize(srcSpace.realPath)\n FileTaskEvent.emit('startWatch', srcSpace, isMove ? FILE_OPERATION.MOVE : FILE_OPERATION.COPY, dstSpace.realPath)\n }\n\n // do\n if (isMove) {\n await moveFiles(srcSpace.realPath, dstSpace.realPath, overwrite)\n return this.filesQueries.moveFiles(srcSpace.dbFile, dstSpace.dbFile, isDir)\n }\n return copyFiles(srcSpace.realPath, dstSpace.realPath, overwrite, recursive)\n }\n\n async delete(user: UserModel, space: SpaceEnv, dav?: { lockTokens: string[] }): Promise<void> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n // check lock conflicts\n const isDir = await isPathIsDir(space.realPath)\n await this.filesLockManager.checkConflicts(space.dbFile, isDir ? DEPTH.INFINITY : DEPTH.RESOURCE, {\n userId: user.id,\n lockTokens: dav?.lockTokens\n })\n // file system deletion\n let forceDeleteInDB = false\n if (space.inTrashRepository) {\n await removeFiles(space.realPath)\n } else {\n const baseTrashPath = realTrashPathFromSpace(user, space)\n if (baseTrashPath) {\n const name = fileName(space.realPath)\n const trashDir = path.join(baseTrashPath, dirName(space.dbFile.path))\n const trashFile = path.join(trashDir, name)\n if (!(await isPathExists(trashDir))) {\n await makeDir(trashDir, true)\n }\n if (await isPathExists(trashFile)) {\n // if a resource already exists in the trash, rename it with the date\n const dstTrash = await uniqueDatedFilePath(trashFile)\n // move the resource on fs\n await moveFiles(trashFile, dstTrash.path)\n // move the resource in db\n const trashFileDB: FileDBProps = { ...space.dbFile, inTrash: true }\n const dstTrashFileDB: FileDBProps = { ...trashFileDB, path: path.join(dirName(trashFileDB.path), fileName(dstTrash.path)) }\n this.filesQueries\n .moveFiles(trashFileDB, dstTrashFileDB, dstTrash.isDir)\n .catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n await moveFiles(space.realPath, trashFile, true)\n } else {\n this.logger.log(`Unable to find trash path for space - *${space.alias}* (${space.id}) : delete permanently : ${space.realPath}`)\n // todo: define a default trash for external paths\n forceDeleteInDB = true\n await removeFiles(space.realPath)\n }\n }\n // remove locks, these locks have already been checked in the `checkConflicts` function\n if (isDir) {\n this.filesLockManager.removeChildLocks(user, space.dbFile).catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n for (const lock of await this.filesLockManager.getLocksByPath(space.dbFile)) {\n this.filesLockManager.removeLock(lock.key).catch((e: Error) => this.logger.error(`${this.delete.name} - ${e}`))\n }\n // delete or move to trash the files in db\n return this.filesQueries.deleteFiles(space.dbFile, isDir, forceDeleteInDB)\n }\n\n async downloadFromUrl(user: UserModel, space: SpaceEnv, url: string): Promise<void> {\n this.logger.log(`${this.downloadFromUrl.name} : ${url}`)\n // create lock\n const rPath = await uniqueFilePathFromDir(space.realPath)\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(space.realPath))\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(fileLock, 'Conflicting lock')\n }\n // tasking\n if (space.task.cacheKey) {\n let headRes: AxiosResponse\n\n try {\n headRes = await this.http.axiosRef({ method: HTTP_METHOD.HEAD, url: url, maxRedirects: 1 })\n } catch (e) {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n this.logger.error(`${this.downloadFromUrl.name} - ${url} : ${e}`)\n throw new FileError(HttpStatus.BAD_REQUEST, 'Unable to download file')\n }\n\n if (regExpPrivateIP.test(headRes.request.socket.remoteAddress)) {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n // prevent SSRF attack\n throw new FileError(HttpStatus.FORBIDDEN, 'Access to internal IP addresses is forbidden')\n }\n\n // attempt to retrieve the Content-Length header\n try {\n if ('content-length' in headRes.headers) {\n space.task.props.totalSize = parseInt(headRes.headers['content-length'], 10) || null\n }\n } catch (e) {\n this.logger.debug(`${this.downloadFromUrl.name} - content-length : ${e}`)\n }\n FileTaskEvent.emit('startWatch', space, FILE_OPERATION.DOWNLOAD, rPath)\n }\n // do\n try {\n const getRes = await this.http.axiosRef({ method: HTTP_METHOD.GET, url: url, responseType: 'stream', maxRedirects: 1 })\n if (regExpPrivateIP.test(getRes.request.socket.remoteAddress)) {\n // Prevent SSRF attacks and perform a DNS-rebinding check if a HEAD request has already been made\n throw new FileError(HttpStatus.FORBIDDEN, 'Access to internal IP addresses is forbidden')\n }\n await writeFromStream(rPath, getRes.data)\n } finally {\n // release lock\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n\n async compress(user: UserModel, space: SpaceEnv, dto: CompressFileDto): Promise<void> {\n // This method is currently used only by files-methods.service, which handles input sanitization.\n // If it is used in other services in the future, make sure to refactor accordingly to sanitize inputs properly.\n const srcPath = dirName(space.realPath)\n // todo: a guest link tasksPath should be in specific directory (guest link has no home)\n const archiveExt = dto.name.endsWith(dto.extension) ? '' : `.${dto.extension}`\n const dstPath = await uniqueFilePathFromDir(path.join(dto.compressInDirectory ? srcPath : user.tasksPath, `${dto.name}${archiveExt}`))\n const archive: Archiver = archiver('tar', {\n gzip: dto.extension === tarGzExtension,\n gzipOptions: {\n level: 9\n }\n })\n // create lock\n let fileLock: FileLock\n if (dto.compressInDirectory) {\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(dstPath))\n const [ok, lock] = await this.filesLockManager.create(user, dbFile, DEPTH.RESOURCE)\n if (!ok) {\n throw new LockConflict(lock, 'Conflicting lock')\n }\n fileLock = lock\n }\n if (space.task?.cacheKey) {\n space.task.props.compressInDirectory = dto.compressInDirectory\n FileTaskEvent.emit('startWatch', space, FILE_OPERATION.COMPRESS, dstPath)\n }\n // do\n try {\n archive.on('error', (error: ArchiverError) => {\n throw error\n })\n const dstStream = fs.createWriteStream(dstPath, { highWaterMark: DEFAULT_HIGH_WATER_MARK })\n archive.pipe(dstStream)\n for (const f of dto.files) {\n if (await isPathIsDir(f.path)) {\n archive.directory(f.path, dto.files.length > 1 ? fileName(f.path) : false)\n } else {\n archive.file(f.path, {\n name: f.rootAlias ? f.name : fileName(f.path)\n })\n }\n }\n await archive.finalize()\n } finally {\n if (fileLock) {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n }\n\n async decompress(user: UserModel, space: SpaceEnv): Promise<void> {\n // checks\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n const extension = path.extname(space.realPath)\n if (!COMPRESSION_EXTENSION.has(extension)) {\n throw new FileError(HttpStatus.BAD_REQUEST, `${extension} is not supported`)\n }\n // make destination folder\n const dstPath = await uniqueFilePathFromDir(path.join(dirName(space.realPath), path.basename(space.realPath, extension)))\n await makeDir(dstPath)\n // create lock\n const dbFile = space.dbFile\n dbFile.path = path.join(dirName(dbFile.path), fileName(dstPath))\n const [ok, fileLock] = await this.filesLockManager.create(user, dbFile, DEPTH.INFINITY)\n if (!ok) {\n throw new LockConflict(fileLock, 'Conflicting lock')\n }\n // tasking\n if (space.task?.cacheKey) FileTaskEvent.emit('startWatch', space, FILE_OPERATION.DECOMPRESS, dstPath)\n // do\n try {\n if (extension === '.zip') {\n await extractZip(space.realPath, dstPath)\n } else {\n await extractTar({\n file: space.realPath,\n cwd: dstPath,\n gzip: COMPRESSION_EXTENSION.get(extension) === tarGzExtension,\n preserveOwner: false\n })\n }\n } finally {\n await this.filesLockManager.removeLock(fileLock.key)\n }\n }\n\n async generateThumbnail(space: SpaceEnv, size: number): Promise<Readable> {\n if (!(await isPathExists(space.realPath))) {\n throw new FileError(HttpStatus.NOT_FOUND, 'Location not found')\n }\n if (getMimeType(space.realPath, false).indexOf('image') === -1) {\n throw new FileError(HttpStatus.BAD_REQUEST, 'File is not an image')\n }\n try {\n return generateThumbnail(space.realPath, size)\n } catch (e) {\n this.logger.warn(e)\n throw new FileError(HttpStatus.BAD_REQUEST, 'File is not an image')\n }\n }\n}\n"],"names":["FilesManager","sendFileFromSpace","space","asAttachment","downloadName","SendFile","realPath","saveStream","user","req","options","fExists","isPathExists","fTmpExists","tmpPath","method","HTTP_METHOD","POST","FileError","HttpStatus","METHOD_NOT_ALLOWED","isPathIsDir","makeDir","dirName","CONFLICT","fileLock","dav","filesLockManager","checkConflicts","dbFile","depth","DEPTH","RESOURCE","userId","id","lockTokens","ok","lock","create","LockConflict","startRange","headers","match","exec","length","BAD_REQUEST","parseInt","size","fileSize","checksum","checksumAlg","writeFromStreamAndChecksum","raw","writeFromStream","path","dirname","moveFiles","e","removeFiles","logger","error","name","INTERNAL_SERVER_ERROR","removeLock","key","warn","saveMultipart","realParentPath","basePath","sep","part","files","dstFile","resolve","filename","startsWith","FORBIDDEN","dstDir","join","pipeline","file","fs","createWriteStream","highWaterMark","DEFAULT_HIGH_WATER_MARK","touch","mtime","checkLocks","NOT_FOUND","touchFile","mkFile","overwrite","checkDocument","checkFileName","fileExtension","extname","Object","values","DOCUMENT_TYPE","indexOf","srcSample","__dirname","SAMPLE_PATH_WITHOUT_EXT","copyFiles","createEmptyFile","mkDir","recursive","copyMove","srcSpace","dstSpace","isMove","mkdirDstParentPath","canAccessToSpace","repository","haveSpaceEnvPermissions","SPACE_OPERATION","ADD","alias","url","quotaIsExceeded","INSUFFICIENT_STORAGE","toLowerCase","PRECONDITION_FAILED","isDir","storageQuota","dirSize","willExceedQuota","INFINITY","delete","task","cacheKey","props","totalSize","FileTaskEvent","emit","FILE_OPERATION","MOVE","COPY","filesQueries","forceDeleteInDB","inTrashRepository","baseTrashPath","realTrashPathFromSpace","fileName","trashDir","trashFile","dstTrash","uniqueDatedFilePath","trashFileDB","inTrash","dstTrashFileDB","catch","log","removeChildLocks","getLocksByPath","deleteFiles","downloadFromUrl","rPath","uniqueFilePathFromDir","headRes","http","axiosRef","HEAD","maxRedirects","regExpPrivateIP","test","request","socket","remoteAddress","debug","DOWNLOAD","getRes","GET","responseType","data","compress","dto","srcPath","archiveExt","endsWith","extension","dstPath","compressInDirectory","tasksPath","archive","archiver","gzip","tarGzExtension","gzipOptions","level","COMPRESS","on","dstStream","pipe","f","directory","rootAlias","finalize","decompress","COMPRESSION_EXTENSION","has","basename","DECOMPRESS","extractZip","extractTar","cwd","get","preserveOwner","generateThumbnail","getMimeType","Logger"],"mappings":"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"}
@@ -14,8 +14,6 @@ Object.defineProperty(exports, "FilesMethods", {
14
14
  });
15
15
  const _common = require("@nestjs/common");
16
16
  const _nodepath = /*#__PURE__*/ _interop_require_default(require("node:path"));
17
- const _nodestream = require("node:stream");
18
- const _image = require("../../../common/image");
19
17
  const _spacesmanagerservice = require("../../spaces/services/spaces-manager.service");
20
18
  const _operations = require("../constants/operations");
21
19
  const _fileerror = require("../models/file-error");
@@ -132,10 +130,7 @@ let FilesMethods = class FilesMethods {
132
130
  }
133
131
  async genThumbnail(space, size) {
134
132
  try {
135
- const pngStream = _nodestream.Readable.from(await this.filesManager.generateThumbnail(space, size));
136
- return new _common.StreamableFile(pngStream, {
137
- type: _image.pngMimeType
138
- });
133
+ return await this.filesManager.generateThumbnail(space, size);
139
134
  } catch (e) {
140
135
  this.handleError(space, this.genThumbnail.name, e);
141
136
  }
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-methods.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { HttpException, HttpStatus, Injectable, Logger, StreamableFile } from '@nestjs/common'\nimport { FastifyReply } from 'fastify'\nimport path from 'node:path'\nimport { Readable } from 'node:stream'\nimport { pngMimeType } from '../../../common/image'\nimport { FastifySpaceRequest } from '../../spaces/interfaces/space-request.interface'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpacesManager } from '../../spaces/services/spaces-manager.service'\nimport { UserModel } from '../../users/models/user.model'\nimport { FILE_OPERATION } from '../constants/operations'\nimport { CompressFileDto, CopyMoveFileDto, DownloadFileDto, MakeFileDto } from '../dto/file-operations.dto'\nimport { FileError } from '../models/file-error'\nimport { LockConflict } from '../models/file-lock-error'\nimport { checkFileName, dirName, fileName, isPathExists, sanitizeName } from '../utils/files'\nimport { SendFile } from '../utils/send-file'\nimport { FilesManager } from './files-manager.service'\n\n@Injectable()\nexport class FilesMethods {\n private readonly logger = new Logger(FilesMethods.name)\n\n constructor(\n private readonly spacesManager: SpacesManager,\n private readonly filesManager: FilesManager\n ) {}\n\n async headOrGet(req: FastifySpaceRequest, res: FastifyReply): Promise<StreamableFile> {\n const sendFile: SendFile = this.filesManager.sendFileFromSpace(req.space)\n try {\n await sendFile.checks()\n return await sendFile.stream(req, res)\n } catch (e) {\n this.handleError(req.space, req.method, e)\n }\n }\n\n async upload(req: FastifySpaceRequest, res: FastifyReply): Promise<void> {\n try {\n await this.filesManager.saveMultipart(req.user, req.space, req)\n } catch (e) {\n // if error we need to close the stream\n // req.raw.destroy()\n this.logger.error(`${this.upload.name} - unable to ${FILE_OPERATION.UPLOAD} ${req.space.url} : ${e}`)\n return res\n .header('Connection', 'close')\n .status(e.httpCode || 500)\n .send({ message: e.message })\n }\n }\n\n async make(user: UserModel, space: SpaceEnv, makeFileDto: MakeFileDto): Promise<void> {\n try {\n if (makeFileDto.type === 'directory') {\n return await this.filesManager.mkDir(user, space)\n } else {\n return await this.filesManager.mkFile(user, space, false, true, true)\n }\n } catch (e) {\n this.handleError(space, `${FILE_OPERATION.MAKE} ${makeFileDto.type}`, e)\n }\n }\n\n copy(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto\n ): Promise<{\n path: string\n name: string\n }> {\n return this.copyMove(user, space, copyMoveFileDto, false)\n }\n\n move(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto\n ): Promise<{\n path: string\n name: string\n }> {\n return this.copyMove(user, space, copyMoveFileDto, true)\n }\n\n async delete(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.delete(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DELETE, e)\n }\n }\n\n async downloadFromUrl(user: UserModel, space: SpaceEnv, downloadDto: DownloadFileDto): Promise<void> {\n checkFileName(space.realPath)\n try {\n return await this.filesManager.downloadFromUrl(user, space, downloadDto.url)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DOWNLOAD, e)\n }\n }\n\n async compress(user: UserModel, space: SpaceEnv, compressFileDto: CompressFileDto): Promise<void> {\n compressFileDto.name = checkFileName(space.realPath)\n try {\n for (const f of compressFileDto.files) {\n // sanitize file name\n f.name = sanitizeName(f.name)\n // handles the case where the file is an anchored file\n let baseSpace: SpaceEnv\n if (f.path) {\n baseSpace = await this.spacesManager.spaceEnv(user, f.path.split('/'))\n f.path = baseSpace.realPath\n } else {\n if (f.rootAlias) {\n baseSpace = await this.spacesManager.spaceEnv(user, path.join(dirName(space.url), f.rootAlias).split('/'))\n f.path = baseSpace.realPath\n } else {\n baseSpace = space\n f.path = path.resolve(dirName(space.realPath), f.name)\n }\n }\n // prevent path traversal\n if (!f.path.startsWith(baseSpace.realBasePath)) {\n return this.handleError(space, FILE_OPERATION.COMPRESS, new FileError(HttpStatus.FORBIDDEN, `${f.name} not allowed`))\n }\n if (!(await isPathExists(f.path))) {\n return this.handleError(space, FILE_OPERATION.COMPRESS, new FileError(HttpStatus.NOT_FOUND, `${f.name} does not exist`))\n }\n }\n return await this.filesManager.compress(user, space, compressFileDto)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.COMPRESS, e)\n }\n }\n\n async decompress(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.decompress(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DECOMPRESS, e)\n }\n }\n\n async genThumbnail(space: SpaceEnv, size: number): Promise<StreamableFile> {\n try {\n const pngStream = Readable.from(await this.filesManager.generateThumbnail(space, size))\n return new StreamableFile(pngStream, { type: pngMimeType })\n } catch (e) {\n this.handleError(space, this.genThumbnail.name, e)\n }\n }\n\n private async copyMove(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto,\n isMove: boolean\n ): Promise<{\n path: string\n name: string\n }> {\n const dstUrl = path.join(copyMoveFileDto.dstDirectory, copyMoveFileDto.dstName ? copyMoveFileDto.dstName : fileName(space.realPath))\n let dstSpace: SpaceEnv\n try {\n dstSpace = await this.spacesManager.spaceEnv(user, dstUrl.split('/'))\n await this.filesManager.copyMove(user, space, dstSpace, isMove)\n } catch (e) {\n this.handleError(space, isMove ? FILE_OPERATION.MOVE : FILE_OPERATION.COPY, e, dstSpace)\n }\n return { path: dirName(dstUrl), name: fileName(dstUrl) }\n }\n\n private handleError(space: SpaceEnv, action: string, e: any, dstSpace?: SpaceEnv) {\n this.logger.error(`unable to ${action} ${space.url}${dstSpace?.url ? ` -> ${dstSpace.url}` : ''} : ${e}`)\n const errorMsg = e.message.split(',')[0]\n if (e instanceof LockConflict) {\n throw new HttpException('The file is locked', HttpStatus.LOCKED)\n } else if (e instanceof FileError) {\n throw new HttpException(errorMsg, e.httpCode)\n }\n throw new HttpException(errorMsg, HttpStatus.INTERNAL_SERVER_ERROR)\n }\n}\n"],"names":["FilesMethods","headOrGet","req","res","sendFile","filesManager","sendFileFromSpace","space","checks","stream","e","handleError","method","upload","saveMultipart","user","logger","error","name","FILE_OPERATION","UPLOAD","url","header","status","httpCode","send","message","make","makeFileDto","type","mkDir","mkFile","MAKE","copy","copyMoveFileDto","copyMove","move","delete","DELETE","downloadFromUrl","downloadDto","checkFileName","realPath","DOWNLOAD","compress","compressFileDto","f","files","sanitizeName","baseSpace","path","spacesManager","spaceEnv","split","rootAlias","join","dirName","resolve","startsWith","realBasePath","COMPRESS","FileError","HttpStatus","FORBIDDEN","isPathExists","NOT_FOUND","decompress","DECOMPRESS","genThumbnail","size","pngStream","Readable","from","generateThumbnail","StreamableFile","pngMimeType","isMove","dstUrl","dstDirectory","dstName","fileName","dstSpace","MOVE","COPY","action","errorMsg","LockConflict","HttpException","LOCKED","INTERNAL_SERVER_ERROR","Logger"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../../../backend/src/applications/files/services/files-methods.service.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\n\nimport { HttpException, HttpStatus, Injectable, Logger, StreamableFile } from '@nestjs/common'\nimport { FastifyReply } from 'fastify'\nimport path from 'node:path'\nimport { Readable } from 'node:stream'\nimport { FastifySpaceRequest } from '../../spaces/interfaces/space-request.interface'\nimport { SpaceEnv } from '../../spaces/models/space-env.model'\nimport { SpacesManager } from '../../spaces/services/spaces-manager.service'\nimport { UserModel } from '../../users/models/user.model'\nimport { FILE_OPERATION } from '../constants/operations'\nimport { CompressFileDto, CopyMoveFileDto, DownloadFileDto, MakeFileDto } from '../dto/file-operations.dto'\nimport { FileError } from '../models/file-error'\nimport { LockConflict } from '../models/file-lock-error'\nimport { checkFileName, dirName, fileName, isPathExists, sanitizeName } from '../utils/files'\nimport { SendFile } from '../utils/send-file'\nimport { FilesManager } from './files-manager.service'\n\n@Injectable()\nexport class FilesMethods {\n private readonly logger = new Logger(FilesMethods.name)\n\n constructor(\n private readonly spacesManager: SpacesManager,\n private readonly filesManager: FilesManager\n ) {}\n\n async headOrGet(req: FastifySpaceRequest, res: FastifyReply): Promise<StreamableFile> {\n const sendFile: SendFile = this.filesManager.sendFileFromSpace(req.space)\n try {\n await sendFile.checks()\n return await sendFile.stream(req, res)\n } catch (e) {\n this.handleError(req.space, req.method, e)\n }\n }\n\n async upload(req: FastifySpaceRequest, res: FastifyReply): Promise<void> {\n try {\n await this.filesManager.saveMultipart(req.user, req.space, req)\n } catch (e) {\n // if error we need to close the stream\n // req.raw.destroy()\n this.logger.error(`${this.upload.name} - unable to ${FILE_OPERATION.UPLOAD} ${req.space.url} : ${e}`)\n return res\n .header('Connection', 'close')\n .status(e.httpCode || 500)\n .send({ message: e.message })\n }\n }\n\n async make(user: UserModel, space: SpaceEnv, makeFileDto: MakeFileDto): Promise<void> {\n try {\n if (makeFileDto.type === 'directory') {\n return await this.filesManager.mkDir(user, space)\n } else {\n return await this.filesManager.mkFile(user, space, false, true, true)\n }\n } catch (e) {\n this.handleError(space, `${FILE_OPERATION.MAKE} ${makeFileDto.type}`, e)\n }\n }\n\n copy(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto\n ): Promise<{\n path: string\n name: string\n }> {\n return this.copyMove(user, space, copyMoveFileDto, false)\n }\n\n move(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto\n ): Promise<{\n path: string\n name: string\n }> {\n return this.copyMove(user, space, copyMoveFileDto, true)\n }\n\n async delete(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.delete(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DELETE, e)\n }\n }\n\n async downloadFromUrl(user: UserModel, space: SpaceEnv, downloadDto: DownloadFileDto): Promise<void> {\n checkFileName(space.realPath)\n try {\n return await this.filesManager.downloadFromUrl(user, space, downloadDto.url)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DOWNLOAD, e)\n }\n }\n\n async compress(user: UserModel, space: SpaceEnv, compressFileDto: CompressFileDto): Promise<void> {\n compressFileDto.name = checkFileName(space.realPath)\n try {\n for (const f of compressFileDto.files) {\n // sanitize file name\n f.name = sanitizeName(f.name)\n // handles the case where the file is an anchored file\n let baseSpace: SpaceEnv\n if (f.path) {\n baseSpace = await this.spacesManager.spaceEnv(user, f.path.split('/'))\n f.path = baseSpace.realPath\n } else {\n if (f.rootAlias) {\n baseSpace = await this.spacesManager.spaceEnv(user, path.join(dirName(space.url), f.rootAlias).split('/'))\n f.path = baseSpace.realPath\n } else {\n baseSpace = space\n f.path = path.resolve(dirName(space.realPath), f.name)\n }\n }\n // prevent path traversal\n if (!f.path.startsWith(baseSpace.realBasePath)) {\n return this.handleError(space, FILE_OPERATION.COMPRESS, new FileError(HttpStatus.FORBIDDEN, `${f.name} not allowed`))\n }\n if (!(await isPathExists(f.path))) {\n return this.handleError(space, FILE_OPERATION.COMPRESS, new FileError(HttpStatus.NOT_FOUND, `${f.name} does not exist`))\n }\n }\n return await this.filesManager.compress(user, space, compressFileDto)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.COMPRESS, e)\n }\n }\n\n async decompress(user: UserModel, space: SpaceEnv): Promise<void> {\n try {\n return await this.filesManager.decompress(user, space)\n } catch (e) {\n this.handleError(space, FILE_OPERATION.DECOMPRESS, e)\n }\n }\n\n async genThumbnail(space: SpaceEnv, size: number): Promise<Readable> {\n try {\n return await this.filesManager.generateThumbnail(space, size)\n } catch (e) {\n this.handleError(space, this.genThumbnail.name, e)\n }\n }\n\n private async copyMove(\n user: UserModel,\n space: SpaceEnv,\n copyMoveFileDto: CopyMoveFileDto,\n isMove: boolean\n ): Promise<{\n path: string\n name: string\n }> {\n const dstUrl = path.join(copyMoveFileDto.dstDirectory, copyMoveFileDto.dstName ? copyMoveFileDto.dstName : fileName(space.realPath))\n let dstSpace: SpaceEnv\n try {\n dstSpace = await this.spacesManager.spaceEnv(user, dstUrl.split('/'))\n await this.filesManager.copyMove(user, space, dstSpace, isMove)\n } catch (e) {\n this.handleError(space, isMove ? FILE_OPERATION.MOVE : FILE_OPERATION.COPY, e, dstSpace)\n }\n return { path: dirName(dstUrl), name: fileName(dstUrl) }\n }\n\n private handleError(space: SpaceEnv, action: string, e: any, dstSpace?: SpaceEnv) {\n this.logger.error(`unable to ${action} ${space.url}${dstSpace?.url ? ` -> ${dstSpace.url}` : ''} : ${e}`)\n const errorMsg = e.message.split(',')[0]\n if (e instanceof LockConflict) {\n throw new HttpException('The file is locked', HttpStatus.LOCKED)\n } else if (e instanceof FileError) {\n throw new HttpException(errorMsg, e.httpCode)\n }\n throw new HttpException(errorMsg, HttpStatus.INTERNAL_SERVER_ERROR)\n }\n}\n"],"names":["FilesMethods","headOrGet","req","res","sendFile","filesManager","sendFileFromSpace","space","checks","stream","e","handleError","method","upload","saveMultipart","user","logger","error","name","FILE_OPERATION","UPLOAD","url","header","status","httpCode","send","message","make","makeFileDto","type","mkDir","mkFile","MAKE","copy","copyMoveFileDto","copyMove","move","delete","DELETE","downloadFromUrl","downloadDto","checkFileName","realPath","DOWNLOAD","compress","compressFileDto","f","files","sanitizeName","baseSpace","path","spacesManager","spaceEnv","split","rootAlias","join","dirName","resolve","startsWith","realBasePath","COMPRESS","FileError","HttpStatus","FORBIDDEN","isPathExists","NOT_FOUND","decompress","DECOMPRESS","genThumbnail","size","generateThumbnail","isMove","dstUrl","dstDirectory","dstName","fileName","dstSpace","MOVE","COPY","action","errorMsg","LockConflict","HttpException","LOCKED","INTERNAL_SERVER_ERROR","Logger"],"mappings":"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"}
@@ -12,18 +12,8 @@ Object.defineProperty(exports, "parsePdf", {
12
12
  return parsePdf;
13
13
  }
14
14
  });
15
- const _nodepath = /*#__PURE__*/ _interop_require_default(require("node:path"));
16
- const _pdf = require("pdfjs-dist/legacy/build/pdf.mjs");
17
- function _interop_require_default(obj) {
18
- return obj && obj.__esModule ? obj : {
19
- default: obj
20
- };
21
- }
22
- // Enable parallel PDF parsing via Node.js worker threads
23
- _pdf.GlobalWorkerOptions.workerSrc = require.resolve('pdfjs-dist/legacy/build/pdf.worker.mjs');
24
- const pdfjsDistDir = _nodepath.default.dirname(require.resolve('pdfjs-dist/package.json'));
25
- const standardFontsDir = _nodepath.default.join(pdfjsDistDir, 'standard_fonts') + '/';
26
- // Type guard to filter true text items
15
+ const _promises = require("node:fs/promises");
16
+ const _unpdf = require("unpdf");
27
17
  function isTextItem(item) {
28
18
  return typeof item.str === 'string' && Array.isArray(item.transform);
29
19
  }
@@ -33,14 +23,13 @@ const ignorePdfBadFormat = new Set([
33
23
  ]);
34
24
  async function parsePdf(filePath, options) {
35
25
  let doc;
26
+ const buffer = await (0, _promises.readFile)(filePath);
36
27
  try {
37
28
  // Load the document, allowing system fonts as fallback
38
- const loadingTask = (0, _pdf.getDocument)({
39
- url: filePath,
29
+ const doc = await (0, _unpdf.getDocumentProxy)(new Uint8Array(buffer), {
40
30
  disableFontFace: true,
41
- standardFontDataUrl: standardFontsDir
31
+ verbosity: 0
42
32
  });
43
- doc = await loadingTask.promise;
44
33
  const fragments = [];
45
34
  let lastY = undefined;
46
35
  for(let pageNum = 1; pageNum <= doc.numPages; pageNum++){
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../../../../backend/src/applications/files/utils/doc-textify/adapters/pdf.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\nimport path from 'node:path'\nimport { getDocument, GlobalWorkerOptions, PDFDocumentProxy } from 'pdfjs-dist/legacy/build/pdf.mjs'\nimport { DocTextifyOptions } from '../interfaces/doc-textify.interfaces'\n\n// Enable parallel PDF parsing via Node.js worker threads\nGlobalWorkerOptions.workerSrc = require.resolve('pdfjs-dist/legacy/build/pdf.worker.mjs')\nconst pdfjsDistDir = path.dirname(require.resolve('pdfjs-dist/package.json'))\nconst standardFontsDir = path.join(pdfjsDistDir, 'standard_fonts') + '/'\n\n// Type guard to filter true text items\nfunction isTextItem(item: any): item is { str: string; transform: [number, number, number, number, number, number] } {\n return typeof item.str === 'string' && Array.isArray(item.transform)\n}\n\nconst ignorePdfBadFormat = new Set([0x0000, 0x0001])\n\n/** Parse PDF files */\nexport async function parsePdf(filePath: string, options: DocTextifyOptions): Promise<string> {\n let doc: PDFDocumentProxy\n\n try {\n // Load the document, allowing system fonts as fallback\n const loadingTask = getDocument({ url: filePath, disableFontFace: true, standardFontDataUrl: standardFontsDir })\n doc = await loadingTask.promise\n const fragments: string[] = []\n let lastY: number | undefined = undefined\n\n for (let pageNum = 1; pageNum <= doc.numPages; pageNum++) {\n const page = await doc.getPage(pageNum)\n const { items } = await page.getTextContent()\n\n for (const item of items) {\n // Skip non-text items\n if (!isTextItem(item)) continue\n\n const currentY = item.transform[5]\n if (lastY !== undefined && currentY !== lastY) {\n fragments.push(options.newlineDelimiter)\n }\n\n fragments.push(item.str)\n lastY = currentY\n }\n page.cleanup()\n }\n\n const content = fragments.join('')\n if (ignorePdfBadFormat.has(content.charCodeAt(0))) {\n return ''\n }\n return content\n } catch (e) {\n if (options.outputErrorToConsole) {\n console.error('Error parsing PDF:', e)\n }\n throw e\n } finally {\n doc?.destroy().catch((e: Error) => console.error(e))\n }\n}\n"],"names":["parsePdf","GlobalWorkerOptions","workerSrc","require","resolve","pdfjsDistDir","path","dirname","standardFontsDir","join","isTextItem","item","str","Array","isArray","transform","ignorePdfBadFormat","Set","filePath","options","doc","loadingTask","getDocument","url","disableFontFace","standardFontDataUrl","promise","fragments","lastY","undefined","pageNum","numPages","page","getPage","items","getTextContent","currentY","push","newlineDelimiter","cleanup","content","has","charCodeAt","e","outputErrorToConsole","console","error","destroy","catch"],"mappings":"AAAA;;;;CAIC;;;;+BAkBqBA;;;eAAAA;;;iEAjBL;qBACkD;;;;;;AAGnE,yDAAyD;AACzDC,wBAAmB,CAACC,SAAS,GAAGC,QAAQC,OAAO,CAAC;AAChD,MAAMC,eAAeC,iBAAI,CAACC,OAAO,CAACJ,QAAQC,OAAO,CAAC;AAClD,MAAMI,mBAAmBF,iBAAI,CAACG,IAAI,CAACJ,cAAc,oBAAoB;AAErE,uCAAuC;AACvC,SAASK,WAAWC,IAAS;IAC3B,OAAO,OAAOA,KAAKC,GAAG,KAAK,YAAYC,MAAMC,OAAO,CAACH,KAAKI,SAAS;AACrE;AAEA,MAAMC,qBAAqB,IAAIC,IAAI;IAAC;IAAQ;CAAO;AAG5C,eAAejB,SAASkB,QAAgB,EAAEC,OAA0B;IACzE,IAAIC;IAEJ,IAAI;QACF,uDAAuD;QACvD,MAAMC,cAAcC,IAAAA,gBAAW,EAAC;YAAEC,KAAKL;YAAUM,iBAAiB;YAAMC,qBAAqBjB;QAAiB;QAC9GY,MAAM,MAAMC,YAAYK,OAAO;QAC/B,MAAMC,YAAsB,EAAE;QAC9B,IAAIC,QAA4BC;QAEhC,IAAK,IAAIC,UAAU,GAAGA,WAAWV,IAAIW,QAAQ,EAAED,UAAW;YACxD,MAAME,OAAO,MAAMZ,IAAIa,OAAO,CAACH;YAC/B,MAAM,EAAEI,KAAK,EAAE,GAAG,MAAMF,KAAKG,cAAc;YAE3C,KAAK,MAAMxB,QAAQuB,MAAO;gBACxB,sBAAsB;gBACtB,IAAI,CAACxB,WAAWC,OAAO;gBAEvB,MAAMyB,WAAWzB,KAAKI,SAAS,CAAC,EAAE;gBAClC,IAAIa,UAAUC,aAAaO,aAAaR,OAAO;oBAC7CD,UAAUU,IAAI,CAAClB,QAAQmB,gBAAgB;gBACzC;gBAEAX,UAAUU,IAAI,CAAC1B,KAAKC,GAAG;gBACvBgB,QAAQQ;YACV;YACAJ,KAAKO,OAAO;QACd;QAEA,MAAMC,UAAUb,UAAUlB,IAAI,CAAC;QAC/B,IAAIO,mBAAmByB,GAAG,CAACD,QAAQE,UAAU,CAAC,KAAK;YACjD,OAAO;QACT;QACA,OAAOF;IACT,EAAE,OAAOG,GAAG;QACV,IAAIxB,QAAQyB,oBAAoB,EAAE;YAChCC,QAAQC,KAAK,CAAC,sBAAsBH;QACtC;QACA,MAAMA;IACR,SAAU;QACRvB,KAAK2B,UAAUC,MAAM,CAACL,IAAaE,QAAQC,KAAK,CAACH;IACnD;AACF"}
1
+ {"version":3,"sources":["../../../../../../../backend/src/applications/files/utils/doc-textify/adapters/pdf.ts"],"sourcesContent":["/*\n * Copyright (C) 2012-2025 Johan Legrand <johan.legrand@sync-in.com>\n * This file is part of Sync-in | The open source file sync and share solution\n * See the LICENSE file for licensing details\n */\nimport { readFile } from 'node:fs/promises'\nimport { getDocumentProxy } from 'unpdf'\nimport type { PDFDocumentProxy } from 'unpdf/pdfjs'\nimport type { DocTextifyOptions } from '../interfaces/doc-textify.interfaces'\n\n// Type guard to filter true text items\ninterface TextItem {\n str: string\n transform: [number, number, number, number, number, number]\n}\nfunction isTextItem(item: any): item is TextItem {\n return typeof item.str === 'string' && Array.isArray(item.transform)\n}\n\nconst ignorePdfBadFormat = new Set([0x0000, 0x0001])\n\n/** Parse PDF files */\nexport async function parsePdf(filePath: string, options: DocTextifyOptions): Promise<string> {\n let doc: PDFDocumentProxy\n const buffer = await readFile(filePath)\n\n try {\n // Load the document, allowing system fonts as fallback\n const doc = await getDocumentProxy(new Uint8Array(buffer), {\n disableFontFace: true,\n verbosity: 0\n })\n const fragments: string[] = []\n let lastY: number | undefined = undefined\n\n for (let pageNum = 1; pageNum <= doc.numPages; pageNum++) {\n const page = await doc.getPage(pageNum)\n const { items } = await page.getTextContent()\n\n for (const item of items) {\n // Skip non-text items\n if (!isTextItem(item)) continue\n\n const currentY = item.transform[5]\n if (lastY !== undefined && currentY !== lastY) {\n fragments.push(options.newlineDelimiter)\n }\n\n fragments.push(item.str)\n lastY = currentY\n }\n page.cleanup()\n }\n\n const content = fragments.join('')\n if (ignorePdfBadFormat.has(content.charCodeAt(0))) {\n return ''\n }\n return content\n } catch (e) {\n if (options.outputErrorToConsole) {\n console.error('Error parsing PDF:', e)\n }\n throw e\n } finally {\n doc?.destroy().catch((e: Error) => console.error(e))\n }\n}\n"],"names":["parsePdf","isTextItem","item","str","Array","isArray","transform","ignorePdfBadFormat","Set","filePath","options","doc","buffer","readFile","getDocumentProxy","Uint8Array","disableFontFace","verbosity","fragments","lastY","undefined","pageNum","numPages","page","getPage","items","getTextContent","currentY","push","newlineDelimiter","cleanup","content","join","has","charCodeAt","e","outputErrorToConsole","console","error","destroy","catch"],"mappings":"AAAA;;;;CAIC;;;;+BAkBqBA;;;eAAAA;;;0BAjBG;uBACQ;AASjC,SAASC,WAAWC,IAAS;IAC3B,OAAO,OAAOA,KAAKC,GAAG,KAAK,YAAYC,MAAMC,OAAO,CAACH,KAAKI,SAAS;AACrE;AAEA,MAAMC,qBAAqB,IAAIC,IAAI;IAAC;IAAQ;CAAO;AAG5C,eAAeR,SAASS,QAAgB,EAAEC,OAA0B;IACzE,IAAIC;IACJ,MAAMC,SAAS,MAAMC,IAAAA,kBAAQ,EAACJ;IAE9B,IAAI;QACF,uDAAuD;QACvD,MAAME,MAAM,MAAMG,IAAAA,uBAAgB,EAAC,IAAIC,WAAWH,SAAS;YACzDI,iBAAiB;YACjBC,WAAW;QACb;QACA,MAAMC,YAAsB,EAAE;QAC9B,IAAIC,QAA4BC;QAEhC,IAAK,IAAIC,UAAU,GAAGA,WAAWV,IAAIW,QAAQ,EAAED,UAAW;YACxD,MAAME,OAAO,MAAMZ,IAAIa,OAAO,CAACH;YAC/B,MAAM,EAAEI,KAAK,EAAE,GAAG,MAAMF,KAAKG,cAAc;YAE3C,KAAK,MAAMxB,QAAQuB,MAAO;gBACxB,sBAAsB;gBACtB,IAAI,CAACxB,WAAWC,OAAO;gBAEvB,MAAMyB,WAAWzB,KAAKI,SAAS,CAAC,EAAE;gBAClC,IAAIa,UAAUC,aAAaO,aAAaR,OAAO;oBAC7CD,UAAUU,IAAI,CAAClB,QAAQmB,gBAAgB;gBACzC;gBAEAX,UAAUU,IAAI,CAAC1B,KAAKC,GAAG;gBACvBgB,QAAQQ;YACV;YACAJ,KAAKO,OAAO;QACd;QAEA,MAAMC,UAAUb,UAAUc,IAAI,CAAC;QAC/B,IAAIzB,mBAAmB0B,GAAG,CAACF,QAAQG,UAAU,CAAC,KAAK;YACjD,OAAO;QACT;QACA,OAAOH;IACT,EAAE,OAAOI,GAAG;QACV,IAAIzB,QAAQ0B,oBAAoB,EAAE;YAChCC,QAAQC,KAAK,CAAC,sBAAsBH;QACtC;QACA,MAAMA;IACR,SAAU;QACRxB,KAAK4B,UAAUC,MAAM,CAACL,IAAaE,QAAQC,KAAK,CAACH;IACnD;AACF"}
@@ -16,7 +16,6 @@ const _common = require("@nestjs/common");
16
16
  const _bcryptjs = /*#__PURE__*/ _interop_require_default(require("bcryptjs"));
17
17
  const _nodefs = require("node:fs");
18
18
  const _nodepath = /*#__PURE__*/ _interop_require_default(require("node:path"));
19
- const _nodestream = require("node:stream");
20
19
  const _promises = require("node:stream/promises");
21
20
  const _constants = require("../../../common/constants");
22
21
  const _functions = require("../../../common/functions");
@@ -206,7 +205,7 @@ let UsersManager = class UsersManager {
206
205
  throw new _common.HttpException(`avatar not found`, _common.HttpStatus.NOT_FOUND);
207
206
  }
208
207
  const avatarFile = (0, _nodefs.createWriteStream)(avatarPath);
209
- const avatarStream = _nodestream.Readable.from(await (0, _image.generateAvatar)(user.getInitials()));
208
+ const avatarStream = await (0, _image.generateAvatar)(user.getInitials());
210
209
  try {
211
210
  await (0, _promises.pipeline)(avatarStream, avatarFile);
212
211
  } catch (e) {