mirror of
https://github.com/denoland/deno.git
synced 2025-01-10 16:11:13 -05:00
140 lines
3.8 KiB
TypeScript
140 lines
3.8 KiB
TypeScript
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
|
|
import { core } from "./core";
|
|
import * as flatbuffers from "./flatbuffers";
|
|
import * as msg from "gen/cli/msg_generated";
|
|
import * as errors from "./errors";
|
|
import * as util from "./util";
|
|
import {
|
|
nextPromiseId,
|
|
recordFromBufMinimal,
|
|
handleAsyncMsgFromRustMinimal
|
|
} from "./dispatch_minimal";
|
|
|
|
const promiseTable = new Map<number, util.Resolvable<msg.Base>>();
|
|
|
|
interface FlatbufferRecord {
|
|
promiseId: number;
|
|
base: msg.Base;
|
|
}
|
|
|
|
function flatbufferRecordFromBuf(buf: Uint8Array): FlatbufferRecord {
|
|
const bb = new flatbuffers.ByteBuffer(buf);
|
|
const base = msg.Base.getRootAsBase(bb);
|
|
return {
|
|
promiseId: base.cmdId(),
|
|
base
|
|
};
|
|
}
|
|
|
|
export function handleAsyncMsgFromRust(ui8: Uint8Array): void {
|
|
const buf32 = new Int32Array(ui8.buffer, ui8.byteOffset, ui8.byteLength / 4);
|
|
const recordMin = recordFromBufMinimal(buf32);
|
|
if (recordMin) {
|
|
// Fast and new
|
|
handleAsyncMsgFromRustMinimal(ui8, recordMin);
|
|
} else {
|
|
// Legacy
|
|
let { promiseId, base } = flatbufferRecordFromBuf(ui8);
|
|
const promise = promiseTable.get(promiseId);
|
|
util.assert(promise != null, `Expecting promise in table. ${promiseId}`);
|
|
promiseTable.delete(promiseId);
|
|
const err = errors.maybeError(base);
|
|
if (err != null) {
|
|
promise!.reject(err);
|
|
} else {
|
|
promise!.resolve(base);
|
|
}
|
|
}
|
|
}
|
|
|
|
function ui8FromArrayBufferView(abv: ArrayBufferView): Uint8Array {
|
|
return new Uint8Array(abv.buffer, abv.byteOffset, abv.byteLength);
|
|
}
|
|
|
|
function sendInternal(
|
|
builder: flatbuffers.Builder,
|
|
innerType: msg.Any,
|
|
inner: flatbuffers.Offset,
|
|
zeroCopy: undefined | ArrayBufferView,
|
|
isSync: true
|
|
): Uint8Array;
|
|
function sendInternal(
|
|
builder: flatbuffers.Builder,
|
|
innerType: msg.Any,
|
|
inner: flatbuffers.Offset,
|
|
zeroCopy: undefined | ArrayBufferView,
|
|
isSync: false
|
|
): Promise<msg.Base>;
|
|
function sendInternal(
|
|
builder: flatbuffers.Builder,
|
|
innerType: msg.Any,
|
|
inner: flatbuffers.Offset,
|
|
zeroCopy: undefined | ArrayBufferView,
|
|
isSync: boolean
|
|
): Promise<msg.Base> | Uint8Array {
|
|
const cmdId = nextPromiseId();
|
|
msg.Base.startBase(builder);
|
|
msg.Base.addInner(builder, inner);
|
|
msg.Base.addInnerType(builder, innerType);
|
|
msg.Base.addSync(builder, isSync);
|
|
msg.Base.addCmdId(builder, cmdId);
|
|
builder.finish(msg.Base.endBase(builder));
|
|
|
|
const control = builder.asUint8Array();
|
|
|
|
const response = core.dispatch(
|
|
control,
|
|
zeroCopy ? ui8FromArrayBufferView(zeroCopy) : undefined
|
|
);
|
|
|
|
builder.inUse = false;
|
|
|
|
if (response == null) {
|
|
util.assert(!isSync);
|
|
const promise = util.createResolvable<msg.Base>();
|
|
promiseTable.set(cmdId, promise);
|
|
return promise;
|
|
} else {
|
|
if (!isSync) {
|
|
// We can easily and correctly allow for sync responses to async calls
|
|
// by creating and returning a promise from the sync response.
|
|
const bb = new flatbuffers.ByteBuffer(response);
|
|
const base = msg.Base.getRootAsBase(bb);
|
|
const err = errors.maybeError(base);
|
|
if (err != null) {
|
|
return Promise.reject(err);
|
|
} else {
|
|
return Promise.resolve(base);
|
|
}
|
|
}
|
|
return response;
|
|
}
|
|
}
|
|
|
|
// @internal
|
|
export function sendAsync(
|
|
builder: flatbuffers.Builder,
|
|
innerType: msg.Any,
|
|
inner: flatbuffers.Offset,
|
|
data?: ArrayBufferView
|
|
): Promise<msg.Base> {
|
|
return sendInternal(builder, innerType, inner, data, false);
|
|
}
|
|
|
|
// @internal
|
|
export function sendSync(
|
|
builder: flatbuffers.Builder,
|
|
innerType: msg.Any,
|
|
inner: flatbuffers.Offset,
|
|
data?: ArrayBufferView
|
|
): null | msg.Base {
|
|
const response = sendInternal(builder, innerType, inner, data, true);
|
|
if (response!.length === 0) {
|
|
return null;
|
|
} else {
|
|
const bb = new flatbuffers.ByteBuffer(response!);
|
|
const baseRes = msg.Base.getRootAsBase(bb);
|
|
errors.maybeThrowError(baseRes);
|
|
return baseRes;
|
|
}
|
|
}
|