1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-12 00:54:02 -05:00

Add error handling for minimal dispatch (#3176)

This commit is contained in:
Bartek Iwańczuk 2019-10-24 23:22:31 +02:00 committed by Ry Dahl
parent 1d8f3cc896
commit 492b87d460
4 changed files with 101 additions and 29 deletions

1
Cargo.lock generated
View file

@ -271,6 +271,7 @@ version = "0.21.0"
dependencies = [ dependencies = [
"ansi_term 0.12.1 (registry+https://github.com/rust-lang/crates.io-index)", "ansi_term 0.12.1 (registry+https://github.com/rust-lang/crates.io-index)",
"atty 0.2.13 (registry+https://github.com/rust-lang/crates.io-index)", "atty 0.2.13 (registry+https://github.com/rust-lang/crates.io-index)",
"byteorder 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)",
"clap 2.33.0 (registry+https://github.com/rust-lang/crates.io-index)", "clap 2.33.0 (registry+https://github.com/rust-lang/crates.io-index)",
"deno 0.21.0", "deno 0.21.0",
"deno_typescript 0.21.0", "deno_typescript 0.21.0",

View file

@ -27,6 +27,7 @@ deno_typescript = { path = "../deno_typescript", version = "0.21.0" }
ansi_term = "0.12.1" ansi_term = "0.12.1"
atty = "0.2.13" atty = "0.2.13"
byteorder = "1.3.2"
clap = "2.33.0" clap = "2.33.0"
dirs = "2.0.2" dirs = "2.0.2"
futures = "0.1.29" futures = "0.1.29"

View file

@ -1,13 +1,17 @@
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license. // Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
import * as util from "./util.ts"; import * as util from "./util.ts";
import { core } from "./core.ts"; import { core } from "./core.ts";
import { TextDecoder } from "./text_encoding.ts";
import { ErrorKind, DenoError } from "./errors.ts";
const promiseTableMin = new Map<number, util.Resolvable<number>>(); const promiseTableMin = new Map<number, util.Resolvable<RecordMinimal>>();
// Note it's important that promiseId starts at 1 instead of 0, because sync // Note it's important that promiseId starts at 1 instead of 0, because sync
// messages are indicated with promiseId 0. If we ever add wrap around logic for // messages are indicated with promiseId 0. If we ever add wrap around logic for
// overflows, this should be taken into account. // overflows, this should be taken into account.
let _nextPromiseId = 1; let _nextPromiseId = 1;
const decoder = new TextDecoder();
function nextPromiseId(): number { function nextPromiseId(): number {
return _nextPromiseId++; return _nextPromiseId++;
} }
@ -17,23 +21,51 @@ export interface RecordMinimal {
opId: number; // Maybe better called dispatchId opId: number; // Maybe better called dispatchId
arg: number; arg: number;
result: number; result: number;
err?: {
kind: ErrorKind;
message: string;
};
} }
export function recordFromBufMinimal( export function recordFromBufMinimal(
opId: number, opId: number,
buf32: Int32Array ui8: Uint8Array
): RecordMinimal { ): RecordMinimal {
if (buf32.length != 3) { const header = ui8.slice(0, 12);
throw Error("Bad message"); const buf32 = new Int32Array(
header.buffer,
header.byteOffset,
header.byteLength / 4
);
const promiseId = buf32[0];
const arg = buf32[1];
const result = buf32[2];
let err;
if (arg < 0) {
const kind = result as ErrorKind;
const message = decoder.decode(ui8.slice(12));
err = { kind, message };
} else if (ui8.length != 12) {
err = { kind: ErrorKind.InvalidData, message: "Bad message" };
} }
return { return {
promiseId: buf32[0], promiseId,
opId, opId,
arg: buf32[1], arg,
result: buf32[2] result,
err
}; };
} }
function unwrapResponse(res: RecordMinimal): number {
if (res.err != null) {
throw new DenoError(res.err!.kind, res.err!.message);
}
return res.result;
}
const scratch32 = new Int32Array(3); const scratch32 = new Int32Array(3);
const scratchBytes = new Uint8Array( const scratchBytes = new Uint8Array(
scratch32.buffer, scratch32.buffer,
@ -43,15 +75,14 @@ const scratchBytes = new Uint8Array(
util.assert(scratchBytes.byteLength === scratch32.length * 4); util.assert(scratchBytes.byteLength === scratch32.length * 4);
export function asyncMsgFromRust(opId: number, ui8: Uint8Array): void { export function asyncMsgFromRust(opId: number, ui8: Uint8Array): void {
const buf32 = new Int32Array(ui8.buffer, ui8.byteOffset, ui8.byteLength / 4); const record = recordFromBufMinimal(opId, ui8);
const record = recordFromBufMinimal(opId, buf32); const { promiseId } = record;
const { promiseId, result } = record;
const promise = promiseTableMin.get(promiseId); const promise = promiseTableMin.get(promiseId);
promiseTableMin.delete(promiseId); promiseTableMin.delete(promiseId);
promise!.resolve(result); promise!.resolve(record);
} }
export function sendAsyncMinimal( export async function sendAsyncMinimal(
opId: number, opId: number,
arg: number, arg: number,
zeroCopy: Uint8Array zeroCopy: Uint8Array
@ -60,22 +91,19 @@ export function sendAsyncMinimal(
scratch32[0] = promiseId; scratch32[0] = promiseId;
scratch32[1] = arg; scratch32[1] = arg;
scratch32[2] = 0; // result scratch32[2] = 0; // result
const promise = util.createResolvable<number>(); const promise = util.createResolvable<RecordMinimal>();
const buf = core.dispatch(opId, scratchBytes, zeroCopy); const buf = core.dispatch(opId, scratchBytes, zeroCopy);
if (buf) { if (buf) {
const buf32 = new Int32Array( const record = recordFromBufMinimal(opId, buf);
buf.buffer,
buf.byteOffset,
buf.byteLength / 4
);
const record = recordFromBufMinimal(opId, buf32);
// Sync result. // Sync result.
promise.resolve(record.result); promise.resolve(record);
} else { } else {
// Async result. // Async result.
promiseTableMin.set(promiseId, promise); promiseTableMin.set(promiseId, promise);
} }
return promise;
const res = await promise;
return unwrapResponse(res);
} }
export function sendSyncMinimal( export function sendSyncMinimal(
@ -86,7 +114,6 @@ export function sendSyncMinimal(
scratch32[0] = 0; // promiseId 0 indicates sync scratch32[0] = 0; // promiseId 0 indicates sync
scratch32[1] = arg; scratch32[1] = arg;
const res = core.dispatch(opId, scratchBytes, zeroCopy)!; const res = core.dispatch(opId, scratchBytes, zeroCopy)!;
const res32 = new Int32Array(res.buffer, res.byteOffset, 3); const resRecord = recordFromBufMinimal(opId, res);
const resRecord = recordFromBufMinimal(opId, res32); return unwrapResponse(resRecord);
return resRecord.result;
} }

View file

@ -4,6 +4,8 @@
//! alternative to flatbuffers using a very simple list of int32s to lay out //! alternative to flatbuffers using a very simple list of int32s to lay out
//! messages. The first i32 is used to determine if a message a flatbuffer //! messages. The first i32 is used to determine if a message a flatbuffer
//! message or a "minimal" message. //! message or a "minimal" message.
use crate::deno_error::GetErrorKind;
use byteorder::{LittleEndian, WriteBytesExt};
use deno::Buf; use deno::Buf;
use deno::CoreOp; use deno::CoreOp;
use deno::ErrBox; use deno::ErrBox;
@ -31,6 +33,44 @@ impl Into<Buf> for Record {
} }
} }
pub struct ErrorRecord {
pub promise_id: i32,
pub arg: i32,
pub error_code: i32,
pub error_message: Vec<u8>,
}
impl Into<Buf> for ErrorRecord {
fn into(self) -> Buf {
let v32: Vec<i32> = vec![self.promise_id, self.arg, self.error_code];
let mut v8: Vec<u8> = Vec::new();
for n in v32 {
v8.write_i32::<LittleEndian>(n).unwrap();
}
let mut message = self.error_message;
// Align to 32bit word, padding with the space character.
message.resize((message.len() + 3usize) & !3usize, b' ');
v8.append(&mut message);
v8.into_boxed_slice()
}
}
#[test]
fn test_error_record() {
let expected = vec![
1, 0, 0, 0, 255, 255, 255, 255, 10, 0, 0, 0, 69, 114, 114, 111, 114, 32,
32, 32,
];
let err_record = ErrorRecord {
promise_id: 1,
arg: -1,
error_code: 10,
error_message: "Error".to_string().as_bytes().to_owned(),
};
let buf: Buf = err_record.into();
assert_eq!(buf, expected.into_boxed_slice());
}
pub fn parse_min_record(bytes: &[u8]) -> Option<Record> { pub fn parse_min_record(bytes: &[u8]) -> Option<Record> {
if bytes.len() % std::mem::size_of::<i32>() != 0 { if bytes.len() % std::mem::size_of::<i32>() != 0 {
return None; return None;
@ -85,15 +125,18 @@ pub fn minimal_op(
match result { match result {
Ok(r) => { Ok(r) => {
record.result = r; record.result = r;
Ok(record.into())
} }
Err(err) => { Err(err) => {
// TODO(ry) The dispatch_minimal doesn't properly pipe errors back to let error_record = ErrorRecord {
// the caller. promise_id: record.promise_id,
debug!("swallowed err {}", err); arg: -1,
record.result = -1; error_code: err.kind() as i32,
error_message: err.to_string().as_bytes().to_owned(),
};
Ok(error_record.into())
} }
} }
Ok(record.into())
})); }));
if is_sync { if is_sync {