This release is 12 versions behind 1.0.15 — the latest version of @std/fs. Jump to latest
Built and signed on GitHub ActionsBuilt and signed on GitHub Actions
Built and signed on GitHub Actions
Helpers for working with the file system
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. import { basename } from "jsr:@std/path@^1.0.4/basename"; import { join } from "jsr:@std/path@^1.0.4/join"; import { resolve } from "jsr:@std/path@^1.0.4/resolve"; import { ensureDir, ensureDirSync } from "./ensure_dir.ts"; import { getFileInfoType } from "./_get_file_info_type.ts"; import { toPathString } from "./_to_path_string.ts"; import { isSubdir } from "./_is_subdir.ts"; const isWindows = Deno.build.os === "windows"; /** Options for {@linkcode copy} and {@linkcode copySync}. */ export interface CopyOptions { /** * Whether to overwrite existing file or directory. * * @default {false} */ overwrite?: boolean; /** * When `true`, will set last modification and access times to the ones of * the original source files. When `false`, timestamp behavior is * OS-dependent. * * > [!NOTE] * > This option is currently unsupported for symbolic links. * * @default {false} */ preserveTimestamps?: boolean; } interface InternalCopyOptions extends CopyOptions { /** @default {false} */ isFolder?: boolean; } function assertIsDate(date: Date | null, name: string): asserts date is Date { if (date === null) { throw new Error(`${name} is unavailable`); } } async function ensureValidCopy( src: string | URL, dest: string | URL, options: InternalCopyOptions, ): Promise<Deno.FileInfo | undefined> { let destStat: Deno.FileInfo; try { destStat = await Deno.lstat(dest); } catch (err) { if (err instanceof Deno.errors.NotFound) { return; } throw err; } if (options.isFolder && !destStat.isDirectory) { throw new Error( `Cannot overwrite non-directory '${dest}' with directory '${src}'`, ); } if (!options.overwrite) { throw new Deno.errors.AlreadyExists(`'${dest}' already exists.`); } return destStat; } function ensureValidCopySync( src: string | URL, dest: string | URL, options: InternalCopyOptions, ): Deno.FileInfo | undefined { let destStat: Deno.FileInfo; try { destStat = Deno.lstatSync(dest); } catch (err) { if (err instanceof Deno.errors.NotFound) { return; } throw err; } if (options.isFolder && !destStat.isDirectory) { throw new Error( `Cannot overwrite non-directory '${dest}' with directory '${src}'`, ); } if (!options.overwrite) { throw new Deno.errors.AlreadyExists(`'${dest}' already exists`); } return destStat; } /* copy file to dest */ async function copyFile( src: string | URL, dest: string | URL, options: InternalCopyOptions, ) { await ensureValidCopy(src, dest, options); await Deno.copyFile(src, dest); if (options.preserveTimestamps) { const statInfo = await Deno.stat(src); assertIsDate(statInfo.atime, "statInfo.atime"); assertIsDate(statInfo.mtime, "statInfo.mtime"); await Deno.utime(dest, statInfo.atime, statInfo.mtime); } } /* copy file to dest synchronously */ function copyFileSync( src: string | URL, dest: string | URL, options: InternalCopyOptions, ) { ensureValidCopySync(src, dest, options); Deno.copyFileSync(src, dest); if (options.preserveTimestamps) { const statInfo = Deno.statSync(src); assertIsDate(statInfo.atime, "statInfo.atime"); assertIsDate(statInfo.mtime, "statInfo.mtime"); Deno.utimeSync(dest, statInfo.atime, statInfo.mtime); } } /* copy symlink to dest */ async function copySymLink( src: string | URL, dest: string | URL, options: InternalCopyOptions, ) { await ensureValidCopy(src, dest, options); const originSrcFilePath = await Deno.readLink(src); const type = getFileInfoType(await Deno.lstat(src)); if (isWindows) { await Deno.symlink(originSrcFilePath, dest, { type: type === "dir" ? "dir" : "file", }); } else { await Deno.symlink(originSrcFilePath, dest); } if (options.preserveTimestamps) { const statInfo = await Deno.lstat(src); assertIsDate(statInfo.atime, "statInfo.atime"); assertIsDate(statInfo.mtime, "statInfo.mtime"); await Deno.utime(dest, statInfo.atime, statInfo.mtime); } } /* copy symlink to dest synchronously */ function copySymlinkSync( src: string | URL, dest: string | URL, options: InternalCopyOptions, ) { ensureValidCopySync(src, dest, options); const originSrcFilePath = Deno.readLinkSync(src); const type = getFileInfoType(Deno.lstatSync(src)); if (isWindows) { Deno.symlinkSync(originSrcFilePath, dest, { type: type === "dir" ? "dir" : "file", }); } else { Deno.symlinkSync(originSrcFilePath, dest); } if (options.preserveTimestamps) { const statInfo = Deno.lstatSync(src); assertIsDate(statInfo.atime, "statInfo.atime"); assertIsDate(statInfo.mtime, "statInfo.mtime"); Deno.utimeSync(dest, statInfo.atime, statInfo.mtime); } } /* copy folder from src to dest. */ async function copyDir( src: string | URL, dest: string | URL, options: CopyOptions, ) { const destStat = await ensureValidCopy(src, dest, { ...options, isFolder: true, }); if (!destStat) { await ensureDir(dest); } if (options.preserveTimestamps) { const srcStatInfo = await Deno.stat(src); assertIsDate(srcStatInfo.atime, "statInfo.atime"); assertIsDate(srcStatInfo.mtime, "statInfo.mtime"); await Deno.utime(dest, srcStatInfo.atime, srcStatInfo.mtime); } src = toPathString(src); dest = toPathString(dest); const promises = []; for await (const entry of Deno.readDir(src)) { const srcPath = join(src, entry.name); const destPath = join(dest, basename(srcPath as string)); if (entry.isSymlink) { promises.push(copySymLink(srcPath, destPath, options)); } else if (entry.isDirectory) { promises.push(copyDir(srcPath, destPath, options)); } else if (entry.isFile) { promises.push(copyFile(srcPath, destPath, options)); } } await Promise.all(promises); } /* copy folder from src to dest synchronously */ function copyDirSync( src: string | URL, dest: string | URL, options: CopyOptions, ) { const destStat = ensureValidCopySync(src, dest, { ...options, isFolder: true, }); if (!destStat) { ensureDirSync(dest); } if (options.preserveTimestamps) { const srcStatInfo = Deno.statSync(src); assertIsDate(srcStatInfo.atime, "statInfo.atime"); assertIsDate(srcStatInfo.mtime, "statInfo.mtime"); Deno.utimeSync(dest, srcStatInfo.atime, srcStatInfo.mtime); } src = toPathString(src); dest = toPathString(dest); for (const entry of Deno.readDirSync(src)) { const srcPath = join(src, entry.name); const destPath = join(dest, basename(srcPath as string)); if (entry.isSymlink) { copySymlinkSync(srcPath, destPath, options); } else if (entry.isDirectory) { copyDirSync(srcPath, destPath, options); } else if (entry.isFile) { copyFileSync(srcPath, destPath, options); } } } /** * Asynchronously copy a file or directory (along with its contents), like * {@linkcode https://www.ibm.com/docs/en/aix/7.3?topic=c-cp-command#cp__cp_flagr | cp -r}. * * Both `src` and `dest` must both be a file or directory. * * Requires `--allow-read` and `--allow-write` permissions. * * @see {@link https://docs.deno.com/runtime/manual/basics/permissions#file-system-access} * for more information on Deno's permissions system. * * @param src The source file/directory path as a string or URL. * @param dest The destination file/directory path as a string or URL. * @param options Options for copying. * * @returns A promise that resolves once the copy operation completes. * * @example Basic usage * ```ts no-eval * import { copy } from "@std/fs/copy"; * * await copy("./foo", "./bar"); * ``` * * This will copy the file or directory at `./foo` to `./bar` without * overwriting. * * @example Overwriting files/directories * ```ts no-eval * import { copy } from "@std/fs/copy"; * * await copy("./foo", "./bar", { overwrite: true }); * ``` * * This will copy the file or directory at `./foo` to `./bar` and overwrite * any existing files or directories. * * @example Preserving timestamps * ```ts no-eval * import { copy } from "@std/fs/copy"; * * await copy("./foo", "./bar", { preserveTimestamps: true }); * ``` * * This will copy the file or directory at `./foo` to `./bar` and set the * last modification and access times to the ones of the original source files. */ export async function copy( src: string | URL, dest: string | URL, options: CopyOptions = {}, ) { src = resolve(toPathString(src)); dest = resolve(toPathString(dest)); if (src === dest) { throw new Error("Source and destination cannot be the same"); } const srcStat = await Deno.lstat(src); if (srcStat.isDirectory && isSubdir(src, dest)) { throw new Error( `Cannot copy '${src}' to a subdirectory of itself: '${dest}'`, ); } if (srcStat.isSymlink) { await copySymLink(src, dest, options); } else if (srcStat.isDirectory) { await copyDir(src, dest, options); } else if (srcStat.isFile) { await copyFile(src, dest, options); } } /** * Synchronously copy a file or directory (along with its contents), like * {@linkcode https://www.ibm.com/docs/en/aix/7.3?topic=c-cp-command#cp__cp_flagr | cp -r}. * * Both `src` and `dest` must both be a file or directory. * * Requires `--allow-read` and `--allow-write` permissions. * * @see {@link https://docs.deno.com/runtime/manual/basics/permissions#file-system-access} * for more information on Deno's permissions system. * * @param src The source file/directory path as a string or URL. * @param dest The destination file/directory path as a string or URL. * @param options Options for copying. * * @returns A void value that returns once the copy operation completes. * * @example Basic usage * ```ts no-eval * import { copySync } from "@std/fs/copy"; * * copySync("./foo", "./bar"); * ``` * * This will copy the file or directory at `./foo` to `./bar` without * overwriting. * * @example Overwriting files/directories * ```ts no-eval * import { copySync } from "@std/fs/copy"; * * copySync("./foo", "./bar", { overwrite: true }); * ``` * * This will copy the file or directory at `./foo` to `./bar` and overwrite * any existing files or directories. * * @example Preserving timestamps * ```ts no-eval * import { copySync } from "@std/fs/copy"; * * copySync("./foo", "./bar", { preserveTimestamps: true }); * ``` * * This will copy the file or directory at `./foo` to `./bar` and set the * last modification and access times to the ones of the original source files. */ export function copySync( src: string | URL, dest: string | URL, options: CopyOptions = {}, ) { src = resolve(toPathString(src)); dest = resolve(toPathString(dest)); if (src === dest) { throw new Error("Source and destination cannot be the same"); } const srcStat = Deno.lstatSync(src); if (srcStat.isDirectory && isSubdir(src, dest)) { throw new Error( `Cannot copy '${src}' to a subdirectory of itself: '${dest}'`, ); } if (srcStat.isSymlink) { copySymlinkSync(src, dest, options); } else if (srcStat.isDirectory) { copyDirSync(src, dest, options); } else if (srcStat.isFile) { copyFileSync(src, dest, options); } }