2023-02-14 11:38:45 -05:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
|
|
|
// Copyright Node.js contributors. All rights reserved. MIT License.
|
|
|
|
|
|
|
|
import {
|
|
|
|
TextDecoder,
|
|
|
|
TextEncoder,
|
|
|
|
} from "internal:deno_web/08_text_encoding.js";
|
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
|
|
|
import { existsSync } from "internal:deno_node/_fs/_fs_exists.ts";
|
|
|
|
import { mkdir, mkdirSync } from "internal:deno_node/_fs/_fs_mkdir.ts";
|
2023-02-14 11:38:45 -05:00
|
|
|
import {
|
|
|
|
ERR_INVALID_ARG_TYPE,
|
|
|
|
ERR_INVALID_OPT_VALUE_ENCODING,
|
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/internal/errors.ts";
|
|
|
|
import { promisify } from "internal:deno_node/internal/util.mjs";
|
2023-02-14 11:38:45 -05:00
|
|
|
|
|
|
|
export type mkdtempCallback = (
|
|
|
|
err: Error | null,
|
|
|
|
directory?: string,
|
|
|
|
) => void;
|
|
|
|
|
|
|
|
// https://nodejs.org/dist/latest-v15.x/docs/api/fs.html#fs_fs_mkdtemp_prefix_options_callback
|
|
|
|
export function mkdtemp(prefix: string, callback: mkdtempCallback): void;
|
|
|
|
export function mkdtemp(
|
|
|
|
prefix: string,
|
|
|
|
options: { encoding: string } | string,
|
|
|
|
callback: mkdtempCallback,
|
|
|
|
): void;
|
|
|
|
export function mkdtemp(
|
|
|
|
prefix: string,
|
|
|
|
optionsOrCallback: { encoding: string } | string | mkdtempCallback,
|
|
|
|
maybeCallback?: mkdtempCallback,
|
|
|
|
) {
|
|
|
|
const callback: mkdtempCallback | undefined =
|
|
|
|
typeof optionsOrCallback == "function" ? optionsOrCallback : maybeCallback;
|
|
|
|
if (!callback) {
|
|
|
|
throw new ERR_INVALID_ARG_TYPE("callback", "function", callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
const encoding: string | undefined = parseEncoding(optionsOrCallback);
|
|
|
|
const path = tempDirPath(prefix);
|
|
|
|
|
|
|
|
mkdir(
|
|
|
|
path,
|
|
|
|
{ recursive: false, mode: 0o700 },
|
|
|
|
(err: Error | null | undefined) => {
|
|
|
|
if (err) callback(err);
|
|
|
|
else callback(null, decode(path, encoding));
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export const mkdtempPromise = promisify(mkdtemp) as (
|
|
|
|
prefix: string,
|
|
|
|
options?: { encoding: string } | string,
|
|
|
|
) => Promise<string>;
|
|
|
|
|
|
|
|
// https://nodejs.org/dist/latest-v15.x/docs/api/fs.html#fs_fs_mkdtempsync_prefix_options
|
|
|
|
export function mkdtempSync(
|
|
|
|
prefix: string,
|
|
|
|
options?: { encoding: string } | string,
|
|
|
|
): string {
|
|
|
|
const encoding: string | undefined = parseEncoding(options);
|
|
|
|
const path = tempDirPath(prefix);
|
|
|
|
|
|
|
|
mkdirSync(path, { recursive: false, mode: 0o700 });
|
|
|
|
return decode(path, encoding);
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseEncoding(
|
|
|
|
optionsOrCallback?: { encoding: string } | string | mkdtempCallback,
|
|
|
|
): string | undefined {
|
|
|
|
let encoding: string | undefined;
|
|
|
|
if (typeof optionsOrCallback == "function") encoding = undefined;
|
|
|
|
else if (optionsOrCallback instanceof Object) {
|
|
|
|
encoding = optionsOrCallback?.encoding;
|
|
|
|
} else encoding = optionsOrCallback;
|
|
|
|
|
|
|
|
if (encoding) {
|
|
|
|
try {
|
|
|
|
new TextDecoder(encoding);
|
|
|
|
} catch {
|
|
|
|
throw new ERR_INVALID_OPT_VALUE_ENCODING(encoding);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return encoding;
|
|
|
|
}
|
|
|
|
|
|
|
|
function decode(str: string, encoding?: string): string {
|
|
|
|
if (!encoding) return str;
|
|
|
|
else {
|
|
|
|
const decoder = new TextDecoder(encoding);
|
|
|
|
const encoder = new TextEncoder();
|
|
|
|
return decoder.decode(encoder.encode(str));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const CHARS = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
|
|
|
function randomName(): string {
|
|
|
|
return [...Array(6)].map(() =>
|
|
|
|
CHARS[Math.floor(Math.random() * CHARS.length)]
|
|
|
|
).join("");
|
|
|
|
}
|
|
|
|
|
|
|
|
function tempDirPath(prefix: string): string {
|
|
|
|
let path: string;
|
|
|
|
do {
|
|
|
|
path = prefix + randomName();
|
|
|
|
} while (existsSync(path));
|
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|