2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2019-10-16 14:39:33 -04:00
|
|
|
import * as path from "../path/mod.ts";
|
2019-05-16 12:19:17 -04:00
|
|
|
import { ensureDir, ensureDirSync } from "./ensure_dir.ts";
|
|
|
|
import { isSubdir, getFileInfoType } from "./utils.ts";
|
2020-02-08 15:15:59 -05:00
|
|
|
import { assert } from "../testing/asserts.ts";
|
2019-05-16 12:19:17 -04:00
|
|
|
|
|
|
|
export interface CopyOptions {
|
|
|
|
/**
|
|
|
|
* overwrite existing file or directory. Default is `false`
|
|
|
|
*/
|
|
|
|
overwrite?: boolean;
|
|
|
|
/**
|
2019-06-19 00:22:01 -04:00
|
|
|
* When `true`, will set last modification and access times to the ones of the
|
|
|
|
* original source files.
|
2019-05-16 12:19:17 -04:00
|
|
|
* When `false`, timestamp behavior is OS-dependent.
|
|
|
|
* Default is `false`.
|
|
|
|
*/
|
|
|
|
preserveTimestamps?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function ensureValidCopy(
|
|
|
|
src: string,
|
|
|
|
dest: string,
|
|
|
|
options: CopyOptions,
|
2019-10-05 12:02:34 -04:00
|
|
|
isCopyFolder = false
|
2020-02-08 15:15:59 -05:00
|
|
|
): Promise<Deno.FileInfo | undefined> {
|
|
|
|
let destStat: Deno.FileInfo;
|
2019-12-18 09:45:19 -05:00
|
|
|
|
|
|
|
try {
|
|
|
|
destStat = await Deno.lstat(dest);
|
|
|
|
} catch (err) {
|
2020-02-24 15:48:35 -05:00
|
|
|
if (err instanceof Deno.errors.NotFound) {
|
2019-12-18 09:45:19 -05:00
|
|
|
return;
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
2020-02-08 15:15:59 -05:00
|
|
|
throw err;
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
|
2019-12-18 09:45:19 -05:00
|
|
|
if (isCopyFolder && !destStat.isDirectory()) {
|
|
|
|
throw new Error(
|
|
|
|
`Cannot overwrite non-directory '${dest}' with directory '${src}'.`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (!options.overwrite) {
|
|
|
|
throw new Error(`'${dest}' already exists.`);
|
|
|
|
}
|
|
|
|
|
2020-02-08 15:15:59 -05:00
|
|
|
return destStat;
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
function ensureValidCopySync(
|
|
|
|
src: string,
|
|
|
|
dest: string,
|
|
|
|
options: CopyOptions,
|
2019-10-05 12:02:34 -04:00
|
|
|
isCopyFolder = false
|
2020-02-08 15:15:59 -05:00
|
|
|
): Deno.FileInfo | undefined {
|
|
|
|
let destStat: Deno.FileInfo;
|
2019-05-16 12:19:17 -04:00
|
|
|
try {
|
|
|
|
destStat = Deno.lstatSync(dest);
|
2019-12-18 09:45:19 -05:00
|
|
|
} catch (err) {
|
2020-02-24 15:48:35 -05:00
|
|
|
if (err instanceof Deno.errors.NotFound) {
|
2019-12-18 09:45:19 -05:00
|
|
|
return;
|
|
|
|
}
|
2020-02-08 15:15:59 -05:00
|
|
|
throw err;
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
|
2020-02-08 15:15:59 -05:00
|
|
|
if (isCopyFolder && !destStat.isDirectory()) {
|
2019-12-18 09:45:19 -05:00
|
|
|
throw new Error(
|
|
|
|
`Cannot overwrite non-directory '${dest}' with directory '${src}'.`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (!options.overwrite) {
|
|
|
|
throw new Error(`'${dest}' already exists.`);
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
|
2020-02-08 15:15:59 -05:00
|
|
|
return destStat;
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* copy file to dest */
|
|
|
|
async function copyFile(
|
|
|
|
src: string,
|
|
|
|
dest: string,
|
|
|
|
options: CopyOptions
|
|
|
|
): Promise<void> {
|
|
|
|
await ensureValidCopy(src, dest, options);
|
|
|
|
await Deno.copyFile(src, dest);
|
|
|
|
if (options.preserveTimestamps) {
|
|
|
|
const statInfo = await Deno.stat(src);
|
2020-02-19 15:36:18 -05:00
|
|
|
assert(statInfo.accessed != null, `statInfo.accessed is unavailable`);
|
|
|
|
assert(statInfo.modified != null, `statInfo.modified is unavailable`);
|
2020-02-08 15:15:59 -05:00
|
|
|
await Deno.utime(dest, statInfo.accessed, statInfo.modified);
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* copy file to dest synchronously */
|
|
|
|
function copyFileSync(src: string, dest: string, options: CopyOptions): void {
|
|
|
|
ensureValidCopySync(src, dest, options);
|
|
|
|
Deno.copyFileSync(src, dest);
|
|
|
|
if (options.preserveTimestamps) {
|
|
|
|
const statInfo = Deno.statSync(src);
|
2020-02-08 15:15:59 -05:00
|
|
|
assert(statInfo.accessed != null, `statInfo.accessed is unavailable`);
|
|
|
|
assert(statInfo.modified != null, `statInfo.modified is unavailable`);
|
|
|
|
Deno.utimeSync(dest, statInfo.accessed, statInfo.modified);
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy symlink to dest */
|
|
|
|
async function copySymLink(
|
|
|
|
src: string,
|
|
|
|
dest: string,
|
|
|
|
options: CopyOptions
|
|
|
|
): Promise<void> {
|
|
|
|
await ensureValidCopy(src, dest, options);
|
|
|
|
const originSrcFilePath = await Deno.readlink(src);
|
|
|
|
const type = getFileInfoType(await Deno.lstat(src));
|
|
|
|
await Deno.symlink(originSrcFilePath, dest, type);
|
|
|
|
if (options.preserveTimestamps) {
|
|
|
|
const statInfo = await Deno.lstat(src);
|
2020-02-08 15:15:59 -05:00
|
|
|
assert(statInfo.accessed != null, `statInfo.accessed is unavailable`);
|
|
|
|
assert(statInfo.modified != null, `statInfo.modified is unavailable`);
|
|
|
|
await Deno.utime(dest, statInfo.accessed, statInfo.modified);
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy symlink to dest synchronously */
|
|
|
|
function copySymlinkSync(
|
|
|
|
src: string,
|
|
|
|
dest: string,
|
|
|
|
options: CopyOptions
|
|
|
|
): void {
|
|
|
|
ensureValidCopySync(src, dest, options);
|
|
|
|
const originSrcFilePath = Deno.readlinkSync(src);
|
|
|
|
const type = getFileInfoType(Deno.lstatSync(src));
|
|
|
|
Deno.symlinkSync(originSrcFilePath, dest, type);
|
|
|
|
if (options.preserveTimestamps) {
|
|
|
|
const statInfo = Deno.lstatSync(src);
|
2020-02-08 15:15:59 -05:00
|
|
|
assert(statInfo.accessed != null, `statInfo.accessed is unavailable`);
|
|
|
|
assert(statInfo.modified != null, `statInfo.modified is unavailable`);
|
|
|
|
Deno.utimeSync(dest, statInfo.accessed, statInfo.modified);
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy folder from src to dest. */
|
|
|
|
async function copyDir(
|
|
|
|
src: string,
|
|
|
|
dest: string,
|
|
|
|
options: CopyOptions
|
|
|
|
): Promise<void> {
|
|
|
|
const destStat = await ensureValidCopy(src, dest, options, true);
|
|
|
|
|
|
|
|
if (!destStat) {
|
|
|
|
await ensureDir(dest);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.preserveTimestamps) {
|
|
|
|
const srcStatInfo = await Deno.stat(src);
|
2020-02-08 15:15:59 -05:00
|
|
|
assert(srcStatInfo.accessed != null, `statInfo.accessed is unavailable`);
|
|
|
|
assert(srcStatInfo.modified != null, `statInfo.modified is unavailable`);
|
|
|
|
await Deno.utime(dest, srcStatInfo.accessed, srcStatInfo.modified);
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const files = await Deno.readDir(src);
|
|
|
|
|
|
|
|
for (const file of files) {
|
2020-02-08 15:15:59 -05:00
|
|
|
assert(file.name != null, "file.name must be set");
|
|
|
|
const srcPath = path.join(src, file.name);
|
2019-05-16 12:19:17 -04:00
|
|
|
const destPath = path.join(dest, path.basename(srcPath as string));
|
|
|
|
if (file.isDirectory()) {
|
|
|
|
await copyDir(srcPath, destPath, options);
|
|
|
|
} else if (file.isFile()) {
|
|
|
|
await copyFile(srcPath, destPath, options);
|
|
|
|
} else if (file.isSymlink()) {
|
|
|
|
await copySymLink(srcPath, destPath, options);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy folder from src to dest synchronously */
|
|
|
|
function copyDirSync(src: string, dest: string, options: CopyOptions): void {
|
2020-02-08 15:15:59 -05:00
|
|
|
const destStat = ensureValidCopySync(src, dest, options, true);
|
2019-05-16 12:19:17 -04:00
|
|
|
|
|
|
|
if (!destStat) {
|
|
|
|
ensureDirSync(dest);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.preserveTimestamps) {
|
|
|
|
const srcStatInfo = Deno.statSync(src);
|
2020-02-08 15:15:59 -05:00
|
|
|
assert(srcStatInfo.accessed != null, `statInfo.accessed is unavailable`);
|
|
|
|
assert(srcStatInfo.modified != null, `statInfo.modified is unavailable`);
|
|
|
|
Deno.utimeSync(dest, srcStatInfo.accessed, srcStatInfo.modified);
|
2019-05-16 12:19:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const files = Deno.readDirSync(src);
|
|
|
|
|
|
|
|
for (const file of files) {
|
2020-02-08 15:15:59 -05:00
|
|
|
assert(file.name != null, "file.name must be set");
|
|
|
|
const srcPath = path.join(src, file.name);
|
2019-05-16 12:19:17 -04:00
|
|
|
const destPath = path.join(dest, path.basename(srcPath as string));
|
|
|
|
if (file.isDirectory()) {
|
|
|
|
copyDirSync(srcPath, destPath, options);
|
|
|
|
} else if (file.isFile()) {
|
|
|
|
copyFileSync(srcPath, destPath, options);
|
|
|
|
} else if (file.isSymlink()) {
|
|
|
|
copySymlinkSync(srcPath, destPath, options);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy a file or directory. The directory can have contents. Like `cp -r`.
|
2019-12-18 09:45:19 -05:00
|
|
|
* Requires the `--allow-read` and `--alow-write` flag.
|
2019-05-16 12:19:17 -04:00
|
|
|
* @param src the file/directory path.
|
2019-06-19 00:22:01 -04:00
|
|
|
* Note that if `src` is a directory it will copy everything inside
|
|
|
|
* of this directory, not the entire directory itself
|
|
|
|
* @param dest the destination path. Note that if `src` is a file, `dest` cannot
|
|
|
|
* be a directory
|
2019-05-16 12:19:17 -04:00
|
|
|
* @param options
|
|
|
|
*/
|
|
|
|
export async function copy(
|
|
|
|
src: string,
|
|
|
|
dest: string,
|
|
|
|
options: CopyOptions = {}
|
|
|
|
): Promise<void> {
|
|
|
|
src = path.resolve(src);
|
|
|
|
dest = path.resolve(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.isDirectory()) {
|
|
|
|
await copyDir(src, dest, options);
|
|
|
|
} else if (srcStat.isFile()) {
|
|
|
|
await copyFile(src, dest, options);
|
|
|
|
} else if (srcStat.isSymlink()) {
|
|
|
|
await copySymLink(src, dest, options);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy a file or directory. The directory can have contents. Like `cp -r`.
|
2019-12-18 09:45:19 -05:00
|
|
|
* Requires the `--allow-read` and `--alow-write` flag.
|
2019-05-16 12:19:17 -04:00
|
|
|
* @param src the file/directory path.
|
2019-06-19 00:22:01 -04:00
|
|
|
* Note that if `src` is a directory it will copy everything inside
|
|
|
|
* of this directory, not the entire directory itself
|
|
|
|
* @param dest the destination path. Note that if `src` is a file, `dest` cannot
|
|
|
|
* be a directory
|
2019-05-16 12:19:17 -04:00
|
|
|
* @param options
|
|
|
|
*/
|
|
|
|
export function copySync(
|
|
|
|
src: string,
|
|
|
|
dest: string,
|
|
|
|
options: CopyOptions = {}
|
|
|
|
): void {
|
|
|
|
src = path.resolve(src);
|
|
|
|
dest = path.resolve(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.isDirectory()) {
|
|
|
|
copyDirSync(src, dest, options);
|
|
|
|
} else if (srcStat.isFile()) {
|
|
|
|
copyFileSync(src, dest, options);
|
|
|
|
} else if (srcStat.isSymlink()) {
|
|
|
|
copySymlinkSync(src, dest, options);
|
|
|
|
}
|
|
|
|
}
|