2019-12-08 20:26:58 -05:00
|
|
|
#![allow(non_snake_case)]
|
|
|
|
|
2021-12-28 10:52:12 -05:00
|
|
|
use std::convert::TryInto;
|
|
|
|
|
2019-12-19 19:15:52 -05:00
|
|
|
use crate::isolate::Isolate;
|
2019-12-08 20:26:58 -05:00
|
|
|
use crate::support::int;
|
2019-12-31 09:40:34 -05:00
|
|
|
use crate::Context;
|
2020-06-03 01:38:34 -04:00
|
|
|
use crate::HandleScope;
|
2019-12-08 20:26:58 -05:00
|
|
|
use crate::Local;
|
2020-03-05 20:41:43 -05:00
|
|
|
use crate::Message;
|
|
|
|
use crate::StackFrame;
|
|
|
|
use crate::StackTrace;
|
2019-12-08 20:26:58 -05:00
|
|
|
use crate::String;
|
|
|
|
use crate::Value;
|
|
|
|
|
|
|
|
extern "C" {
|
2020-04-13 08:43:56 -04:00
|
|
|
fn v8__Message__Get(this: *const Message) -> *const String;
|
2019-12-31 09:40:34 -05:00
|
|
|
fn v8__Message__GetSourceLine(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Message,
|
|
|
|
context: *const Context,
|
|
|
|
) -> *const String;
|
|
|
|
fn v8__Message__GetScriptResourceName(this: *const Message) -> *const Value;
|
2019-12-31 09:40:34 -05:00
|
|
|
fn v8__Message__GetLineNumber(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const Message,
|
|
|
|
context: *const Context,
|
2019-12-31 09:40:34 -05:00
|
|
|
) -> int;
|
2020-04-13 08:43:56 -04:00
|
|
|
fn v8__Message__GetStartPosition(this: *const Message) -> int;
|
|
|
|
fn v8__Message__GetEndPosition(this: *const Message) -> int;
|
|
|
|
fn v8__Message__GetWasmFunctionIndex(this: *const Message) -> int;
|
|
|
|
fn v8__Message__ErrorLevel(this: *const Message) -> int;
|
|
|
|
fn v8__Message__GetStartColumn(this: *const Message) -> int;
|
|
|
|
fn v8__Message__GetEndColumn(this: *const Message) -> int;
|
|
|
|
fn v8__Message__IsSharedCrossOrigin(this: *const Message) -> bool;
|
|
|
|
fn v8__Message__IsOpaque(this: *const Message) -> bool;
|
|
|
|
fn v8__Message__GetStackTrace(this: *const Message) -> *const StackTrace;
|
|
|
|
|
2021-12-28 10:52:12 -05:00
|
|
|
fn v8__StackTrace__CurrentStackTrace(
|
|
|
|
isolate: *mut Isolate,
|
|
|
|
frame_limit: int,
|
|
|
|
) -> *const StackTrace;
|
2020-04-13 08:43:56 -04:00
|
|
|
fn v8__StackTrace__GetFrameCount(this: *const StackTrace) -> int;
|
2019-12-31 11:17:26 -05:00
|
|
|
fn v8__StackTrace__GetFrame(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const StackTrace,
|
2019-12-31 11:17:26 -05:00
|
|
|
isolate: *mut Isolate,
|
|
|
|
index: u32,
|
2020-04-13 08:43:56 -04:00
|
|
|
) -> *const StackFrame;
|
2019-12-31 11:17:26 -05:00
|
|
|
|
2020-04-13 08:43:56 -04:00
|
|
|
fn v8__StackFrame__GetLineNumber(this: *const StackFrame) -> int;
|
|
|
|
fn v8__StackFrame__GetColumn(this: *const StackFrame) -> int;
|
|
|
|
fn v8__StackFrame__GetScriptId(this: *const StackFrame) -> int;
|
|
|
|
fn v8__StackFrame__GetScriptName(this: *const StackFrame) -> *const String;
|
2019-12-31 11:17:26 -05:00
|
|
|
fn v8__StackFrame__GetScriptNameOrSourceURL(
|
2020-04-13 08:43:56 -04:00
|
|
|
this: *const StackFrame,
|
|
|
|
) -> *const String;
|
|
|
|
fn v8__StackFrame__GetFunctionName(this: *const StackFrame) -> *const String;
|
|
|
|
fn v8__StackFrame__IsEval(this: *const StackFrame) -> bool;
|
|
|
|
fn v8__StackFrame__IsConstructor(this: *const StackFrame) -> bool;
|
|
|
|
fn v8__StackFrame__IsWasm(this: *const StackFrame) -> bool;
|
|
|
|
fn v8__StackFrame__IsUserJavaScript(this: *const StackFrame) -> bool;
|
|
|
|
|
|
|
|
fn v8__Exception__Error(message: *const String) -> *const Value;
|
|
|
|
fn v8__Exception__RangeError(message: *const String) -> *const Value;
|
|
|
|
fn v8__Exception__ReferenceError(message: *const String) -> *const Value;
|
|
|
|
fn v8__Exception__SyntaxError(message: *const String) -> *const Value;
|
|
|
|
fn v8__Exception__TypeError(message: *const String) -> *const Value;
|
2019-12-08 20:26:58 -05:00
|
|
|
|
|
|
|
fn v8__Exception__CreateMessage(
|
2020-04-13 08:43:56 -04:00
|
|
|
isolate: *mut Isolate,
|
|
|
|
exception: *const Value,
|
|
|
|
) -> *const Message;
|
|
|
|
fn v8__Exception__GetStackTrace(exception: *const Value)
|
|
|
|
-> *const StackTrace;
|
2019-12-08 20:26:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl StackTrace {
|
2021-12-28 10:52:12 -05:00
|
|
|
/// Grab a snapshot of the current JavaScript execution stack.
|
|
|
|
pub fn current_stack_trace<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
|
|
|
frame_limit: usize,
|
|
|
|
) -> Option<Local<'s, StackTrace>> {
|
|
|
|
let frame_limit = frame_limit.try_into().ok()?;
|
|
|
|
unsafe {
|
|
|
|
scope.cast_local(|sd| {
|
|
|
|
v8__StackTrace__CurrentStackTrace(sd.get_isolate_ptr(), frame_limit)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-19 08:14:19 -05:00
|
|
|
/// Returns the number of StackFrames.
|
2019-12-31 11:17:26 -05:00
|
|
|
pub fn get_frame_count(&self) -> usize {
|
2019-12-08 20:26:58 -05:00
|
|
|
unsafe { v8__StackTrace__GetFrameCount(self) as usize }
|
|
|
|
}
|
2019-12-31 11:17:26 -05:00
|
|
|
|
|
|
|
/// Returns a StackFrame at a particular index.
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn get_frame<'s>(
|
2019-12-31 11:17:26 -05:00
|
|
|
&self,
|
2020-06-03 01:38:34 -04:00
|
|
|
scope: &mut HandleScope<'s>,
|
2019-12-31 11:17:26 -05:00
|
|
|
index: usize,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Option<Local<'s, StackFrame>> {
|
2019-12-31 11:17:26 -05:00
|
|
|
unsafe {
|
2020-06-03 01:38:34 -04:00
|
|
|
scope.cast_local(|sd| {
|
|
|
|
v8__StackTrace__GetFrame(self, sd.get_isolate_ptr(), index as u32)
|
2020-05-31 09:00:23 -04:00
|
|
|
})
|
2019-12-31 11:17:26 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl StackFrame {
|
2020-12-10 06:47:32 -05:00
|
|
|
/// Returns the number, 1-based, of the line for the associated function call.
|
2019-12-31 11:17:26 -05:00
|
|
|
/// This method will return Message::kNoLineNumberInfo if it is unable to
|
|
|
|
/// retrieve the line number, or if kLineNumber was not passed as an option
|
|
|
|
/// when capturing the StackTrace.
|
|
|
|
pub fn get_line_number(&self) -> usize {
|
|
|
|
unsafe { v8__StackFrame__GetLineNumber(self) as usize }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the 1-based column offset on the line for the associated function
|
|
|
|
/// call.
|
|
|
|
/// This method will return Message::kNoColumnInfo if it is unable to retrieve
|
|
|
|
/// the column number, or if kColumnOffset was not passed as an option when
|
|
|
|
/// capturing the StackTrace.
|
|
|
|
pub fn get_column(&self) -> usize {
|
|
|
|
unsafe { v8__StackFrame__GetColumn(self) as usize }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the id of the script for the function for this StackFrame.
|
|
|
|
/// This method will return Message::kNoScriptIdInfo if it is unable to
|
|
|
|
/// retrieve the script id, or if kScriptId was not passed as an option when
|
|
|
|
/// capturing the StackTrace.
|
|
|
|
pub fn get_script_id(&self) -> usize {
|
|
|
|
unsafe { v8__StackFrame__GetScriptId(self) as usize }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the name of the resource that contains the script for the
|
|
|
|
/// function for this StackFrame.
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn get_script_name<'s>(
|
2019-12-31 11:17:26 -05:00
|
|
|
&self,
|
2020-06-03 01:38:34 -04:00
|
|
|
scope: &mut HandleScope<'s>,
|
|
|
|
) -> Option<Local<'s, String>> {
|
2020-05-31 09:00:23 -04:00
|
|
|
unsafe { scope.cast_local(|_| v8__StackFrame__GetScriptName(self)) }
|
2019-12-31 11:17:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the name of the resource that contains the script for the
|
|
|
|
/// function for this StackFrame or sourceURL value if the script name
|
|
|
|
/// is undefined and its source ends with //# sourceURL=... string or
|
|
|
|
/// deprecated //@ sourceURL=... string.
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn get_script_name_or_source_url<'s>(
|
2019-12-31 11:17:26 -05:00
|
|
|
&self,
|
2020-06-03 01:38:34 -04:00
|
|
|
scope: &mut HandleScope<'s>,
|
|
|
|
) -> Option<Local<'s, String>> {
|
2020-05-31 09:00:23 -04:00
|
|
|
unsafe {
|
|
|
|
scope.cast_local(|_| v8__StackFrame__GetScriptNameOrSourceURL(self))
|
|
|
|
}
|
2019-12-31 11:17:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the name of the function associated with this stack frame.
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn get_function_name<'s>(
|
2019-12-31 11:17:26 -05:00
|
|
|
&self,
|
2020-06-03 01:38:34 -04:00
|
|
|
scope: &mut HandleScope<'s>,
|
|
|
|
) -> Option<Local<'s, String>> {
|
2020-05-31 09:00:23 -04:00
|
|
|
unsafe { scope.cast_local(|_| v8__StackFrame__GetFunctionName(self)) }
|
2019-12-31 11:17:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns whether or not the associated function is compiled via a call to
|
|
|
|
/// eval().
|
|
|
|
pub fn is_eval(&self) -> bool {
|
|
|
|
unsafe { v8__StackFrame__IsEval(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns whether or not the associated function is called as a
|
|
|
|
/// constructor via "new".
|
|
|
|
pub fn is_constructor(&self) -> bool {
|
|
|
|
unsafe { v8__StackFrame__IsConstructor(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns whether or not the associated functions is defined in wasm.
|
|
|
|
pub fn is_wasm(&self) -> bool {
|
|
|
|
unsafe { v8__StackFrame__IsWasm(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns whether or not the associated function is defined by the user.
|
|
|
|
pub fn is_user_javascript(&self) -> bool {
|
|
|
|
unsafe { v8__StackFrame__IsUserJavaScript(self) }
|
|
|
|
}
|
2019-12-08 20:26:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Message {
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn get<'s>(&self, scope: &mut HandleScope<'s>) -> Local<'s, String> {
|
2020-05-31 09:00:23 -04:00
|
|
|
unsafe { scope.cast_local(|_| v8__Message__Get(self)) }.unwrap()
|
2019-12-08 20:26:58 -05:00
|
|
|
}
|
2019-12-19 21:34:07 -05:00
|
|
|
|
2019-12-31 11:17:26 -05:00
|
|
|
/// Exception stack trace. By default stack traces are not captured for
|
|
|
|
/// uncaught exceptions. SetCaptureStackTraceForUncaughtExceptions allows
|
|
|
|
/// to change this option.
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn get_stack_trace<'s>(
|
2019-12-31 11:17:26 -05:00
|
|
|
&self,
|
2020-06-03 01:38:34 -04:00
|
|
|
scope: &mut HandleScope<'s>,
|
|
|
|
) -> Option<Local<'s, StackTrace>> {
|
2020-05-31 09:00:23 -04:00
|
|
|
unsafe { scope.cast_local(|_| v8__Message__GetStackTrace(self)) }
|
2019-12-31 11:17:26 -05:00
|
|
|
}
|
|
|
|
|
2019-12-31 09:40:34 -05:00
|
|
|
pub fn get_source_line<'s>(
|
|
|
|
&self,
|
2020-06-03 01:38:34 -04:00
|
|
|
scope: &mut HandleScope<'s>,
|
2019-12-31 09:40:34 -05:00
|
|
|
) -> Option<Local<'s, String>> {
|
2020-06-25 20:56:24 -04:00
|
|
|
unsafe {
|
|
|
|
scope.cast_local(|sd| {
|
|
|
|
v8__Message__GetSourceLine(self, sd.get_current_context())
|
|
|
|
})
|
|
|
|
}
|
2019-12-31 09:40:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the resource name for the script from where the function causing
|
|
|
|
/// the error originates.
|
|
|
|
pub fn get_script_resource_name<'s>(
|
|
|
|
&self,
|
2020-06-03 01:38:34 -04:00
|
|
|
scope: &mut HandleScope<'s>,
|
2019-12-31 09:40:34 -05:00
|
|
|
) -> Option<Local<'s, Value>> {
|
2020-05-31 09:00:23 -04:00
|
|
|
unsafe { scope.cast_local(|_| v8__Message__GetScriptResourceName(self)) }
|
2019-12-31 09:40:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the number, 1-based, of the line where the error occurred.
|
2020-06-25 20:56:24 -04:00
|
|
|
pub fn get_line_number(&self, scope: &mut HandleScope) -> Option<usize> {
|
|
|
|
let i = unsafe {
|
|
|
|
v8__Message__GetLineNumber(self, &*scope.get_current_context())
|
|
|
|
};
|
2019-12-31 09:40:34 -05:00
|
|
|
if i < 0 {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(i as usize)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the index within the script of the first character where
|
|
|
|
/// the error occurred.
|
|
|
|
pub fn get_start_position(&self) -> int {
|
|
|
|
unsafe { v8__Message__GetStartPosition(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the index within the script of the last character where
|
|
|
|
/// the error occurred.
|
|
|
|
pub fn get_end_position(&self) -> int {
|
|
|
|
unsafe { v8__Message__GetEndPosition(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the Wasm function index where the error occurred. Returns -1 if
|
|
|
|
/// message is not from a Wasm script.
|
|
|
|
pub fn get_wasm_function_index(&self) -> int {
|
|
|
|
unsafe { v8__Message__GetWasmFunctionIndex(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the error level of the message.
|
|
|
|
pub fn error_level(&self) -> int {
|
|
|
|
unsafe { v8__Message__ErrorLevel(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the index within the line of the first character where
|
|
|
|
/// the error occurred.
|
|
|
|
pub fn get_start_column(&self) -> usize {
|
|
|
|
unsafe { v8__Message__GetStartColumn(self) as usize }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the index within the line of the last character where
|
|
|
|
/// the error occurred.
|
|
|
|
pub fn get_end_column(&self) -> usize {
|
|
|
|
unsafe { v8__Message__GetEndColumn(self) as usize }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Passes on the value set by the embedder when it fed the script from which
|
|
|
|
/// this Message was generated to V8.
|
|
|
|
pub fn is_shared_cross_origin(&self) -> bool {
|
|
|
|
unsafe { v8__Message__IsSharedCrossOrigin(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_opaque(&self) -> bool {
|
|
|
|
unsafe { v8__Message__IsOpaque(self) }
|
|
|
|
}
|
2019-12-08 20:26:58 -05:00
|
|
|
}
|
|
|
|
|
2020-01-17 02:22:11 -05:00
|
|
|
/// Create new error objects by calling the corresponding error object
|
|
|
|
/// constructor with the message.
|
2020-11-18 09:17:25 -05:00
|
|
|
#[derive(Debug)]
|
2020-01-17 02:22:11 -05:00
|
|
|
pub struct Exception;
|
2019-12-08 20:26:58 -05:00
|
|
|
|
2020-01-17 02:22:11 -05:00
|
|
|
impl Exception {
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn error<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
2020-01-17 02:22:11 -05:00
|
|
|
message: Local<String>,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Local<'s, Value> {
|
2020-05-31 14:41:43 -04:00
|
|
|
Self::new_error_with(scope, message, v8__Exception__Error)
|
2020-01-17 02:22:11 -05:00
|
|
|
}
|
2019-12-08 20:26:58 -05:00
|
|
|
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn range_error<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
2020-01-17 02:22:11 -05:00
|
|
|
message: Local<String>,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Local<'s, Value> {
|
2020-05-31 14:41:43 -04:00
|
|
|
Self::new_error_with(scope, message, v8__Exception__RangeError)
|
2020-01-17 02:22:11 -05:00
|
|
|
}
|
2019-12-08 20:26:58 -05:00
|
|
|
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn reference_error<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
2020-01-17 02:22:11 -05:00
|
|
|
message: Local<String>,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Local<'s, Value> {
|
2020-05-31 14:41:43 -04:00
|
|
|
Self::new_error_with(scope, message, v8__Exception__ReferenceError)
|
2020-01-17 02:22:11 -05:00
|
|
|
}
|
2019-12-08 20:26:58 -05:00
|
|
|
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn syntax_error<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
2020-01-17 02:22:11 -05:00
|
|
|
message: Local<String>,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Local<'s, Value> {
|
2020-05-31 14:41:43 -04:00
|
|
|
Self::new_error_with(scope, message, v8__Exception__SyntaxError)
|
2020-01-17 02:22:11 -05:00
|
|
|
}
|
2019-12-08 20:26:58 -05:00
|
|
|
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn type_error<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
2020-01-17 02:22:11 -05:00
|
|
|
message: Local<String>,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Local<'s, Value> {
|
2020-05-31 14:41:43 -04:00
|
|
|
Self::new_error_with(scope, message, v8__Exception__TypeError)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Internal helper to make the above error constructors less repetitive.
|
2020-06-03 01:38:34 -04:00
|
|
|
fn new_error_with<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
2020-05-31 14:41:43 -04:00
|
|
|
message: Local<String>,
|
|
|
|
contructor: unsafe extern "C" fn(*const String) -> *const Value,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Local<'s, Value> {
|
2021-02-19 17:01:18 -05:00
|
|
|
unsafe {
|
|
|
|
scope.enter();
|
|
|
|
let error = scope.cast_local(|_| (contructor)(&*message)).unwrap();
|
|
|
|
scope.exit();
|
|
|
|
error
|
|
|
|
}
|
2020-01-17 02:22:11 -05:00
|
|
|
}
|
2019-12-08 20:26:58 -05:00
|
|
|
|
2020-01-17 02:22:11 -05:00
|
|
|
/// Creates an error message for the given exception.
|
|
|
|
/// Will try to reconstruct the original stack trace from the exception value,
|
|
|
|
/// or capture the current stack trace if not available.
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn create_message<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
2020-01-17 02:22:11 -05:00
|
|
|
exception: Local<Value>,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Local<'s, Message> {
|
2020-05-31 09:00:23 -04:00
|
|
|
unsafe {
|
2020-06-03 01:38:34 -04:00
|
|
|
scope.cast_local(|sd| {
|
|
|
|
v8__Exception__CreateMessage(sd.get_isolate_ptr(), &*exception)
|
2020-05-31 09:00:23 -04:00
|
|
|
})
|
|
|
|
}
|
|
|
|
.unwrap()
|
2020-01-17 02:22:11 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the original stack trace that was captured at the creation time
|
|
|
|
/// of a given exception, or an empty handle if not available.
|
2020-06-03 01:38:34 -04:00
|
|
|
pub fn get_stack_trace<'s>(
|
|
|
|
scope: &mut HandleScope<'s>,
|
2020-01-17 02:22:11 -05:00
|
|
|
exception: Local<Value>,
|
2020-06-03 01:38:34 -04:00
|
|
|
) -> Option<Local<'s, StackTrace>> {
|
2020-05-31 09:00:23 -04:00
|
|
|
unsafe { scope.cast_local(|_| v8__Exception__GetStackTrace(&*exception)) }
|
2020-01-17 02:22:11 -05:00
|
|
|
}
|
2019-12-08 20:26:58 -05:00
|
|
|
}
|