// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license. use std::borrow::Cow; use std::collections::HashSet; use std::fmt; use std::fmt::Debug; use std::fmt::Display; use std::fmt::Formatter; use anyhow::Error; use crate::realm::JsRealm; use crate::runtime::GetErrorClassFn; use crate::runtime::JsRuntime; use crate::source_map::apply_source_map; use crate::source_map::get_source_line; use crate::url::Url; /// A generic wrapper that can encapsulate any concrete error type. // TODO(ry) Deprecate AnyError and encourage deno_core::anyhow::Error instead. pub type AnyError = anyhow::Error; /// Creates a new error with a caller-specified error class name and message. pub fn custom_error( class: &'static str, message: impl Into>, ) -> Error { CustomError { class, message: message.into(), } .into() } pub fn generic_error(message: impl Into>) -> Error { custom_error("Error", message) } pub fn type_error(message: impl Into>) -> Error { custom_error("TypeError", message) } pub fn range_error(message: impl Into>) -> Error { custom_error("RangeError", message) } pub fn invalid_hostname(hostname: &str) -> Error { type_error(format!("Invalid hostname: '{hostname}'")) } pub fn uri_error(message: impl Into>) -> Error { custom_error("URIError", message) } pub fn bad_resource(message: impl Into>) -> Error { custom_error("BadResource", message) } pub fn bad_resource_id() -> Error { custom_error("BadResource", "Bad resource ID") } pub fn not_supported() -> Error { custom_error("NotSupported", "The operation is not supported") } pub fn resource_unavailable() -> Error { custom_error( "Busy", "Resource is unavailable because it is in use by a promise", ) } /// A simple error type that lets the creator specify both the error message and /// the error class name. This type is private; externally it only ever appears /// wrapped in an `anyhow::Error`. To retrieve the error class name from a wrapped /// `CustomError`, use the function `get_custom_error_class()`. #[derive(Debug)] struct CustomError { class: &'static str, message: Cow<'static, str>, } impl Display for CustomError { fn fmt(&self, f: &mut Formatter) -> fmt::Result { f.write_str(&self.message) } } impl std::error::Error for CustomError {} /// If this error was crated with `custom_error()`, return the specified error /// class name. In all other cases this function returns `None`. pub fn get_custom_error_class(error: &Error) -> Option<&'static str> { error.downcast_ref::().map(|e| e.class) } pub fn to_v8_error<'a>( scope: &mut v8::HandleScope<'a>, get_class: GetErrorClassFn, error: &Error, ) -> v8::Local<'a, v8::Value> { let tc_scope = &mut v8::TryCatch::new(scope); let cb = JsRealm::state_from_scope(tc_scope) .borrow() .js_build_custom_error_cb .clone() .expect("Custom error builder must be set"); let cb = cb.open(tc_scope); let this = v8::undefined(tc_scope).into(); let class = v8::String::new(tc_scope, get_class(error)).unwrap(); let message = v8::String::new(tc_scope, &format!("{error:#}")).unwrap(); let mut args = vec![class.into(), message.into()]; if let Some(code) = crate::error_codes::get_error_code(error) { args.push(v8::String::new(tc_scope, code).unwrap().into()); } let maybe_exception = cb.call(tc_scope, this, &args); match maybe_exception { Some(exception) => exception, None => { let mut msg = "Custom error class must have a builder registered".to_string(); if tc_scope.has_caught() { let e = tc_scope.exception().unwrap(); let js_error = JsError::from_v8_exception(tc_scope, e); msg = format!("{}: {}", msg, js_error.exception_message); } panic!("{}", msg); } } } /// A `JsError` represents an exception coming from V8, with stack frames and /// line numbers. The deno_cli crate defines another `JsError` type, which wraps /// the one defined here, that adds source map support and colorful formatting. /// When updating this struct, also update errors_are_equal_without_cause() in /// fmt_error.rs. #[derive(Debug, PartialEq, Clone, serde::Deserialize, serde::Serialize)] #[serde(rename_all = "camelCase")] pub struct JsError { pub name: Option, pub message: Option, pub stack: Option, pub cause: Option>, pub exception_message: String, pub frames: Vec, pub source_line: Option, pub source_line_frame_index: Option, pub aggregated: Option>, } #[derive(Debug, Eq, PartialEq, Clone, serde::Deserialize, serde::Serialize)] #[serde(rename_all = "camelCase")] pub struct JsStackFrame { pub type_name: Option, pub function_name: Option, pub method_name: Option, pub file_name: Option, pub line_number: Option, pub column_number: Option, pub eval_origin: Option, // Warning! isToplevel has inconsistent snake<>camel case, "typo" originates in v8: // https://source.chromium.org/search?q=isToplevel&sq=&ss=chromium%2Fchromium%2Fsrc:v8%2F #[serde(rename = "isToplevel")] pub is_top_level: Option, pub is_eval: bool, pub is_native: bool, pub is_constructor: bool, pub is_async: bool, pub is_promise_all: bool, pub promise_index: Option, } impl JsStackFrame { pub fn from_location( file_name: Option, line_number: Option, column_number: Option, ) -> Self { Self { type_name: None, function_name: None, method_name: None, file_name, line_number, column_number, eval_origin: None, is_top_level: None, is_eval: false, is_native: false, is_constructor: false, is_async: false, is_promise_all: false, promise_index: None, } } /// Gets the source mapped stack frame corresponding to the /// (script_resource_name, line_number, column_number) from a v8 message. /// For non-syntax errors, it should also correspond to the first stack frame. pub fn from_v8_message<'a>( scope: &'a mut v8::HandleScope, message: v8::Local<'a, v8::Message>, ) -> Option { let f = message.get_script_resource_name(scope)?; let f: v8::Local = f.try_into().ok()?; let f = f.to_rust_string_lossy(scope); let l = message.get_line_number(scope)? as i64; // V8's column numbers are 0-based, we want 1-based. let c = message.get_start_column() as i64 + 1; let state_rc = JsRuntime::state_from(scope); let (getter, cache) = { let state = state_rc.borrow(); ( state.source_map_getter.clone(), state.source_map_cache.clone(), ) }; if let Some(source_map_getter) = getter { let mut cache = cache.borrow_mut(); let (f, l, c) = apply_source_map(f, l, c, &mut cache, &**source_map_getter); Some(JsStackFrame::from_location(Some(f), Some(l), Some(c))) } else { Some(JsStackFrame::from_location(Some(f), Some(l), Some(c))) } } pub fn maybe_format_location(&self) -> Option { Some(format!( "{}:{}:{}", self.file_name.as_ref()?, self.line_number?, self.column_number? )) } } fn get_property<'a>( scope: &mut v8::HandleScope<'a>, object: v8::Local, key: &str, ) -> Option> { let key = v8::String::new(scope, key).unwrap(); object.get(scope, key.into()) } #[derive(Default, serde::Deserialize)] pub(crate) struct NativeJsError { pub name: Option, pub message: Option, // Warning! .stack is special so handled by itself // stack: Option, } impl JsError { pub fn from_v8_exception( scope: &mut v8::HandleScope, exception: v8::Local, ) -> Self { Self::inner_from_v8_exception(scope, exception, Default::default()) } pub fn from_v8_message<'a>( scope: &'a mut v8::HandleScope, msg: v8::Local<'a, v8::Message>, ) -> Self { // Create a new HandleScope because we're creating a lot of new local // handles below. let scope = &mut v8::HandleScope::new(scope); let exception_message = msg.get(scope).to_rust_string_lossy(scope); // Convert them into Vec let mut frames: Vec = vec![]; let mut source_line = None; let mut source_line_frame_index = None; if let Some(stack_frame) = JsStackFrame::from_v8_message(scope, msg) { frames = vec![stack_frame]; } { let state_rc = JsRuntime::state_from(scope); let (getter, cache) = { let state = state_rc.borrow(); ( state.source_map_getter.clone(), state.source_map_cache.clone(), ) }; if let Some(source_map_getter) = getter { let mut cache = cache.borrow_mut(); for (i, frame) in frames.iter().enumerate() { if let (Some(file_name), Some(line_number)) = (&frame.file_name, frame.line_number) { if !file_name.trim_start_matches('[').starts_with("ext:") { source_line = get_source_line( file_name, line_number, &mut cache, &**source_map_getter, ); source_line_frame_index = Some(i); break; } } } } } Self { name: None, message: None, exception_message, cause: None, source_line, source_line_frame_index, frames, stack: None, aggregated: None, } } fn inner_from_v8_exception<'a>( scope: &'a mut v8::HandleScope, exception: v8::Local<'a, v8::Value>, mut seen: HashSet>, ) -> Self { // Create a new HandleScope because we're creating a lot of new local // handles below. let scope = &mut v8::HandleScope::new(scope); let msg = v8::Exception::create_message(scope, exception); let mut exception_message = None; let context_state_rc = JsRealm::state_from_scope(scope); let js_format_exception_cb = context_state_rc.borrow().js_format_exception_cb.clone(); if let Some(format_exception_cb) = js_format_exception_cb { let format_exception_cb = format_exception_cb.open(scope); let this = v8::undefined(scope).into(); let formatted = format_exception_cb.call(scope, this, &[exception]); if let Some(formatted) = formatted { if formatted.is_string() { exception_message = Some(formatted.to_rust_string_lossy(scope)); } } } if is_instance_of_error(scope, exception) { let v8_exception = exception; // The exception is a JS Error object. let exception: v8::Local = exception.try_into().unwrap(); let cause = get_property(scope, exception, "cause"); let e: NativeJsError = serde_v8::from_v8(scope, exception.into()).unwrap_or_default(); // Get the message by formatting error.name and error.message. let name = e.name.clone().unwrap_or_else(|| "Error".to_string()); let message_prop = e.message.clone().unwrap_or_default(); let exception_message = exception_message.unwrap_or_else(|| { if !name.is_empty() && !message_prop.is_empty() { format!("Uncaught {name}: {message_prop}") } else if !name.is_empty() { format!("Uncaught {name}") } else if !message_prop.is_empty() { format!("Uncaught {message_prop}") } else { "Uncaught".to_string() } }); let cause = cause.and_then(|cause| { if cause.is_undefined() || seen.contains(&exception) { None } else { seen.insert(exception); Some(Box::new(JsError::inner_from_v8_exception( scope, cause, seen, ))) } }); // Access error.stack to ensure that prepareStackTrace() has been called. // This should populate error.__callSiteEvals. let stack = get_property(scope, exception, "stack"); let stack: Option> = stack.and_then(|s| s.try_into().ok()); let stack = stack.map(|s| s.to_rust_string_lossy(scope)); // Read an array of structured frames from error.__callSiteEvals. let frames_v8 = get_property(scope, exception, "__callSiteEvals"); // Ignore non-array values let frames_v8: Option> = frames_v8.and_then(|a| a.try_into().ok()); // Convert them into Vec let mut frames: Vec = match frames_v8 { Some(frames_v8) => serde_v8::from_v8(scope, frames_v8.into()).unwrap(), None => vec![], }; let mut source_line = None; let mut source_line_frame_index = None; // When the stack frame array is empty, but the source location given by // (script_resource_name, line_number, start_column + 1) exists, this is // likely a syntax error. For the sake of formatting we treat it like it // was given as a single stack frame. if frames.is_empty() { if let Some(stack_frame) = JsStackFrame::from_v8_message(scope, msg) { frames = vec![stack_frame]; } } { let state_rc = JsRuntime::state_from(scope); let (getter, cache) = { let state = state_rc.borrow(); ( state.source_map_getter.clone(), state.source_map_cache.clone(), ) }; if let Some(source_map_getter) = getter { let mut cache = cache.borrow_mut(); for (i, frame) in frames.iter().enumerate() { if let (Some(file_name), Some(line_number)) = (&frame.file_name, frame.line_number) { if !file_name.trim_start_matches('[').starts_with("ext:") { source_line = get_source_line( file_name, line_number, &mut cache, &**source_map_getter, ); source_line_frame_index = Some(i); break; } } } } else if let Some(frame) = frames.first() { if let Some(file_name) = &frame.file_name { if !file_name.trim_start_matches('[').starts_with("ext:") { source_line = msg .get_source_line(scope) .map(|v| v.to_rust_string_lossy(scope)); source_line_frame_index = Some(0); } } } } let mut aggregated: Option> = None; if is_aggregate_error(scope, v8_exception) { // Read an array of stored errors, this is only defined for `AggregateError` let aggregated_errors = get_property(scope, exception, "errors"); let aggregated_errors: Option> = aggregated_errors.and_then(|a| a.try_into().ok()); if let Some(errors) = aggregated_errors { if errors.length() > 0 { let mut agg = vec![]; for i in 0..errors.length() { let error = errors.get_index(scope, i).unwrap(); let js_error = Self::from_v8_exception(scope, error); agg.push(js_error); } aggregated = Some(agg); } } }; Self { name: e.name, message: e.message, exception_message, cause, source_line, source_line_frame_index, frames, stack, aggregated, } } else { let exception_message = exception_message .unwrap_or_else(|| msg.get(scope).to_rust_string_lossy(scope)); // The exception is not a JS Error object. // Get the message given by V8::Exception::create_message(), and provide // empty frames. Self { name: None, message: None, exception_message, cause: None, source_line: None, source_line_frame_index: None, frames: vec![], stack: None, aggregated: None, } } } } impl std::error::Error for JsError {} impl Display for JsError { fn fmt(&self, f: &mut Formatter) -> fmt::Result { if let Some(stack) = &self.stack { let stack_lines = stack.lines(); if stack_lines.count() > 1 { return write!(f, "{stack}"); } } write!(f, "{}", self.exception_message)?; let location = self.frames.first().and_then(|f| f.maybe_format_location()); if let Some(location) = location { write!(f, "\n at {location}")?; } Ok(()) } } // TODO(piscisaureus): rusty_v8 should implement the Error trait on // values of type v8::Global. pub(crate) fn to_v8_type_error( scope: &mut v8::HandleScope, err: Error, ) -> v8::Global { let err_string = err.to_string(); let error_chain = err .chain() .skip(1) .filter(|e| e.to_string() != err_string) .map(|e| e.to_string()) .collect::>(); let message = if !error_chain.is_empty() { format!( "{}\n Caused by:\n {}", err_string, error_chain.join("\n ") ) } else { err_string }; let message = v8::String::new(scope, &message).unwrap(); let exception = v8::Exception::type_error(scope, message); v8::Global::new(scope, exception) } /// Implements `value instanceof primordials.Error` in JS. Similar to /// `Value::is_native_error()` but more closely matches the semantics /// of `instanceof`. `Value::is_native_error()` also checks for static class /// inheritance rather than just scanning the prototype chain, which doesn't /// work with our WebIDL implementation of `DOMException`. pub(crate) fn is_instance_of_error( scope: &mut v8::HandleScope, value: v8::Local, ) -> bool { if !value.is_object() { return false; } let message = v8::String::empty(scope); let error_prototype = v8::Exception::error(scope, message) .to_object(scope) .unwrap() .get_prototype(scope) .unwrap(); let mut maybe_prototype = value.to_object(scope).unwrap().get_prototype(scope); while let Some(prototype) = maybe_prototype { if !prototype.is_object() { return false; } if prototype.strict_equals(error_prototype) { return true; } maybe_prototype = prototype .to_object(scope) .and_then(|o| o.get_prototype(scope)); } false } /// Implements `value instanceof primordials.AggregateError` in JS, /// by walking the prototype chain, and comparing each links constructor `name` property. /// /// NOTE: There is currently no way to detect `AggregateError` via `rusty_v8`, /// as v8 itself doesn't expose `v8__Exception__AggregateError`, /// and we cannot create bindings for it. This forces us to rely on `name` inference. pub(crate) fn is_aggregate_error( scope: &mut v8::HandleScope, value: v8::Local, ) -> bool { let mut maybe_prototype = Some(value); while let Some(prototype) = maybe_prototype { if !prototype.is_object() { return false; } let prototype = prototype.to_object(scope).unwrap(); let prototype_name = match get_property(scope, prototype, "constructor") { Some(constructor) => { let ctor = constructor.to_object(scope).unwrap(); get_property(scope, ctor, "name").map(|v| v.to_rust_string_lossy(scope)) } None => return false, }; if prototype_name == Some(String::from("AggregateError")) { return true; } maybe_prototype = prototype.get_prototype(scope); } false } const DATA_URL_ABBREV_THRESHOLD: usize = 150; pub fn format_file_name(file_name: &str) -> String { abbrev_file_name(file_name).unwrap_or_else(|| file_name.to_string()) } fn abbrev_file_name(file_name: &str) -> Option { if file_name.len() <= DATA_URL_ABBREV_THRESHOLD { return None; } let url = Url::parse(file_name).ok()?; if url.scheme() != "data" { return None; } let (head, tail) = url.path().split_once(',')?; let len = tail.len(); let start = tail.get(0..20)?; let end = tail.get(len - 20..)?; Some(format!("{}:{},{}......{}", url.scheme(), head, start, end)) } #[cfg(test)] mod tests { use super::*; #[test] fn test_bad_resource() { let err = bad_resource("Resource has been closed"); assert_eq!(err.to_string(), "Resource has been closed"); } #[test] fn test_bad_resource_id() { let err = bad_resource_id(); assert_eq!(err.to_string(), "Bad resource ID"); } }