2023-02-14 17:38:45 +01:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
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 22:31:38 -04:00
|
|
|
import { validateFunction } from "internal:deno_node/internal/validators.mjs";
|
|
|
|
import { normalizeEncoding, slowCases } from "internal:deno_node/internal/normalize_encoding.mjs";
|
2023-02-14 17:38:45 +01:00
|
|
|
export { normalizeEncoding, slowCases };
|
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 22:31:38 -04:00
|
|
|
import { ObjectCreate, StringPrototypeToUpperCase } from "internal:deno_node/internal/primordials.mjs";
|
|
|
|
import { ERR_UNKNOWN_SIGNAL } from "internal:deno_node/internal/errors.ts";
|
|
|
|
import { os } from "internal:deno_node/internal_binding/constants.ts";
|
2023-02-14 17:38:45 +01:00
|
|
|
|
|
|
|
export const customInspectSymbol = Symbol.for("nodejs.util.inspect.custom");
|
|
|
|
export const kEnumerableProperty = Object.create(null);
|
|
|
|
kEnumerableProperty.enumerable = true;
|
|
|
|
|
|
|
|
export const kEmptyObject = Object.freeze(Object.create(null));
|
|
|
|
|
|
|
|
export function once(callback) {
|
|
|
|
let called = false;
|
|
|
|
return function (...args) {
|
|
|
|
if (called) return;
|
|
|
|
called = true;
|
|
|
|
Reflect.apply(callback, this, args);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function createDeferredPromise() {
|
|
|
|
let resolve;
|
|
|
|
let reject;
|
|
|
|
const promise = new Promise((res, rej) => {
|
|
|
|
resolve = res;
|
|
|
|
reject = rej;
|
|
|
|
});
|
|
|
|
|
|
|
|
return { promise, resolve, reject };
|
|
|
|
}
|
|
|
|
|
|
|
|
// In addition to being accessible through util.promisify.custom,
|
|
|
|
// this symbol is registered globally and can be accessed in any environment as
|
|
|
|
// Symbol.for('nodejs.util.promisify.custom').
|
|
|
|
const kCustomPromisifiedSymbol = Symbol.for("nodejs.util.promisify.custom");
|
|
|
|
// This is an internal Node symbol used by functions returning multiple
|
|
|
|
// arguments, e.g. ['bytesRead', 'buffer'] for fs.read().
|
|
|
|
const kCustomPromisifyArgsSymbol = Symbol.for(
|
|
|
|
"nodejs.util.promisify.customArgs",
|
|
|
|
);
|
|
|
|
|
|
|
|
export const customPromisifyArgs = kCustomPromisifyArgsSymbol;
|
|
|
|
|
|
|
|
export function promisify(
|
|
|
|
original,
|
|
|
|
) {
|
|
|
|
validateFunction(original, "original");
|
|
|
|
if (original[kCustomPromisifiedSymbol]) {
|
|
|
|
const fn = original[kCustomPromisifiedSymbol];
|
|
|
|
|
|
|
|
validateFunction(fn, "util.promisify.custom");
|
|
|
|
|
|
|
|
return Object.defineProperty(fn, kCustomPromisifiedSymbol, {
|
|
|
|
value: fn,
|
|
|
|
enumerable: false,
|
|
|
|
writable: false,
|
|
|
|
configurable: true,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Names to create an object from in case the callback receives multiple
|
|
|
|
// arguments, e.g. ['bytesRead', 'buffer'] for fs.read.
|
|
|
|
const argumentNames = original[kCustomPromisifyArgsSymbol];
|
|
|
|
function fn(...args) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
args.push((err, ...values) => {
|
|
|
|
if (err) {
|
|
|
|
return reject(err);
|
|
|
|
}
|
|
|
|
if (argumentNames !== undefined && values.length > 1) {
|
|
|
|
const obj = {};
|
|
|
|
for (let i = 0; i < argumentNames.length; i++) {
|
|
|
|
obj[argumentNames[i]] = values[i];
|
|
|
|
}
|
|
|
|
resolve(obj);
|
|
|
|
} else {
|
|
|
|
resolve(values[0]);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
Reflect.apply(original, this, args);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Object.setPrototypeOf(fn, Object.getPrototypeOf(original));
|
|
|
|
|
|
|
|
Object.defineProperty(fn, kCustomPromisifiedSymbol, {
|
|
|
|
value: fn,
|
|
|
|
enumerable: false,
|
|
|
|
writable: false,
|
|
|
|
configurable: true,
|
|
|
|
});
|
|
|
|
return Object.defineProperties(
|
|
|
|
fn,
|
|
|
|
Object.getOwnPropertyDescriptors(original),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
let signalsToNamesMapping;
|
|
|
|
function getSignalsToNamesMapping() {
|
|
|
|
if (signalsToNamesMapping !== undefined) {
|
|
|
|
return signalsToNamesMapping;
|
|
|
|
}
|
|
|
|
|
|
|
|
signalsToNamesMapping = ObjectCreate(null);
|
|
|
|
for (const key in os.signals) {
|
|
|
|
signalsToNamesMapping[os.signals[key]] = key;
|
|
|
|
}
|
|
|
|
|
|
|
|
return signalsToNamesMapping;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function convertToValidSignal(signal) {
|
|
|
|
if (typeof signal === "number" && getSignalsToNamesMapping()[signal]) {
|
|
|
|
return signal;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof signal === "string") {
|
|
|
|
const signalName = os.signals[StringPrototypeToUpperCase(signal)];
|
|
|
|
if (signalName) return signalName;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new ERR_UNKNOWN_SIGNAL(signal);
|
|
|
|
}
|
|
|
|
|
|
|
|
promisify.custom = kCustomPromisifiedSymbol;
|
|
|
|
|
|
|
|
export default {
|
|
|
|
convertToValidSignal,
|
|
|
|
createDeferredPromise,
|
|
|
|
customInspectSymbol,
|
|
|
|
customPromisifyArgs,
|
|
|
|
kEmptyObject,
|
|
|
|
kEnumerableProperty,
|
|
|
|
normalizeEncoding,
|
|
|
|
once,
|
|
|
|
promisify,
|
|
|
|
slowCases,
|
|
|
|
};
|