mirror of https://github.com/boa-dev/boa.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
110 lines
3.9 KiB
110 lines
3.9 KiB
//! This module implements the global `SyntaxError` object. |
|
//! |
|
//! The `SyntaxError` object represents an error when trying to interpret syntactically invalid code. |
|
//! It is thrown when the JavaScript context encounters tokens or token order that does not conform |
|
//! to the syntax of the language when parsing code. |
|
//! |
|
//! More information: |
|
//! - [MDN documentation][mdn] |
|
//! - [ECMAScript reference][spec] |
|
//! |
|
//! [spec]: https://tc39.es/ecma262/#sec-native-error-types-used-in-this-standard-syntaxerror |
|
//! [mdn]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/SyntaxError |
|
|
|
use crate::{ |
|
builtins::{BuiltInBuilder, BuiltInConstructor, BuiltInObject, IntrinsicObject}, |
|
context::intrinsics::{Intrinsics, StandardConstructor, StandardConstructors}, |
|
object::{internal_methods::get_prototype_from_constructor, JsObject, ObjectData}, |
|
property::Attribute, |
|
realm::Realm, |
|
string::utf16, |
|
Context, JsArgs, JsResult, JsValue, |
|
}; |
|
use boa_profiler::Profiler; |
|
|
|
use super::{Error, ErrorKind}; |
|
|
|
/// JavaScript `SyntaxError` implementation. |
|
#[derive(Debug, Clone, Copy)] |
|
pub(crate) struct SyntaxError; |
|
|
|
impl IntrinsicObject for SyntaxError { |
|
fn init(realm: &Realm) { |
|
let _timer = Profiler::global().start_event(Self::NAME, "init"); |
|
|
|
let attribute = Attribute::WRITABLE | Attribute::NON_ENUMERABLE | Attribute::CONFIGURABLE; |
|
BuiltInBuilder::from_standard_constructor::<Self>(realm) |
|
.prototype(realm.intrinsics().constructors().error().constructor()) |
|
.inherits(Some(realm.intrinsics().constructors().error().prototype())) |
|
.property(utf16!("name"), Self::NAME, attribute) |
|
.property(utf16!("message"), "", attribute) |
|
.build(); |
|
} |
|
|
|
fn get(intrinsics: &Intrinsics) -> JsObject { |
|
Self::STANDARD_CONSTRUCTOR(intrinsics.constructors()).constructor() |
|
} |
|
} |
|
|
|
impl BuiltInObject for SyntaxError { |
|
const NAME: &'static str = "SyntaxError"; |
|
} |
|
|
|
impl BuiltInConstructor for SyntaxError { |
|
const LENGTH: usize = 1; |
|
|
|
const STANDARD_CONSTRUCTOR: fn(&StandardConstructors) -> &StandardConstructor = |
|
StandardConstructors::syntax_error; |
|
|
|
/// Create a new error object. |
|
fn constructor( |
|
new_target: &JsValue, |
|
args: &[JsValue], |
|
context: &mut Context<'_>, |
|
) -> JsResult<JsValue> { |
|
// 1. If NewTarget is undefined, let newTarget be the active function object; else let newTarget be NewTarget. |
|
let new_target = &if new_target.is_undefined() { |
|
context |
|
.vm |
|
.active_function |
|
.clone() |
|
.unwrap_or_else(|| { |
|
context |
|
.intrinsics() |
|
.constructors() |
|
.syntax_error() |
|
.constructor() |
|
}) |
|
.into() |
|
} else { |
|
new_target.clone() |
|
}; |
|
// 2. Let O be ? OrdinaryCreateFromConstructor(newTarget, "%NativeError.prototype%", « [[ErrorData]] »). |
|
let prototype = get_prototype_from_constructor( |
|
new_target, |
|
StandardConstructors::syntax_error, |
|
context, |
|
)?; |
|
let o = JsObject::from_proto_and_data_with_shared_shape( |
|
context.root_shape(), |
|
prototype, |
|
ObjectData::error(ErrorKind::Syntax), |
|
); |
|
|
|
// 3. If message is not undefined, then |
|
let message = args.get_or_undefined(0); |
|
if !message.is_undefined() { |
|
// a. Let msg be ? ToString(message). |
|
let msg = message.to_string(context)?; |
|
|
|
// b. Perform CreateNonEnumerableDataPropertyOrThrow(O, "message", msg). |
|
o.create_non_enumerable_data_property_or_throw(utf16!("message"), msg, context); |
|
} |
|
|
|
// 4. Perform ? InstallErrorCause(O, options). |
|
Error::install_error_cause(&o, args.get_or_undefined(1), context)?; |
|
|
|
// 5. Return O. |
|
Ok(o.into()) |
|
} |
|
}
|
|
|