deno/ext/node/polyfills/_fs/_fs_mkdtemp.ts
Daniel Osvaldo Rahmanto e60f5d2c52
Some checks are pending
ci / test release linux-x86_64 (push) Blocked by required conditions
ci / test release macos-x86_64 (push) Blocked by required conditions
ci / test debug windows-x86_64 (push) Blocked by required conditions
ci / test release windows-x86_64 (push) Blocked by required conditions
ci / build libs (push) Blocked by required conditions
ci / pre-build (push) Waiting to run
ci / test debug linux-aarch64 (push) Blocked by required conditions
ci / test release linux-aarch64 (push) Blocked by required conditions
ci / test debug macos-aarch64 (push) Blocked by required conditions
ci / test release macos-aarch64 (push) Blocked by required conditions
ci / bench release linux-x86_64 (push) Blocked by required conditions
ci / lint debug linux-x86_64 (push) Blocked by required conditions
ci / lint debug macos-x86_64 (push) Blocked by required conditions
ci / lint debug windows-x86_64 (push) Blocked by required conditions
ci / test debug linux-x86_64 (push) Blocked by required conditions
ci / test debug macos-x86_64 (push) Blocked by required conditions
ci / publish canary (push) Blocked by required conditions
fix(ext/node): fs.mkdtemp and fs.mkdtempSync compatibility (#30602)
`fs.mkdtemp` and `fs.mkdtempSync` now accept `Buffer` and `Uint8Array`
path. The implementation has been moved to Rust, including directory
suffix generation and directory creation.
2025-09-05 16:12:42 -04:00

150 lines
4.5 KiB
TypeScript

// Copyright 2018-2025 the Deno authors. MIT license.
// Copyright Node.js contributors. All rights reserved. MIT License.
import { normalizeEncoding, promisify } from "ext:deno_node/internal/util.mjs";
import { primordials } from "ext:core/mod.js";
import { makeCallback } from "ext:deno_node/_fs/_fs_common.ts";
import { Buffer } from "node:buffer";
import {
getValidatedPathToString,
warnOnNonPortableTemplate,
} from "ext:deno_node/internal/fs/utils.mjs";
import {
denoErrorToNodeError,
ERR_INVALID_ARG_TYPE,
} from "ext:deno_node/internal/errors.ts";
import { op_node_mkdtemp, op_node_mkdtemp_sync } from "ext:core/ops";
import type { Encoding } from "node:crypto";
const { PromisePrototypeThen } = primordials;
export type MkdtempCallback = (
err: Error | null,
directory?: string,
) => void;
export type MkdtempBufferCallback = (
err: Error | null,
directory?: Buffer<ArrayBufferLike>,
) => void;
type MkdTempPromise = (
prefix: string | Buffer | Uint8Array | URL,
options?: { encoding: string } | string,
) => Promise<string>;
type MkdTempPromiseBuffer = (
prefix: string | Buffer | Uint8Array | URL,
options: { encoding: "buffer" } | "buffer",
) => Promise<Buffer<ArrayBufferLike>>;
// https://nodejs.org/dist/latest-v15.x/docs/api/fs.html#fs_fs_mkdtemp_prefix_options_callback
export function mkdtemp(
prefix: string | Buffer | Uint8Array | URL,
callback: MkdtempCallback,
): void;
export function mkdtemp(
prefix: string | Buffer | Uint8Array | URL,
options: { encoding: "buffer" } | "buffer",
callback: MkdtempBufferCallback,
): void;
export function mkdtemp(
prefix: string | Buffer | Uint8Array | URL,
options: { encoding: string } | string,
callback: MkdtempCallback,
): void;
export function mkdtemp(
prefix: string | Buffer | Uint8Array | URL,
options: { encoding: string } | string | MkdtempCallback | undefined,
callback?: MkdtempCallback | MkdtempBufferCallback,
) {
if (typeof options === "function") {
callback = options;
options = undefined;
}
callback = makeCallback(callback);
const encoding = parseEncoding(options);
prefix = getValidatedPathToString(prefix, "prefix");
warnOnNonPortableTemplate(prefix);
PromisePrototypeThen(
op_node_mkdtemp(prefix),
(path: string) => callback(null, decode(path, encoding)),
(err: Error) =>
callback(denoErrorToNodeError(err, {
syscall: "mkdtemp",
path: `${prefix}XXXXXX`,
})),
);
}
export const mkdtempPromise = promisify(mkdtemp) as
| MkdTempPromise
| MkdTempPromiseBuffer;
// https://nodejs.org/dist/latest-v15.x/docs/api/fs.html#fs_fs_mkdtempsync_prefix_options
export function mkdtempSync(
prefix: string | Buffer | Uint8Array | URL,
options?: { encoding: "buffer" } | "buffer",
): Buffer<ArrayBufferLike>;
export function mkdtempSync(
prefix: string | Buffer | Uint8Array | URL,
options?: { encoding: string } | string,
): string;
export function mkdtempSync(
prefix: string | Buffer | Uint8Array | URL,
options?: { encoding: string } | string,
): string | Buffer<ArrayBufferLike> {
const encoding = parseEncoding(options);
prefix = getValidatedPathToString(prefix, "prefix");
warnOnNonPortableTemplate(prefix);
try {
const path = op_node_mkdtemp_sync(prefix) as string;
return decode(path, encoding);
} catch (err) {
throw denoErrorToNodeError(err as Error, {
syscall: "mkdtemp",
path: `${prefix}XXXXXX`,
});
}
}
function decode(str: string, encoding: Encoding): string;
function decode(str: string, encoding: "buffer"): Buffer<ArrayBufferLike>;
function decode(
str: string,
encoding: Encoding | "buffer",
): string | Buffer<ArrayBufferLike> {
if (encoding === "utf8") return str;
const buffer = Buffer.from(str);
if (encoding === "buffer") return buffer;
// deno-lint-ignore prefer-primordials
return buffer.toString(encoding);
}
function parseEncoding(
options: string | { encoding?: string } | undefined,
): Encoding | "buffer" {
let encoding: string | undefined;
if (typeof options === "undefined" || options === null) {
encoding = "utf8";
} else if (typeof options === "string") {
encoding = options;
} else if (typeof options === "object") {
encoding = options.encoding ?? "utf8";
} else {
throw new ERR_INVALID_ARG_TYPE("options", ["string", "Object"], options);
}
if (encoding === "buffer") {
return encoding;
}
const parsedEncoding = normalizeEncoding(encoding);
if (!parsedEncoding) {
throw new ERR_INVALID_ARG_TYPE("encoding", encoding, "is invalid encoding");
}
return parsedEncoding;
}