2019-01-22 04:03:30 +09:00
|
|
|
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
|
2019-03-28 04:29:36 +01:00
|
|
|
import {
|
|
|
|
Reader,
|
|
|
|
Writer,
|
|
|
|
Seeker,
|
|
|
|
Closer,
|
|
|
|
ReadResult,
|
|
|
|
SeekMode,
|
|
|
|
SyncReader,
|
|
|
|
SyncWriter,
|
|
|
|
SyncSeeker
|
|
|
|
} from "./io";
|
2018-09-27 00:56:39 -04:00
|
|
|
import * as dispatch from "./dispatch";
|
2019-05-03 00:06:43 -04:00
|
|
|
import { sendAsyncMinimal } from "./dispatch_minimal";
|
2019-03-30 14:45:36 -04:00
|
|
|
import * as msg from "gen/cli/msg_generated";
|
2018-09-27 00:56:39 -04:00
|
|
|
import { assert } from "./util";
|
2018-10-17 13:04:28 -04:00
|
|
|
import * as flatbuffers from "./flatbuffers";
|
2018-09-27 00:56:39 -04:00
|
|
|
|
2019-05-03 00:06:43 -04:00
|
|
|
const OP_READ = 1;
|
|
|
|
const OP_WRITE = 2;
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
function reqOpen(
|
2018-09-27 00:56:39 -04:00
|
|
|
filename: string,
|
2019-03-28 04:29:36 +01:00
|
|
|
mode: OpenMode
|
|
|
|
): [flatbuffers.Builder, msg.Any, flatbuffers.Offset] {
|
2018-10-17 13:04:28 -04:00
|
|
|
const builder = flatbuffers.createBuilder();
|
2018-09-27 00:56:39 -04:00
|
|
|
const filename_ = builder.createString(filename);
|
2018-12-12 18:05:58 +01:00
|
|
|
const mode_ = builder.createString(mode);
|
2019-04-07 20:51:43 -04:00
|
|
|
const inner = msg.Open.createOpen(builder, filename_, 0, mode_);
|
2019-03-28 04:29:36 +01:00
|
|
|
return [builder, msg.Any.Open, inner];
|
|
|
|
}
|
|
|
|
|
|
|
|
function resOpen(baseRes: null | msg.Base): File {
|
2018-09-27 00:56:39 -04:00
|
|
|
assert(baseRes != null);
|
2018-10-03 21:18:23 -04:00
|
|
|
assert(msg.Any.OpenRes === baseRes!.innerType());
|
|
|
|
const res = new msg.OpenRes();
|
2018-10-03 21:12:23 -04:00
|
|
|
assert(baseRes!.inner(res) != null);
|
2018-10-11 00:59:36 +09:00
|
|
|
const rid = res.rid();
|
2019-03-10 04:30:38 +11:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-use-before-define
|
2018-10-11 00:59:36 +09:00
|
|
|
return new File(rid);
|
2018-09-27 00:56:39 -04:00
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
/** Open a file and return an instance of the `File` object
|
|
|
|
* synchronously.
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
2019-03-28 04:29:36 +01:00
|
|
|
* const file = Deno.openSync("/foo/bar.txt");
|
2018-10-15 07:29:50 +11:00
|
|
|
*/
|
2019-03-28 04:29:36 +01:00
|
|
|
export function openSync(filename: string, mode: OpenMode = "r"): File {
|
|
|
|
return resOpen(dispatch.sendSync(...reqOpen(filename, mode)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Open a file and return an instance of the `File` object.
|
|
|
|
*
|
|
|
|
* (async () => {
|
|
|
|
* const file = await Deno.open("/foo/bar.txt");
|
|
|
|
* })();
|
|
|
|
*/
|
|
|
|
export async function open(
|
|
|
|
filename: string,
|
|
|
|
mode: OpenMode = "r"
|
|
|
|
): Promise<File> {
|
|
|
|
return resOpen(await dispatch.sendAsync(...reqOpen(filename, mode)));
|
|
|
|
}
|
|
|
|
|
|
|
|
function reqRead(
|
|
|
|
rid: number,
|
|
|
|
p: Uint8Array
|
|
|
|
): [flatbuffers.Builder, msg.Any, flatbuffers.Offset, Uint8Array] {
|
2018-10-17 13:04:28 -04:00
|
|
|
const builder = flatbuffers.createBuilder();
|
2019-04-07 20:51:43 -04:00
|
|
|
const inner = msg.Read.createRead(builder, rid);
|
2019-03-28 04:29:36 +01:00
|
|
|
return [builder, msg.Any.Read, inner, p];
|
|
|
|
}
|
|
|
|
|
|
|
|
function resRead(baseRes: null | msg.Base): ReadResult {
|
2018-09-27 00:56:39 -04:00
|
|
|
assert(baseRes != null);
|
2018-10-03 21:18:23 -04:00
|
|
|
assert(msg.Any.ReadRes === baseRes!.innerType());
|
|
|
|
const res = new msg.ReadRes();
|
2018-10-03 21:12:23 -04:00
|
|
|
assert(baseRes!.inner(res) != null);
|
2018-09-27 00:56:39 -04:00
|
|
|
return { nread: res.nread(), eof: res.eof() };
|
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
/** Read synchronously from a file ID into an array buffer.
|
|
|
|
*
|
|
|
|
* Return `ReadResult` for the operation.
|
|
|
|
*
|
2019-04-03 20:53:54 +08:00
|
|
|
* const file = Deno.openSync("/foo/bar.txt");
|
|
|
|
* const buf = new Uint8Array(100);
|
|
|
|
* const { nread, eof } = Deno.readSync(file.rid, buf);
|
2019-04-04 17:34:00 +08:00
|
|
|
* const text = new TextDecoder().decode(buf);
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
|
|
|
*/
|
2019-03-28 04:29:36 +01:00
|
|
|
export function readSync(rid: number, p: Uint8Array): ReadResult {
|
|
|
|
return resRead(dispatch.sendSync(...reqRead(rid, p)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Read from a file ID into an array buffer.
|
|
|
|
*
|
|
|
|
* Resolves with the `ReadResult` for the operation.
|
|
|
|
*
|
2019-04-03 20:53:54 +08:00
|
|
|
* (async () => {
|
2019-03-28 04:29:36 +01:00
|
|
|
* const file = await Deno.open("/foo/bar.txt");
|
|
|
|
* const buf = new Uint8Array(100);
|
|
|
|
* const { nread, eof } = await Deno.read(file.rid, buf);
|
2019-04-04 17:34:00 +08:00
|
|
|
* const text = new TextDecoder().decode(buf);
|
2019-04-03 20:53:54 +08:00
|
|
|
* })();
|
2019-03-28 04:29:36 +01:00
|
|
|
*/
|
|
|
|
export async function read(rid: number, p: Uint8Array): Promise<ReadResult> {
|
2019-05-03 00:06:43 -04:00
|
|
|
const nread = await sendAsyncMinimal(OP_READ, rid, p);
|
|
|
|
if (nread < 0) {
|
|
|
|
throw new Error("read error");
|
|
|
|
} else if (nread == 0) {
|
|
|
|
return { nread, eof: true };
|
|
|
|
} else {
|
|
|
|
return { nread, eof: false };
|
|
|
|
}
|
2019-03-28 04:29:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function reqWrite(
|
|
|
|
rid: number,
|
|
|
|
p: Uint8Array
|
|
|
|
): [flatbuffers.Builder, msg.Any, flatbuffers.Offset, Uint8Array] {
|
2018-10-17 13:04:28 -04:00
|
|
|
const builder = flatbuffers.createBuilder();
|
2019-04-07 20:51:43 -04:00
|
|
|
const inner = msg.Write.createWrite(builder, rid);
|
2019-03-28 04:29:36 +01:00
|
|
|
return [builder, msg.Any.Write, inner, p];
|
|
|
|
}
|
|
|
|
|
|
|
|
function resWrite(baseRes: null | msg.Base): number {
|
2018-09-27 00:56:39 -04:00
|
|
|
assert(baseRes != null);
|
2018-10-03 21:18:23 -04:00
|
|
|
assert(msg.Any.WriteRes === baseRes!.innerType());
|
|
|
|
const res = new msg.WriteRes();
|
2018-10-03 21:12:23 -04:00
|
|
|
assert(baseRes!.inner(res) != null);
|
2018-09-27 00:56:39 -04:00
|
|
|
return res.nbyte();
|
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
/** Write synchronously to the file ID the contents of the array buffer.
|
2019-02-18 15:26:41 -08:00
|
|
|
*
|
2019-03-28 04:29:36 +01:00
|
|
|
* Resolves with the number of bytes written.
|
|
|
|
*
|
|
|
|
* const encoder = new TextEncoder();
|
|
|
|
* const data = encoder.encode("Hello world\n");
|
|
|
|
* const file = Deno.openSync("/foo/bar.txt");
|
|
|
|
* Deno.writeSync(file.rid, data);
|
2019-02-18 15:26:41 -08:00
|
|
|
*/
|
2019-03-28 04:29:36 +01:00
|
|
|
export function writeSync(rid: number, p: Uint8Array): number {
|
|
|
|
return resWrite(dispatch.sendSync(...reqWrite(rid, p)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Write to the file ID the contents of the array buffer.
|
|
|
|
*
|
|
|
|
* Resolves with the number of bytes written.
|
|
|
|
*
|
2019-04-03 20:53:54 +08:00
|
|
|
* (async () => {
|
2019-03-28 04:29:36 +01:00
|
|
|
* const encoder = new TextEncoder();
|
|
|
|
* const data = encoder.encode("Hello world\n");
|
|
|
|
* const file = await Deno.open("/foo/bar.txt");
|
|
|
|
* await Deno.write(file.rid, data);
|
2019-04-03 20:53:54 +08:00
|
|
|
* })();
|
2019-03-28 04:29:36 +01:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
export async function write(rid: number, p: Uint8Array): Promise<number> {
|
2019-05-03 00:06:43 -04:00
|
|
|
let result = await sendAsyncMinimal(OP_WRITE, rid, p);
|
|
|
|
if (result < 0) {
|
|
|
|
throw new Error("write error");
|
|
|
|
} else {
|
|
|
|
return result;
|
|
|
|
}
|
2019-03-28 04:29:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function reqSeek(
|
2019-02-18 15:26:41 -08:00
|
|
|
rid: number,
|
|
|
|
offset: number,
|
|
|
|
whence: SeekMode
|
2019-03-28 04:29:36 +01:00
|
|
|
): [flatbuffers.Builder, msg.Any, flatbuffers.Offset] {
|
2019-02-18 15:26:41 -08:00
|
|
|
const builder = flatbuffers.createBuilder();
|
2019-04-07 20:51:43 -04:00
|
|
|
const inner = msg.Seek.createSeek(builder, rid, offset, whence);
|
2019-03-28 04:29:36 +01:00
|
|
|
return [builder, msg.Any.Seek, inner];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Seek a file ID synchronously to the given offset under mode given by `whence`.
|
|
|
|
*
|
2019-04-03 20:53:54 +08:00
|
|
|
* const file = Deno.openSync("/foo/bar.txt");
|
|
|
|
* Deno.seekSync(file.rid, 0, 0);
|
2019-03-28 04:29:36 +01:00
|
|
|
*/
|
|
|
|
export function seekSync(rid: number, offset: number, whence: SeekMode): void {
|
|
|
|
dispatch.sendSync(...reqSeek(rid, offset, whence));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Seek a file ID to the given offset under mode given by `whence`.
|
|
|
|
*
|
2019-04-03 20:53:54 +08:00
|
|
|
* (async () => {
|
2019-03-28 04:29:36 +01:00
|
|
|
* const file = await Deno.open("/foo/bar.txt");
|
|
|
|
* await Deno.seek(file.rid, 0, 0);
|
2019-04-03 20:53:54 +08:00
|
|
|
* })();
|
2019-03-28 04:29:36 +01:00
|
|
|
*/
|
|
|
|
export async function seek(
|
|
|
|
rid: number,
|
|
|
|
offset: number,
|
|
|
|
whence: SeekMode
|
|
|
|
): Promise<void> {
|
|
|
|
await dispatch.sendAsync(...reqSeek(rid, offset, whence));
|
2019-02-18 15:26:41 -08:00
|
|
|
}
|
|
|
|
|
2018-10-15 07:29:50 +11:00
|
|
|
/** Close the file ID. */
|
2018-10-11 00:59:36 +09:00
|
|
|
export function close(rid: number): void {
|
2018-10-17 13:04:28 -04:00
|
|
|
const builder = flatbuffers.createBuilder();
|
2019-04-07 20:51:43 -04:00
|
|
|
const inner = msg.Close.createClose(builder, rid);
|
2018-10-03 21:18:23 -04:00
|
|
|
dispatch.sendSync(builder, msg.Any.Close, inner);
|
2018-09-27 00:56:39 -04:00
|
|
|
}
|
2019-03-10 04:30:38 +11:00
|
|
|
|
|
|
|
/** The Deno abstraction for reading and writing files. */
|
2019-03-28 04:29:36 +01:00
|
|
|
export class File
|
|
|
|
implements
|
|
|
|
Reader,
|
|
|
|
SyncReader,
|
|
|
|
Writer,
|
|
|
|
SyncWriter,
|
|
|
|
Seeker,
|
|
|
|
SyncSeeker,
|
|
|
|
Closer {
|
2019-03-10 04:30:38 +11:00
|
|
|
constructor(readonly rid: number) {}
|
|
|
|
|
|
|
|
write(p: Uint8Array): Promise<number> {
|
|
|
|
return write(this.rid, p);
|
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
writeSync(p: Uint8Array): number {
|
|
|
|
return writeSync(this.rid, p);
|
|
|
|
}
|
|
|
|
|
2019-03-10 04:30:38 +11:00
|
|
|
read(p: Uint8Array): Promise<ReadResult> {
|
|
|
|
return read(this.rid, p);
|
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
readSync(p: Uint8Array): ReadResult {
|
|
|
|
return readSync(this.rid, p);
|
|
|
|
}
|
|
|
|
|
2019-03-10 04:30:38 +11:00
|
|
|
seek(offset: number, whence: SeekMode): Promise<void> {
|
|
|
|
return seek(this.rid, offset, whence);
|
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
seekSync(offset: number, whence: SeekMode): void {
|
|
|
|
return seekSync(this.rid, offset, whence);
|
|
|
|
}
|
|
|
|
|
2019-03-10 04:30:38 +11:00
|
|
|
close(): void {
|
|
|
|
close(this.rid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** An instance of `File` for stdin. */
|
|
|
|
export const stdin = new File(0);
|
|
|
|
/** An instance of `File` for stdout. */
|
|
|
|
export const stdout = new File(1);
|
|
|
|
/** An instance of `File` for stderr. */
|
|
|
|
export const stderr = new File(2);
|
|
|
|
|
|
|
|
export type OpenMode =
|
|
|
|
/** Read-only. Default. Starts at beginning of file. */
|
|
|
|
| "r"
|
|
|
|
/** Read-write. Start at beginning of file. */
|
|
|
|
| "r+"
|
|
|
|
/** Write-only. Opens and truncates existing file or creates new one for
|
|
|
|
* writing only.
|
|
|
|
*/
|
|
|
|
| "w"
|
|
|
|
/** Read-write. Opens and truncates existing file or creates new one for
|
|
|
|
* writing and reading.
|
|
|
|
*/
|
|
|
|
| "w+"
|
|
|
|
/** Write-only. Opens existing file or creates new one. Each write appends
|
|
|
|
* content to the end of file.
|
|
|
|
*/
|
|
|
|
| "a"
|
|
|
|
/** Read-write. Behaves like "a" and allows to read from file. */
|
|
|
|
| "a+"
|
|
|
|
/** Write-only. Exclusive create - creates new file only if one doesn't exist
|
|
|
|
* already.
|
|
|
|
*/
|
|
|
|
| "x"
|
|
|
|
/** Read-write. Behaves like `x` and allows to read from file. */
|
|
|
|
| "x+";
|
|
|
|
|
|
|
|
/** A factory function for creating instances of `File` associated with the
|
|
|
|
* supplied file name.
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export function create(filename: string): Promise<File> {
|
|
|
|
return open(filename, "w+");
|
|
|
|
}
|