deno.land / std@0.224.0 / fs / ensure_symlink.ts
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.import { dirname } from "../path/dirname.ts";import { resolve } from "../path/resolve.ts";import { ensureDir, ensureDirSync } from "./ensure_dir.ts";import { getFileInfoType } from "./_get_file_info_type.ts";import { toPathString } from "./_to_path_string.ts";
const isWindows = 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); }}
/** * Asynchronously ensures that the link exists, and points to a valid file. If * the directory structure does not exist, it is created. If the link already * exists, it is not modified but error is thrown if it is not point to the * given target. * * Requires the `--allow-read` and `--allow-write` flag. * * @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 * ```ts * import { ensureSymlink } from "https://deno.land/std@$STD_VERSION/fs/ensure_symlink.ts"; * * 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: Deno.SymlinkOptions | undefined = isWindows ? { type: srcFilePathType === "dir" ? "dir" : "file", } : undefined;
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 directory structure does not exist, it is created. If the link already * exists, it is not modified but error is thrown if it is not point to the * given target. * * Requires the `--allow-read` and `--allow-write` flag. * * @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 * ```ts * import { ensureSymlinkSync } from "https://deno.land/std@$STD_VERSION/fs/ensure_symlink.ts"; * * 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: Deno.SymlinkOptions | undefined = isWindows ? { type: srcFilePathType === "dir" ? "dir" : "file", } : undefined;
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}`, ); } }}
Version Info