mirror of
https://github.com/denoland/std.git
synced 2024-11-21 20:50:22 +00:00
c0427a02bb
Co-authored-by: Yoshiya Hinosawa <stibium121@gmail.com>
144 lines
4.9 KiB
TypeScript
144 lines
4.9 KiB
TypeScript
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
import { dirname } from "@std/path/dirname";
|
|
import { resolve } from "@std/path/resolve";
|
|
import { ensureDir, ensureDirSync } from "./ensure_dir.ts";
|
|
import { getFileInfoType, type PathType } from "./_get_file_info_type.ts";
|
|
import { toPathString } from "./_to_path_string.ts";
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
const isWindows = (globalThis as any).Deno?.build.os === "windows";
|
|
|
|
function resolveSymlinkTarget(target: string | URL, linkName: string | URL) {
|
|
if (typeof target !== "string") return target; // URL is always absolute path
|
|
if (typeof linkName === "string") {
|
|
return resolve(dirname(linkName), target);
|
|
} else {
|
|
return new URL(target, linkName);
|
|
}
|
|
}
|
|
|
|
function getSymlinkOption(
|
|
type: PathType | undefined,
|
|
): Deno.SymlinkOptions | undefined {
|
|
return isWindows ? { type: type === "dir" ? "dir" : "file" } : undefined;
|
|
}
|
|
|
|
/**
|
|
* Asynchronously ensures that the link exists, and points to a valid file.
|
|
*
|
|
* If the parent directories for the link do not exist, they are created. If the
|
|
* link already exists, and it is not modified, this function does nothing. If
|
|
* the link already exists, and it does not point to the given target, an error
|
|
* is thrown.
|
|
*
|
|
* 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 target The source file path as a string or URL.
|
|
* @param linkName The destination link path as a string or URL.
|
|
*
|
|
* @returns A void promise that resolves once the link exists.
|
|
*
|
|
* @example Usage
|
|
* ```ts ignore
|
|
* import { ensureSymlink } from "@std/fs/ensure-symlink";
|
|
*
|
|
* await ensureSymlink("./folder/targetFile.dat", "./folder/targetFile.link.dat");
|
|
* ```
|
|
*/
|
|
export async function ensureSymlink(
|
|
target: string | URL,
|
|
linkName: string | URL,
|
|
) {
|
|
const targetRealPath = resolveSymlinkTarget(target, linkName);
|
|
const srcStatInfo = await Deno.lstat(targetRealPath);
|
|
const srcFilePathType = getFileInfoType(srcStatInfo);
|
|
|
|
await ensureDir(dirname(toPathString(linkName)));
|
|
|
|
const options = getSymlinkOption(srcFilePathType);
|
|
|
|
try {
|
|
await Deno.symlink(target, linkName, options);
|
|
} catch (error) {
|
|
if (!(error instanceof Deno.errors.AlreadyExists)) {
|
|
throw error;
|
|
}
|
|
const linkStatInfo = await Deno.lstat(linkName);
|
|
if (!linkStatInfo.isSymlink) {
|
|
const type = getFileInfoType(linkStatInfo);
|
|
throw new Deno.errors.AlreadyExists(
|
|
`A '${type}' already exists at the path: ${linkName}`,
|
|
);
|
|
}
|
|
const linkPath = await Deno.readLink(linkName);
|
|
const linkRealPath = resolve(linkPath);
|
|
if (linkRealPath !== targetRealPath) {
|
|
throw new Deno.errors.AlreadyExists(
|
|
`A symlink targeting to an undesired path already exists: ${linkName} -> ${linkRealPath}`,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Synchronously ensures that the link exists, and points to a valid file.
|
|
*
|
|
* If the parent directories for the link do not exist, they are created. If the
|
|
* link already exists, and it is not modified, this function does nothing. If
|
|
* the link already exists, and it does not point to the given target, an error
|
|
* is thrown.
|
|
*
|
|
* 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 target The source file path as a string or URL.
|
|
* @param linkName The destination link path as a string or URL.
|
|
* @returns A void value that returns once the link exists.
|
|
*
|
|
* @example Usage
|
|
* ```ts ignore
|
|
* import { ensureSymlinkSync } from "@std/fs/ensure-symlink";
|
|
*
|
|
* ensureSymlinkSync("./folder/targetFile.dat", "./folder/targetFile.link.dat");
|
|
* ```
|
|
*/
|
|
export function ensureSymlinkSync(
|
|
target: string | URL,
|
|
linkName: string | URL,
|
|
) {
|
|
const targetRealPath = resolveSymlinkTarget(target, linkName);
|
|
const srcStatInfo = Deno.lstatSync(targetRealPath);
|
|
const srcFilePathType = getFileInfoType(srcStatInfo);
|
|
|
|
ensureDirSync(dirname(toPathString(linkName)));
|
|
|
|
const options = getSymlinkOption(srcFilePathType);
|
|
|
|
try {
|
|
Deno.symlinkSync(target, linkName, options);
|
|
} catch (error) {
|
|
if (!(error instanceof Deno.errors.AlreadyExists)) {
|
|
throw error;
|
|
}
|
|
const linkStatInfo = Deno.lstatSync(linkName);
|
|
if (!linkStatInfo.isSymlink) {
|
|
const type = getFileInfoType(linkStatInfo);
|
|
throw new Deno.errors.AlreadyExists(
|
|
`A '${type}' already exists at the path: ${linkName}`,
|
|
);
|
|
}
|
|
const linkPath = Deno.readLinkSync(linkName);
|
|
const linkRealPath = resolve(linkPath);
|
|
if (linkRealPath !== targetRealPath) {
|
|
throw new Deno.errors.AlreadyExists(
|
|
`A symlink targeting to an undesired path already exists: ${linkName} -> ${linkRealPath}`,
|
|
);
|
|
}
|
|
}
|
|
}
|