2023-02-14 11:38:45 -05:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
|
|
|
|
|
|
|
import { TextEncoder } from "internal:deno_web/08_text_encoding.js";
|
|
|
|
import {
|
|
|
|
intoCallbackAPIWithIntercept,
|
|
|
|
MaybeEmpty,
|
|
|
|
notImplemented,
|
refactor(core): include_js_files! 'dir' option doesn't change specifiers (#18019)
This commit changes "include_js_files!" macro from "deno_core"
in a way that "dir" option doesn't cause specifiers to be rewritten
to include it.
Example:
```
include_js_files! {
dir "js",
"hello.js",
}
```
The above definition required embedders to use:
`import ... from "internal:<ext_name>/js/hello.js"`.
But with this change, the "js" directory in which the files are stored
is an implementation detail, which for embedders results in:
`import ... from "internal:<ext_name>/hello.js"`.
The directory the files are stored in, is an implementation detail and
in some cases might result in a significant size difference for the
snapshot. As an example, in "deno_node" extension, we store the
source code in "polyfills" directory; which resulted in each specifier
to look like "internal:deno_node/polyfills/<module_name>", but with
this change it's "internal:deno_node/<module_name>".
Given that "deno_node" has over 100 files, many of them having
several import specifiers to the same extension, this change removes
10 characters from each import specifier.
2023-03-04 21:31:38 -05:00
|
|
|
} from "internal:deno_node/_utils.ts";
|
|
|
|
import { fromFileUrl } from "internal:deno_node/path.ts";
|
|
|
|
import { promisify } from "internal:deno_node/internal/util.mjs";
|
2023-02-14 11:38:45 -05: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,
|
|
|
|
) {
|
|
|
|
path = path instanceof URL ? fromFileUrl(path) : path;
|
|
|
|
|
|
|
|
let cb: ReadlinkCallback | undefined;
|
|
|
|
if (typeof optOrCallback === "function") {
|
|
|
|
cb = optOrCallback;
|
|
|
|
} else {
|
|
|
|
cb = callback;
|
|
|
|
}
|
|
|
|
|
|
|
|
const encoding = getEncoding(optOrCallback);
|
|
|
|
|
|
|
|
intoCallbackAPIWithIntercept<string, Uint8Array | string>(
|
|
|
|
Deno.readLink,
|
|
|
|
(data: string): string | Uint8Array => maybeEncode(data, encoding),
|
|
|
|
cb,
|
|
|
|
path,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export const readlinkPromise = promisify(readlink) as (
|
|
|
|
path: string | URL,
|
|
|
|
opt?: ReadlinkOptions,
|
|
|
|
) => Promise<string | Uint8Array>;
|
|
|
|
|
|
|
|
export function readlinkSync(
|
|
|
|
path: string | URL,
|
|
|
|
opt?: ReadlinkOptions,
|
|
|
|
): string | Uint8Array {
|
|
|
|
path = path instanceof URL ? fromFileUrl(path) : path;
|
|
|
|
|
|
|
|
return maybeEncode(Deno.readLinkSync(path), getEncoding(opt));
|
|
|
|
}
|