mirror of
https://github.com/denoland/deno.git
synced 2025-01-12 00:54:02 -05:00
feat(node_compat): Add a read method to the FileHandle class (#19359)
ref: #19165 The FileHandle class has many missing methods compared to node.
This commit is contained in:
parent
0197f42e6b
commit
262571e63e
4 changed files with 97 additions and 12 deletions
|
@ -1,15 +1,16 @@
|
||||||
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
||||||
import * as path from "../../../../test_util/std/path/mod.ts";
|
import * as path from "../../../../test_util/std/path/mod.ts";
|
||||||
|
import * as fs from "node:fs/promises";
|
||||||
import {
|
import {
|
||||||
assert,
|
assert,
|
||||||
assertEquals,
|
assertEquals,
|
||||||
} from "../../../../test_util/std/testing/asserts.ts";
|
} from "../../../../test_util/std/testing/asserts.ts";
|
||||||
|
import { Buffer } from "node:buffer";
|
||||||
|
|
||||||
const moduleDir = path.dirname(path.fromFileUrl(import.meta.url));
|
const moduleDir = path.dirname(path.fromFileUrl(import.meta.url));
|
||||||
const testData = path.resolve(moduleDir, "testdata", "hello.txt");
|
const testData = path.resolve(moduleDir, "testdata", "hello.txt");
|
||||||
|
|
||||||
Deno.test("readFileSuccess", async function () {
|
Deno.test("readFileSuccess", async function () {
|
||||||
const fs = await import("node:fs/promises");
|
|
||||||
const fileHandle = await fs.open(testData);
|
const fileHandle = await fs.open(testData);
|
||||||
const data = await fileHandle.readFile();
|
const data = await fileHandle.readFile();
|
||||||
|
|
||||||
|
@ -18,3 +19,42 @@ Deno.test("readFileSuccess", async function () {
|
||||||
|
|
||||||
await fileHandle.close();
|
await fileHandle.close();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
Deno.test("read", async function () {
|
||||||
|
const fileHandle = await fs.open(testData);
|
||||||
|
const byteLength = "hello world".length;
|
||||||
|
|
||||||
|
const buf = new Buffer(byteLength);
|
||||||
|
await fileHandle.read(buf, 0, byteLength, 0);
|
||||||
|
|
||||||
|
assertEquals(new TextDecoder().decode(buf as Uint8Array), "hello world");
|
||||||
|
|
||||||
|
await fileHandle.close();
|
||||||
|
});
|
||||||
|
|
||||||
|
Deno.test("read specify opt", async function () {
|
||||||
|
const fileHandle = await fs.open(testData);
|
||||||
|
const byteLength = "hello world".length;
|
||||||
|
|
||||||
|
const opt = {
|
||||||
|
buffer: new Buffer(byteLength),
|
||||||
|
offset: 6,
|
||||||
|
length: 5,
|
||||||
|
};
|
||||||
|
let res = await fileHandle.read(opt);
|
||||||
|
|
||||||
|
assertEquals(res.bytesRead, byteLength);
|
||||||
|
assertEquals(new TextDecoder().decode(res.buffer as Uint8Array), "world");
|
||||||
|
|
||||||
|
const opt2 = {
|
||||||
|
buffer: new Buffer(byteLength),
|
||||||
|
length: 5,
|
||||||
|
position: 0,
|
||||||
|
};
|
||||||
|
res = await fileHandle.read(opt2);
|
||||||
|
|
||||||
|
assertEquals(res.bytesRead, byteLength);
|
||||||
|
assertEquals(new TextDecoder().decode(res.buffer as Uint8Array), "hello");
|
||||||
|
|
||||||
|
await fileHandle.close();
|
||||||
|
});
|
||||||
|
|
|
@ -33,6 +33,13 @@ export type BinaryOptionsArgument =
|
||||||
| ({ encoding: BinaryEncodings } & FileOptions);
|
| ({ encoding: BinaryEncodings } & FileOptions);
|
||||||
export type FileOptionsArgument = Encodings | FileOptions;
|
export type FileOptionsArgument = Encodings | FileOptions;
|
||||||
|
|
||||||
|
export type ReadOptions = {
|
||||||
|
buffer: Buffer | Uint8Array;
|
||||||
|
offset: number;
|
||||||
|
length: number;
|
||||||
|
position: number | null;
|
||||||
|
};
|
||||||
|
|
||||||
export interface WriteFileOptions extends FileOptions {
|
export interface WriteFileOptions extends FileOptions {
|
||||||
mode?: number;
|
mode?: number;
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,7 @@ import { Buffer } from "ext:deno_node/buffer.ts";
|
||||||
import { ERR_INVALID_ARG_TYPE } from "ext:deno_node/internal/errors.ts";
|
import { ERR_INVALID_ARG_TYPE } from "ext:deno_node/internal/errors.ts";
|
||||||
import * as io from "ext:deno_io/12_io.js";
|
import * as io from "ext:deno_io/12_io.js";
|
||||||
import * as fs from "ext:deno_fs/30_fs.js";
|
import * as fs from "ext:deno_fs/30_fs.js";
|
||||||
|
import { ReadOptions } from "ext:deno_node/_fs/_fs_common.ts";
|
||||||
import {
|
import {
|
||||||
validateOffsetLengthRead,
|
validateOffsetLengthRead,
|
||||||
validatePosition,
|
validatePosition,
|
||||||
|
@ -12,13 +13,6 @@ import {
|
||||||
validateInteger,
|
validateInteger,
|
||||||
} from "ext:deno_node/internal/validators.mjs";
|
} from "ext:deno_node/internal/validators.mjs";
|
||||||
|
|
||||||
type readOptions = {
|
|
||||||
buffer: Buffer | Uint8Array;
|
|
||||||
offset: number;
|
|
||||||
length: number;
|
|
||||||
position: number | null;
|
|
||||||
};
|
|
||||||
|
|
||||||
type readSyncOptions = {
|
type readSyncOptions = {
|
||||||
offset: number;
|
offset: number;
|
||||||
length: number;
|
length: number;
|
||||||
|
@ -35,7 +29,7 @@ type Callback = BinaryCallback;
|
||||||
export function read(fd: number, callback: Callback): void;
|
export function read(fd: number, callback: Callback): void;
|
||||||
export function read(
|
export function read(
|
||||||
fd: number,
|
fd: number,
|
||||||
options: readOptions,
|
options: ReadOptions,
|
||||||
callback: Callback,
|
callback: Callback,
|
||||||
): void;
|
): void;
|
||||||
export function read(
|
export function read(
|
||||||
|
@ -48,7 +42,7 @@ export function read(
|
||||||
): void;
|
): void;
|
||||||
export function read(
|
export function read(
|
||||||
fd: number,
|
fd: number,
|
||||||
optOrBufferOrCb?: Buffer | Uint8Array | readOptions | Callback,
|
optOrBufferOrCb?: Buffer | Uint8Array | ReadOptions | Callback,
|
||||||
offsetOrCallback?: number | Callback,
|
offsetOrCallback?: number | Callback,
|
||||||
length?: number,
|
length?: number,
|
||||||
position?: number | null,
|
position?: number | null,
|
||||||
|
@ -86,7 +80,7 @@ export function read(
|
||||||
length = buffer.byteLength;
|
length = buffer.byteLength;
|
||||||
position = null;
|
position = null;
|
||||||
} else {
|
} else {
|
||||||
const opt = optOrBufferOrCb as readOptions;
|
const opt = optOrBufferOrCb as ReadOptions;
|
||||||
if (
|
if (
|
||||||
!(opt.buffer instanceof Buffer) && !(opt.buffer instanceof Uint8Array)
|
!(opt.buffer instanceof Buffer) && !(opt.buffer instanceof Uint8Array)
|
||||||
) {
|
) {
|
||||||
|
|
|
@ -1,13 +1,20 @@
|
||||||
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
||||||
import { EventEmitter } from "ext:deno_node/events.ts";
|
import { EventEmitter } from "ext:deno_node/events.ts";
|
||||||
import { Buffer } from "ext:deno_node/buffer.ts";
|
import { Buffer } from "ext:deno_node/buffer.ts";
|
||||||
import { promises } from "ext:deno_node/fs.ts";
|
import { promises, read } from "ext:deno_node/fs.ts";
|
||||||
|
import type { Buffer } from "ext:deno_node/buffer.ts";
|
||||||
import {
|
import {
|
||||||
BinaryOptionsArgument,
|
BinaryOptionsArgument,
|
||||||
FileOptionsArgument,
|
FileOptionsArgument,
|
||||||
|
ReadOptions,
|
||||||
TextOptionsArgument,
|
TextOptionsArgument,
|
||||||
} from "ext:deno_node/_fs/_fs_common.ts";
|
} from "ext:deno_node/_fs/_fs_common.ts";
|
||||||
|
|
||||||
|
interface ReadResult {
|
||||||
|
bytesRead: number;
|
||||||
|
buffer: Buffer;
|
||||||
|
}
|
||||||
|
|
||||||
export class FileHandle extends EventEmitter {
|
export class FileHandle extends EventEmitter {
|
||||||
#rid: number;
|
#rid: number;
|
||||||
constructor(rid: number) {
|
constructor(rid: number) {
|
||||||
|
@ -19,6 +26,43 @@ export class FileHandle extends EventEmitter {
|
||||||
return this.rid;
|
return this.rid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
read(
|
||||||
|
buffer: Buffer,
|
||||||
|
offset?: number,
|
||||||
|
length?: number,
|
||||||
|
position?: number | null,
|
||||||
|
): Promise<ReadResult>;
|
||||||
|
read(options?: ReadOptions): Promise<ReadResult>;
|
||||||
|
read(
|
||||||
|
bufferOrOpt: Buffer | ReadOptions,
|
||||||
|
offset?: number,
|
||||||
|
length?: number,
|
||||||
|
position?: number | null,
|
||||||
|
): Promise<ReadResult> {
|
||||||
|
if (bufferOrOpt instanceof Buffer) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
read(
|
||||||
|
this.fd,
|
||||||
|
bufferOrOpt,
|
||||||
|
offset,
|
||||||
|
length,
|
||||||
|
position,
|
||||||
|
(err, bytesRead, buffer) => {
|
||||||
|
if (err) reject(err);
|
||||||
|
else resolve({ buffer: buffer, bytesRead: bytesRead });
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
read(this.fd, bufferOrOpt, (err, bytesRead, buffer) => {
|
||||||
|
if (err) reject(err);
|
||||||
|
else resolve({ buffer: buffer, bytesRead: bytesRead });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
readFile(
|
readFile(
|
||||||
opt?: TextOptionsArgument | BinaryOptionsArgument | FileOptionsArgument,
|
opt?: TextOptionsArgument | BinaryOptionsArgument | FileOptionsArgument,
|
||||||
): Promise<string | Buffer> {
|
): Promise<string | Buffer> {
|
||||||
|
|
Loading…
Reference in a new issue