mirror of
https://github.com/denoland/deno.git
synced 2024-12-31 11:34:15 -05:00
Implement Deno.symlink() for windows (#5533)
This commit is contained in:
parent
88b24261ba
commit
6072755ead
9 changed files with 155 additions and 244 deletions
28
cli/js/lib.deno.unstable.d.ts
vendored
28
cli/js/lib.deno.unstable.d.ts
vendored
|
@ -41,48 +41,44 @@ declare namespace Deno {
|
||||||
* Requires `allow-read` and `allow-write` permissions. */
|
* Requires `allow-read` and `allow-write` permissions. */
|
||||||
export function link(oldpath: string, newpath: string): Promise<void>;
|
export function link(oldpath: string, newpath: string): Promise<void>;
|
||||||
|
|
||||||
/** **UNSTABLE**: `type` argument type may be changed to `"dir" | "file"`.
|
export type SymlinkOptions = {
|
||||||
*
|
type: "file" | "dir";
|
||||||
* **UNSTABLE**: needs security review.
|
};
|
||||||
|
|
||||||
|
/** **UNSTABLE**: needs security review.
|
||||||
*
|
*
|
||||||
* Creates `newpath` as a symbolic link to `oldpath`.
|
* Creates `newpath` as a symbolic link to `oldpath`.
|
||||||
*
|
*
|
||||||
* The type argument can be set to `dir` or `file`. This argument is only
|
* The options.type parameter can be set to `file` or `dir`. This argument is only
|
||||||
* available on Windows and ignored on other platforms.
|
* available on Windows and ignored on other platforms.
|
||||||
*
|
*
|
||||||
* NOTE: This function is not yet implemented on Windows.
|
|
||||||
*
|
|
||||||
* ```ts
|
* ```ts
|
||||||
* Deno.symlinkSync("old/name", "new/name");
|
* Deno.symlinkSync("old/name", "new/name");
|
||||||
* ```
|
* ```
|
||||||
*
|
*
|
||||||
* Requires `allow-read` and `allow-write` permissions. */
|
* Requires `allow-write` permission. */
|
||||||
export function symlinkSync(
|
export function symlinkSync(
|
||||||
oldpath: string,
|
oldpath: string,
|
||||||
newpath: string,
|
newpath: string,
|
||||||
type?: string
|
options?: SymlinkOptions
|
||||||
): void;
|
): void;
|
||||||
|
|
||||||
/** **UNSTABLE**: `type` argument may be changed to `"dir" | "file"`
|
/** **UNSTABLE**: needs security review.
|
||||||
*
|
|
||||||
* **UNSTABLE**: needs security review.
|
|
||||||
*
|
*
|
||||||
* Creates `newpath` as a symbolic link to `oldpath`.
|
* Creates `newpath` as a symbolic link to `oldpath`.
|
||||||
*
|
*
|
||||||
* The type argument can be set to `dir` or `file`. This argument is only
|
* The options.type parameter can be set to `file` or `dir`. This argument is only
|
||||||
* available on Windows and ignored on other platforms.
|
* available on Windows and ignored on other platforms.
|
||||||
*
|
*
|
||||||
* NOTE: This function is not yet implemented on Windows.
|
|
||||||
*
|
|
||||||
* ```ts
|
* ```ts
|
||||||
* await Deno.symlink("old/name", "new/name");
|
* await Deno.symlink("old/name", "new/name");
|
||||||
* ```
|
* ```
|
||||||
*
|
*
|
||||||
* Requires `allow-read` and `allow-write` permissions. */
|
* Requires `allow-write` permission. */
|
||||||
export function symlink(
|
export function symlink(
|
||||||
oldpath: string,
|
oldpath: string,
|
||||||
newpath: string,
|
newpath: string,
|
||||||
type?: string
|
options?: SymlinkOptions
|
||||||
): Promise<void>;
|
): Promise<void>;
|
||||||
|
|
||||||
/** **UNSTABLE** */
|
/** **UNSTABLE** */
|
||||||
|
|
|
@ -1,26 +1,22 @@
|
||||||
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
||||||
import { sendSync, sendAsync } from "../dispatch_json.ts";
|
import { sendSync, sendAsync } from "../dispatch_json.ts";
|
||||||
import * as util from "../../util.ts";
|
|
||||||
import { build } from "../../build.ts";
|
export type symlinkOptions = {
|
||||||
|
type: "file" | "dir";
|
||||||
|
};
|
||||||
|
|
||||||
export function symlinkSync(
|
export function symlinkSync(
|
||||||
oldpath: string,
|
oldpath: string,
|
||||||
newpath: string,
|
newpath: string,
|
||||||
type?: string
|
options?: symlinkOptions
|
||||||
): void {
|
): void {
|
||||||
if (build.os === "windows" && type) {
|
sendSync("op_symlink", { oldpath, newpath, options });
|
||||||
return util.notImplemented();
|
|
||||||
}
|
|
||||||
sendSync("op_symlink", { oldpath, newpath });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function symlink(
|
export async function symlink(
|
||||||
oldpath: string,
|
oldpath: string,
|
||||||
newpath: string,
|
newpath: string,
|
||||||
type?: string
|
options?: symlinkOptions
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
if (build.os === "windows" && type) {
|
await sendAsync("op_symlink", { oldpath, newpath, options });
|
||||||
return util.notImplemented();
|
|
||||||
}
|
|
||||||
await sendAsync("op_symlink", { oldpath, newpath });
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -83,27 +83,23 @@ unitTest(
|
||||||
{ perms: { write: true, read: true } },
|
{ perms: { write: true, read: true } },
|
||||||
function removeSyncDanglingSymlinkSuccess(): void {
|
function removeSyncDanglingSymlinkSuccess(): void {
|
||||||
const danglingSymlinkPath = Deno.makeTempDirSync() + "/dangling_symlink";
|
const danglingSymlinkPath = Deno.makeTempDirSync() + "/dangling_symlink";
|
||||||
// TODO(#3832): Remove "not Implemented" error checking when symlink creation is implemented for Windows
|
|
||||||
let errOnWindows;
|
|
||||||
try {
|
|
||||||
Deno.symlinkSync("unexistent_file", danglingSymlinkPath);
|
|
||||||
} catch (err) {
|
|
||||||
errOnWindows = err;
|
|
||||||
}
|
|
||||||
if (Deno.build.os === "windows") {
|
if (Deno.build.os === "windows") {
|
||||||
assertEquals(errOnWindows.message, "not implemented");
|
Deno.symlinkSync("unexistent_file", danglingSymlinkPath, {
|
||||||
|
type: "file",
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
const pathInfo = Deno.lstatSync(danglingSymlinkPath);
|
Deno.symlinkSync("unexistent_file", danglingSymlinkPath);
|
||||||
assert(pathInfo.isSymlink);
|
|
||||||
Deno.removeSync(danglingSymlinkPath);
|
|
||||||
let err;
|
|
||||||
try {
|
|
||||||
Deno.lstatSync(danglingSymlinkPath);
|
|
||||||
} catch (e) {
|
|
||||||
err = e;
|
|
||||||
}
|
|
||||||
assert(err instanceof Deno.errors.NotFound);
|
|
||||||
}
|
}
|
||||||
|
const pathInfo = Deno.lstatSync(danglingSymlinkPath);
|
||||||
|
assert(pathInfo.isSymlink);
|
||||||
|
Deno.removeSync(danglingSymlinkPath);
|
||||||
|
let err;
|
||||||
|
try {
|
||||||
|
Deno.lstatSync(danglingSymlinkPath);
|
||||||
|
} catch (e) {
|
||||||
|
err = e;
|
||||||
|
}
|
||||||
|
assert(err instanceof Deno.errors.NotFound);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -116,28 +112,22 @@ unitTest(
|
||||||
const filePath = tempDir + "/test.txt";
|
const filePath = tempDir + "/test.txt";
|
||||||
const validSymlinkPath = tempDir + "/valid_symlink";
|
const validSymlinkPath = tempDir + "/valid_symlink";
|
||||||
Deno.writeFileSync(filePath, data, { mode: 0o666 });
|
Deno.writeFileSync(filePath, data, { mode: 0o666 });
|
||||||
// TODO(#3832): Remove "not Implemented" error checking when symlink creation is implemented for Windows
|
|
||||||
let errOnWindows;
|
|
||||||
try {
|
|
||||||
Deno.symlinkSync(filePath, validSymlinkPath);
|
|
||||||
} catch (err) {
|
|
||||||
errOnWindows = err;
|
|
||||||
}
|
|
||||||
if (Deno.build.os === "windows") {
|
if (Deno.build.os === "windows") {
|
||||||
assertEquals(errOnWindows.message, "not implemented");
|
Deno.symlinkSync(filePath, validSymlinkPath, { type: "file" });
|
||||||
} else {
|
} else {
|
||||||
const symlinkPathInfo = Deno.statSync(validSymlinkPath);
|
Deno.symlinkSync(filePath, validSymlinkPath);
|
||||||
assert(symlinkPathInfo.isFile);
|
|
||||||
Deno.removeSync(validSymlinkPath);
|
|
||||||
let err;
|
|
||||||
try {
|
|
||||||
Deno.statSync(validSymlinkPath);
|
|
||||||
} catch (e) {
|
|
||||||
err = e;
|
|
||||||
}
|
|
||||||
Deno.removeSync(filePath);
|
|
||||||
assert(err instanceof Deno.errors.NotFound);
|
|
||||||
}
|
}
|
||||||
|
const symlinkPathInfo = Deno.statSync(validSymlinkPath);
|
||||||
|
assert(symlinkPathInfo.isFile);
|
||||||
|
Deno.removeSync(validSymlinkPath);
|
||||||
|
let err;
|
||||||
|
try {
|
||||||
|
Deno.statSync(validSymlinkPath);
|
||||||
|
} catch (e) {
|
||||||
|
err = e;
|
||||||
|
}
|
||||||
|
Deno.removeSync(filePath);
|
||||||
|
assert(err instanceof Deno.errors.NotFound);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -319,27 +309,23 @@ unitTest(
|
||||||
{ perms: { write: true, read: true } },
|
{ perms: { write: true, read: true } },
|
||||||
async function removeDanglingSymlinkSuccess(): Promise<void> {
|
async function removeDanglingSymlinkSuccess(): Promise<void> {
|
||||||
const danglingSymlinkPath = Deno.makeTempDirSync() + "/dangling_symlink";
|
const danglingSymlinkPath = Deno.makeTempDirSync() + "/dangling_symlink";
|
||||||
// TODO(#3832): Remove "not Implemented" error checking when symlink creation is implemented for Windows
|
|
||||||
let errOnWindows;
|
|
||||||
try {
|
|
||||||
Deno.symlinkSync("unexistent_file", danglingSymlinkPath);
|
|
||||||
} catch (e) {
|
|
||||||
errOnWindows = e;
|
|
||||||
}
|
|
||||||
if (Deno.build.os === "windows") {
|
if (Deno.build.os === "windows") {
|
||||||
assertEquals(errOnWindows.message, "not implemented");
|
Deno.symlinkSync("unexistent_file", danglingSymlinkPath, {
|
||||||
|
type: "file",
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
const pathInfo = Deno.lstatSync(danglingSymlinkPath);
|
Deno.symlinkSync("unexistent_file", danglingSymlinkPath);
|
||||||
assert(pathInfo.isSymlink);
|
|
||||||
await Deno.remove(danglingSymlinkPath);
|
|
||||||
let err;
|
|
||||||
try {
|
|
||||||
Deno.lstatSync(danglingSymlinkPath);
|
|
||||||
} catch (e) {
|
|
||||||
err = e;
|
|
||||||
}
|
|
||||||
assert(err instanceof Deno.errors.NotFound);
|
|
||||||
}
|
}
|
||||||
|
const pathInfo = Deno.lstatSync(danglingSymlinkPath);
|
||||||
|
assert(pathInfo.isSymlink);
|
||||||
|
await Deno.remove(danglingSymlinkPath);
|
||||||
|
let err;
|
||||||
|
try {
|
||||||
|
Deno.lstatSync(danglingSymlinkPath);
|
||||||
|
} catch (e) {
|
||||||
|
err = e;
|
||||||
|
}
|
||||||
|
assert(err instanceof Deno.errors.NotFound);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -352,28 +338,22 @@ unitTest(
|
||||||
const filePath = tempDir + "/test.txt";
|
const filePath = tempDir + "/test.txt";
|
||||||
const validSymlinkPath = tempDir + "/valid_symlink";
|
const validSymlinkPath = tempDir + "/valid_symlink";
|
||||||
Deno.writeFileSync(filePath, data, { mode: 0o666 });
|
Deno.writeFileSync(filePath, data, { mode: 0o666 });
|
||||||
// TODO(#3832): Remove "not Implemented" error checking when symlink creation is implemented for Windows
|
|
||||||
let errOnWindows;
|
|
||||||
try {
|
|
||||||
Deno.symlinkSync(filePath, validSymlinkPath);
|
|
||||||
} catch (e) {
|
|
||||||
errOnWindows = e;
|
|
||||||
}
|
|
||||||
if (Deno.build.os === "windows") {
|
if (Deno.build.os === "windows") {
|
||||||
assertEquals(errOnWindows.message, "not implemented");
|
Deno.symlinkSync(filePath, validSymlinkPath, { type: "file" });
|
||||||
} else {
|
} else {
|
||||||
const symlinkPathInfo = Deno.statSync(validSymlinkPath);
|
Deno.symlinkSync(filePath, validSymlinkPath);
|
||||||
assert(symlinkPathInfo.isFile);
|
|
||||||
await Deno.remove(validSymlinkPath);
|
|
||||||
let err;
|
|
||||||
try {
|
|
||||||
Deno.statSync(validSymlinkPath);
|
|
||||||
} catch (e) {
|
|
||||||
err = e;
|
|
||||||
}
|
|
||||||
Deno.removeSync(filePath);
|
|
||||||
assert(err instanceof Deno.errors.NotFound);
|
|
||||||
}
|
}
|
||||||
|
const symlinkPathInfo = Deno.statSync(validSymlinkPath);
|
||||||
|
assert(symlinkPathInfo.isFile);
|
||||||
|
await Deno.remove(validSymlinkPath);
|
||||||
|
let err;
|
||||||
|
try {
|
||||||
|
Deno.statSync(validSymlinkPath);
|
||||||
|
} catch (e) {
|
||||||
|
err = e;
|
||||||
|
}
|
||||||
|
Deno.removeSync(filePath);
|
||||||
|
assert(err instanceof Deno.errors.NotFound);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -8,22 +8,12 @@ unitTest(
|
||||||
const oldname = testDir + "/oldname";
|
const oldname = testDir + "/oldname";
|
||||||
const newname = testDir + "/newname";
|
const newname = testDir + "/newname";
|
||||||
Deno.mkdirSync(oldname);
|
Deno.mkdirSync(oldname);
|
||||||
let errOnWindows;
|
|
||||||
// Just for now, until we implement symlink for Windows.
|
// Just for now, until we implement symlink for Windows.
|
||||||
try {
|
Deno.symlinkSync(oldname, newname);
|
||||||
Deno.symlinkSync(oldname, newname);
|
const newNameInfoLStat = Deno.lstatSync(newname);
|
||||||
} catch (e) {
|
const newNameInfoStat = Deno.statSync(newname);
|
||||||
errOnWindows = e;
|
assert(newNameInfoLStat.isSymlink);
|
||||||
}
|
assert(newNameInfoStat.isDirectory);
|
||||||
if (errOnWindows) {
|
|
||||||
assertEquals(Deno.build.os, "windows");
|
|
||||||
assertEquals(errOnWindows.message, "not implemented");
|
|
||||||
} else {
|
|
||||||
const newNameInfoLStat = Deno.lstatSync(newname);
|
|
||||||
const newNameInfoStat = Deno.statSync(newname);
|
|
||||||
assert(newNameInfoLStat.isSymlink);
|
|
||||||
assert(newNameInfoStat.isDirectory);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -38,28 +28,6 @@ unitTest(function symlinkSyncPerm(): void {
|
||||||
assertEquals(err.name, "PermissionDenied");
|
assertEquals(err.name, "PermissionDenied");
|
||||||
});
|
});
|
||||||
|
|
||||||
// Just for now, until we implement symlink for Windows.
|
|
||||||
// Symlink with type should succeed on other platforms with type ignored
|
|
||||||
unitTest(
|
|
||||||
{ perms: { write: true } },
|
|
||||||
function symlinkSyncNotImplemented(): void {
|
|
||||||
const testDir = Deno.makeTempDirSync();
|
|
||||||
const oldname = testDir + "/oldname";
|
|
||||||
const newname = testDir + "/newname";
|
|
||||||
let err;
|
|
||||||
try {
|
|
||||||
Deno.symlinkSync(oldname, newname, "dir");
|
|
||||||
} catch (e) {
|
|
||||||
err = e;
|
|
||||||
}
|
|
||||||
if (err) {
|
|
||||||
assertEquals(Deno.build.os, "windows");
|
|
||||||
// from cli/js/util.ts:notImplemented
|
|
||||||
assertEquals(err.message, "not implemented");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
unitTest(
|
unitTest(
|
||||||
{ perms: { read: true, write: true } },
|
{ perms: { read: true, write: true } },
|
||||||
async function symlinkSuccess(): Promise<void> {
|
async function symlinkSuccess(): Promise<void> {
|
||||||
|
@ -67,20 +35,10 @@ unitTest(
|
||||||
const oldname = testDir + "/oldname";
|
const oldname = testDir + "/oldname";
|
||||||
const newname = testDir + "/newname";
|
const newname = testDir + "/newname";
|
||||||
Deno.mkdirSync(oldname);
|
Deno.mkdirSync(oldname);
|
||||||
let errOnWindows;
|
await Deno.symlink(oldname, newname);
|
||||||
// Just for now, until we implement symlink for Windows.
|
const newNameInfoLStat = Deno.lstatSync(newname);
|
||||||
try {
|
const newNameInfoStat = Deno.statSync(newname);
|
||||||
await Deno.symlink(oldname, newname);
|
assert(newNameInfoLStat.isSymlink, "NOT SYMLINK");
|
||||||
} catch (e) {
|
assert(newNameInfoStat.isDirectory, "NOT DIRECTORY");
|
||||||
errOnWindows = e;
|
|
||||||
}
|
|
||||||
if (errOnWindows) {
|
|
||||||
assertEquals(errOnWindows.message, "not implemented");
|
|
||||||
} else {
|
|
||||||
const newNameInfoLStat = Deno.lstatSync(newname);
|
|
||||||
const newNameInfoStat = Deno.statSync(newname);
|
|
||||||
assert(newNameInfoLStat.isSymlink);
|
|
||||||
assert(newNameInfoStat.isDirectory);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
|
@ -687,6 +687,15 @@ struct SymlinkArgs {
|
||||||
promise_id: Option<u64>,
|
promise_id: Option<u64>,
|
||||||
oldpath: String,
|
oldpath: String,
|
||||||
newpath: String,
|
newpath: String,
|
||||||
|
#[cfg(not(unix))]
|
||||||
|
options: Option<SymlinkOptions>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(not(unix))]
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
#[serde(rename_all = "camelCase")]
|
||||||
|
struct SymlinkOptions {
|
||||||
|
_type: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_symlink(
|
fn op_symlink(
|
||||||
|
@ -710,13 +719,34 @@ fn op_symlink(
|
||||||
symlink(&oldpath, &newpath)?;
|
symlink(&oldpath, &newpath)?;
|
||||||
Ok(json!({}))
|
Ok(json!({}))
|
||||||
}
|
}
|
||||||
// TODO Implement symlink, use type for Windows
|
|
||||||
#[cfg(not(unix))]
|
#[cfg(not(unix))]
|
||||||
{
|
{
|
||||||
// Unlike with chmod/chown, here we don't
|
use std::os::windows::fs::{symlink_dir, symlink_file};
|
||||||
// require `oldpath` to exist on Windows
|
|
||||||
let _ = oldpath; // avoid unused warning
|
match args.options {
|
||||||
Err(OpError::not_implemented())
|
Some(options) => match options._type.as_ref() {
|
||||||
|
"file" => symlink_file(&oldpath, &newpath)?,
|
||||||
|
"dir" => symlink_dir(&oldpath, &newpath)?,
|
||||||
|
_ => return Err(OpError::type_error("unsupported type".to_string())),
|
||||||
|
},
|
||||||
|
None => {
|
||||||
|
let old_meta = std::fs::metadata(&oldpath);
|
||||||
|
match old_meta {
|
||||||
|
Ok(metadata) => {
|
||||||
|
if metadata.is_file() {
|
||||||
|
symlink_file(&oldpath, &newpath)?
|
||||||
|
} else if metadata.is_dir() {
|
||||||
|
symlink_dir(&oldpath, &newpath)?
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(_) => return Err(OpError::type_error(
|
||||||
|
"you must pass a `options` argument for non-existent target path in windows"
|
||||||
|
.to_string(),
|
||||||
|
)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
Ok(json!({}))
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,8 @@ import { ensureDir, ensureDirSync } from "./ensure_dir.ts";
|
||||||
import { isSubdir, getFileInfoType } from "./_util.ts";
|
import { isSubdir, getFileInfoType } from "./_util.ts";
|
||||||
import { assert } from "../testing/asserts.ts";
|
import { assert } from "../testing/asserts.ts";
|
||||||
|
|
||||||
|
const isWindows = Deno.build.os === "windows";
|
||||||
|
|
||||||
export interface CopyOptions {
|
export interface CopyOptions {
|
||||||
/**
|
/**
|
||||||
* overwrite existing file or directory. Default is `false`
|
* overwrite existing file or directory. Default is `false`
|
||||||
|
@ -111,7 +113,13 @@ async function copySymLink(
|
||||||
await ensureValidCopy(src, dest, options);
|
await ensureValidCopy(src, dest, options);
|
||||||
const originSrcFilePath = await Deno.readLink(src);
|
const originSrcFilePath = await Deno.readLink(src);
|
||||||
const type = getFileInfoType(await Deno.lstat(src));
|
const type = getFileInfoType(await Deno.lstat(src));
|
||||||
await Deno.symlink(originSrcFilePath, dest, type);
|
if (isWindows) {
|
||||||
|
await Deno.symlink(originSrcFilePath, dest, {
|
||||||
|
type: type === "dir" ? "dir" : "file",
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
await Deno.symlink(originSrcFilePath, dest);
|
||||||
|
}
|
||||||
if (options.preserveTimestamps) {
|
if (options.preserveTimestamps) {
|
||||||
const statInfo = await Deno.lstat(src);
|
const statInfo = await Deno.lstat(src);
|
||||||
assert(statInfo.atime instanceof Date, `statInfo.atime is unavailable`);
|
assert(statInfo.atime instanceof Date, `statInfo.atime is unavailable`);
|
||||||
|
@ -129,7 +137,14 @@ function copySymlinkSync(
|
||||||
ensureValidCopySync(src, dest, options);
|
ensureValidCopySync(src, dest, options);
|
||||||
const originSrcFilePath = Deno.readLinkSync(src);
|
const originSrcFilePath = Deno.readLinkSync(src);
|
||||||
const type = getFileInfoType(Deno.lstatSync(src));
|
const type = getFileInfoType(Deno.lstatSync(src));
|
||||||
Deno.symlinkSync(originSrcFilePath, dest, type);
|
if (isWindows) {
|
||||||
|
Deno.symlinkSync(originSrcFilePath, dest, {
|
||||||
|
type: type === "dir" ? "dir" : "file",
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
Deno.symlinkSync(originSrcFilePath, dest);
|
||||||
|
}
|
||||||
|
|
||||||
if (options.preserveTimestamps) {
|
if (options.preserveTimestamps) {
|
||||||
const statInfo = Deno.lstatSync(src);
|
const statInfo = Deno.lstatSync(src);
|
||||||
assert(statInfo.atime instanceof Date, `statInfo.atime is unavailable`);
|
assert(statInfo.atime instanceof Date, `statInfo.atime is unavailable`);
|
||||||
|
|
|
@ -14,9 +14,6 @@ import { ensureSymlink, ensureSymlinkSync } from "./ensure_symlink.ts";
|
||||||
|
|
||||||
const testdataDir = path.resolve("fs", "testdata");
|
const testdataDir = path.resolve("fs", "testdata");
|
||||||
|
|
||||||
// TODO(axetroy): Add test for Windows once symlink is implemented for Windows.
|
|
||||||
const isWindows = Deno.build.os === "windows";
|
|
||||||
|
|
||||||
function testCopy(name: string, cb: (tempDir: string) => Promise<void>): void {
|
function testCopy(name: string, cb: (tempDir: string) => Promise<void>): void {
|
||||||
Deno.test({
|
Deno.test({
|
||||||
name,
|
name,
|
||||||
|
@ -257,14 +254,6 @@ testCopy(
|
||||||
const srcLink = path.join(dir, "0.txt");
|
const srcLink = path.join(dir, "0.txt");
|
||||||
const destLink = path.join(tempDir, "0_copy.txt");
|
const destLink = path.join(tempDir, "0_copy.txt");
|
||||||
|
|
||||||
if (isWindows) {
|
|
||||||
await assertThrowsAsync(
|
|
||||||
// (): Promise<void> => copy(srcLink, destLink),
|
|
||||||
(): Promise<void> => ensureSymlink(srcLink, destLink)
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(
|
assert(
|
||||||
(await Deno.lstat(srcLink)).isSymlink,
|
(await Deno.lstat(srcLink)).isSymlink,
|
||||||
`'${srcLink}' should be symlink type`
|
`'${srcLink}' should be symlink type`
|
||||||
|
@ -285,14 +274,6 @@ testCopy(
|
||||||
const srcLink = path.join(tempDir, "copy_dir_link");
|
const srcLink = path.join(tempDir, "copy_dir_link");
|
||||||
const destLink = path.join(tempDir, "copy_dir_link_copy");
|
const destLink = path.join(tempDir, "copy_dir_link_copy");
|
||||||
|
|
||||||
if (isWindows) {
|
|
||||||
await assertThrowsAsync(
|
|
||||||
// (): Promise<void> => copy(srcLink, destLink),
|
|
||||||
(): Promise<void> => ensureSymlink(srcLink, destLink)
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await ensureSymlink(srcDir, srcLink);
|
await ensureSymlink(srcDir, srcLink);
|
||||||
|
|
||||||
assert(
|
assert(
|
||||||
|
@ -497,14 +478,6 @@ testCopySync(
|
||||||
const srcLink = path.join(dir, "0.txt");
|
const srcLink = path.join(dir, "0.txt");
|
||||||
const destLink = path.join(tempDir, "0_copy.txt");
|
const destLink = path.join(tempDir, "0_copy.txt");
|
||||||
|
|
||||||
if (isWindows) {
|
|
||||||
assertThrows(
|
|
||||||
// (): void => copySync(srcLink, destLink),
|
|
||||||
(): void => ensureSymlinkSync(srcLink, destLink)
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(
|
assert(
|
||||||
Deno.lstatSync(srcLink).isSymlink,
|
Deno.lstatSync(srcLink).isSymlink,
|
||||||
`'${srcLink}' should be symlink type`
|
`'${srcLink}' should be symlink type`
|
||||||
|
@ -525,14 +498,6 @@ testCopySync(
|
||||||
const srcLink = path.join(tempDir, "copy_dir_link");
|
const srcLink = path.join(tempDir, "copy_dir_link");
|
||||||
const destLink = path.join(tempDir, "copy_dir_link_copy");
|
const destLink = path.join(tempDir, "copy_dir_link_copy");
|
||||||
|
|
||||||
if (isWindows) {
|
|
||||||
assertThrows(
|
|
||||||
// (): void => copySync(srcLink, destLink),
|
|
||||||
(): void => ensureSymlinkSync(srcLink, destLink)
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ensureSymlinkSync(originDir, srcLink);
|
ensureSymlinkSync(originDir, srcLink);
|
||||||
|
|
||||||
assert(
|
assert(
|
||||||
|
|
|
@ -28,7 +28,14 @@ export async function ensureSymlink(src: string, dest: string): Promise<void> {
|
||||||
|
|
||||||
await ensureDir(path.dirname(dest));
|
await ensureDir(path.dirname(dest));
|
||||||
|
|
||||||
await Deno.symlink(src, dest, srcFilePathType);
|
ensureDirSync(path.dirname(dest));
|
||||||
|
if (Deno.build.os === "windows") {
|
||||||
|
await Deno.symlink(src, dest, {
|
||||||
|
type: srcFilePathType === "dir" ? "dir" : "file",
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
await Deno.symlink(src, dest);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -54,6 +61,11 @@ export function ensureSymlinkSync(src: string, dest: string): void {
|
||||||
}
|
}
|
||||||
|
|
||||||
ensureDirSync(path.dirname(dest));
|
ensureDirSync(path.dirname(dest));
|
||||||
|
if (Deno.build.os === "windows") {
|
||||||
Deno.symlinkSync(src, dest, srcFilePathType);
|
Deno.symlinkSync(src, dest, {
|
||||||
|
type: srcFilePathType === "dir" ? "dir" : "file",
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
Deno.symlinkSync(src, dest);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,7 +9,6 @@ import * as path from "../path/mod.ts";
|
||||||
import { ensureSymlink, ensureSymlinkSync } from "./ensure_symlink.ts";
|
import { ensureSymlink, ensureSymlinkSync } from "./ensure_symlink.ts";
|
||||||
|
|
||||||
const testdataDir = path.resolve("fs", "testdata");
|
const testdataDir = path.resolve("fs", "testdata");
|
||||||
const isWindows = Deno.build.os === "windows";
|
|
||||||
|
|
||||||
Deno.test("ensureSymlinkIfItNotExist", async function (): Promise<void> {
|
Deno.test("ensureSymlinkIfItNotExist", async function (): Promise<void> {
|
||||||
const testDir = path.join(testdataDir, "link_file_1");
|
const testDir = path.join(testdataDir, "link_file_1");
|
||||||
|
@ -52,17 +51,7 @@ Deno.test("ensureSymlinkIfItExist", async function (): Promise<void> {
|
||||||
await Deno.mkdir(testDir, { recursive: true });
|
await Deno.mkdir(testDir, { recursive: true });
|
||||||
await Deno.writeFile(testFile, new Uint8Array());
|
await Deno.writeFile(testFile, new Uint8Array());
|
||||||
|
|
||||||
if (isWindows) {
|
await ensureSymlink(testFile, linkFile);
|
||||||
await assertThrowsAsync(
|
|
||||||
(): Promise<void> => ensureSymlink(testFile, linkFile),
|
|
||||||
Error,
|
|
||||||
"not implemented"
|
|
||||||
);
|
|
||||||
await Deno.remove(testDir, { recursive: true });
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
await ensureSymlink(testFile, linkFile);
|
|
||||||
}
|
|
||||||
|
|
||||||
const srcStat = await Deno.lstat(testFile);
|
const srcStat = await Deno.lstat(testFile);
|
||||||
const linkStat = await Deno.lstat(linkFile);
|
const linkStat = await Deno.lstat(linkFile);
|
||||||
|
@ -81,17 +70,7 @@ Deno.test("ensureSymlinkSyncIfItExist", function (): void {
|
||||||
Deno.mkdirSync(testDir, { recursive: true });
|
Deno.mkdirSync(testDir, { recursive: true });
|
||||||
Deno.writeFileSync(testFile, new Uint8Array());
|
Deno.writeFileSync(testFile, new Uint8Array());
|
||||||
|
|
||||||
if (isWindows) {
|
ensureSymlinkSync(testFile, linkFile);
|
||||||
assertThrows(
|
|
||||||
(): void => ensureSymlinkSync(testFile, linkFile),
|
|
||||||
Error,
|
|
||||||
"not implemented"
|
|
||||||
);
|
|
||||||
Deno.removeSync(testDir, { recursive: true });
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
ensureSymlinkSync(testFile, linkFile);
|
|
||||||
}
|
|
||||||
|
|
||||||
const srcStat = Deno.lstatSync(testFile);
|
const srcStat = Deno.lstatSync(testFile);
|
||||||
|
|
||||||
|
@ -111,17 +90,7 @@ Deno.test("ensureSymlinkDirectoryIfItExist", async function (): Promise<void> {
|
||||||
await Deno.mkdir(testDir, { recursive: true });
|
await Deno.mkdir(testDir, { recursive: true });
|
||||||
await Deno.writeFile(testFile, new Uint8Array());
|
await Deno.writeFile(testFile, new Uint8Array());
|
||||||
|
|
||||||
if (isWindows) {
|
await ensureSymlink(testDir, linkDir);
|
||||||
await assertThrowsAsync(
|
|
||||||
(): Promise<void> => ensureSymlink(testDir, linkDir),
|
|
||||||
Error,
|
|
||||||
"not implemented"
|
|
||||||
);
|
|
||||||
await Deno.remove(testDir, { recursive: true });
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
await ensureSymlink(testDir, linkDir);
|
|
||||||
}
|
|
||||||
|
|
||||||
const testDirStat = await Deno.lstat(testDir);
|
const testDirStat = await Deno.lstat(testDir);
|
||||||
const linkDirStat = await Deno.lstat(linkDir);
|
const linkDirStat = await Deno.lstat(linkDir);
|
||||||
|
@ -143,17 +112,7 @@ Deno.test("ensureSymlinkSyncDirectoryIfItExist", function (): void {
|
||||||
Deno.mkdirSync(testDir, { recursive: true });
|
Deno.mkdirSync(testDir, { recursive: true });
|
||||||
Deno.writeFileSync(testFile, new Uint8Array());
|
Deno.writeFileSync(testFile, new Uint8Array());
|
||||||
|
|
||||||
if (isWindows) {
|
ensureSymlinkSync(testDir, linkDir);
|
||||||
assertThrows(
|
|
||||||
(): void => ensureSymlinkSync(testDir, linkDir),
|
|
||||||
Error,
|
|
||||||
"not implemented"
|
|
||||||
);
|
|
||||||
Deno.removeSync(testDir, { recursive: true });
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
ensureSymlinkSync(testDir, linkDir);
|
|
||||||
}
|
|
||||||
|
|
||||||
const testDirStat = Deno.lstatSync(testDir);
|
const testDirStat = Deno.lstatSync(testDir);
|
||||||
const linkDirStat = Deno.lstatSync(linkDir);
|
const linkDirStat = Deno.lstatSync(linkDir);
|
||||||
|
|
Loading…
Reference in a new issue