2024-01-01 14:58:21 -05:00
|
|
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
2023-02-14 17:38:45 +01:00
|
|
|
|
2023-06-27 15:18:22 +09:00
|
|
|
// TODO(petamoriken): enable prefer-primordials for node polyfills
|
|
|
|
// deno-lint-ignore-file prefer-primordials
|
|
|
|
|
2023-03-08 07:44:54 -04:00
|
|
|
import { TextEncoder } from "ext:deno_web/08_text_encoding.js";
|
2024-10-18 10:11:06 -07:00
|
|
|
import { MaybeEmpty, notImplemented } from "ext:deno_node/_utils.ts";
|
2023-06-16 06:43:59 -04:00
|
|
|
import { pathFromURL } from "ext:deno_web/00_infra.js";
|
2023-03-08 07:44:54 -04:00
|
|
|
import { promisify } from "ext:deno_node/internal/util.mjs";
|
2024-10-18 10:11:06 -07:00
|
|
|
import { denoErrorToNodeError } from "ext:deno_node/internal/errors.ts";
|
2023-02-14 17:38:45 +01:00
|
|
|
|
|
|
|
type ReadlinkCallback = (
|
|
|
|
err: MaybeEmpty<Error>,
|
|
|
|
linkString: MaybeEmpty<string | Uint8Array>,
|
|
|
|
) => void;
|
|
|
|
|
|
|
|
interface ReadlinkOptions {
|
|
|
|
encoding?: string | null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function maybeEncode(
|
|
|
|
data: string,
|
|
|
|
encoding: string | null,
|
|
|
|
): string | Uint8Array {
|
|
|
|
if (encoding === "buffer") {
|
|
|
|
return new TextEncoder().encode(data);
|
|
|
|
}
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getEncoding(
|
|
|
|
optOrCallback?: ReadlinkOptions | ReadlinkCallback,
|
|
|
|
): string | null {
|
|
|
|
if (!optOrCallback || typeof optOrCallback === "function") {
|
|
|
|
return null;
|
|
|
|
} else {
|
|
|
|
if (optOrCallback.encoding) {
|
|
|
|
if (
|
|
|
|
optOrCallback.encoding === "utf8" ||
|
|
|
|
optOrCallback.encoding === "utf-8"
|
|
|
|
) {
|
|
|
|
return "utf8";
|
|
|
|
} else if (optOrCallback.encoding === "buffer") {
|
|
|
|
return "buffer";
|
|
|
|
} else {
|
|
|
|
notImplemented(`fs.readlink encoding=${optOrCallback.encoding}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function readlink(
|
|
|
|
path: string | URL,
|
|
|
|
optOrCallback: ReadlinkCallback | ReadlinkOptions,
|
|
|
|
callback?: ReadlinkCallback,
|
|
|
|
) {
|
2023-06-16 06:43:59 -04:00
|
|
|
path = path instanceof URL ? pathFromURL(path) : path;
|
2023-02-14 17:38:45 +01:00
|
|
|
|
|
|
|
let cb: ReadlinkCallback | undefined;
|
|
|
|
if (typeof optOrCallback === "function") {
|
|
|
|
cb = optOrCallback;
|
|
|
|
} else {
|
|
|
|
cb = callback;
|
|
|
|
}
|
|
|
|
|
|
|
|
const encoding = getEncoding(optOrCallback);
|
|
|
|
|
2024-10-18 10:11:06 -07:00
|
|
|
Deno.readLink(path).then((data: string) => {
|
|
|
|
const res = maybeEncode(data, encoding);
|
|
|
|
if (cb) cb(null, res);
|
|
|
|
}, (err: Error) => {
|
|
|
|
if (cb) {
|
|
|
|
(cb as (e: Error) => void)(denoErrorToNodeError(err, {
|
|
|
|
syscall: "readlink",
|
|
|
|
path,
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
});
|
2023-02-14 17:38:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export const readlinkPromise = promisify(readlink) as (
|
|
|
|
path: string | URL,
|
|
|
|
opt?: ReadlinkOptions,
|
|
|
|
) => Promise<string | Uint8Array>;
|
|
|
|
|
|
|
|
export function readlinkSync(
|
|
|
|
path: string | URL,
|
|
|
|
opt?: ReadlinkOptions,
|
|
|
|
): string | Uint8Array {
|
2023-06-16 06:43:59 -04:00
|
|
|
path = path instanceof URL ? pathFromURL(path) : path;
|
2023-02-14 17:38:45 +01:00
|
|
|
|
2024-10-18 10:11:06 -07:00
|
|
|
try {
|
|
|
|
return maybeEncode(Deno.readLinkSync(path), getEncoding(opt));
|
|
|
|
} catch (error) {
|
|
|
|
throw denoErrorToNodeError(error, {
|
|
|
|
syscall: "readlink",
|
|
|
|
path,
|
|
|
|
});
|
|
|
|
}
|
2023-02-14 17:38:45 +01:00
|
|
|
}
|