Skip to content

Commit

Permalink
Implement String.prototype.isWellFormed and `String.prototype.toWel…
Browse files Browse the repository at this point in the history
…lFormed` (boa-dev#3187)
  • Loading branch information
raskad authored Jul 30, 2023
1 parent 2b01ef1 commit 4f9175b
Show file tree
Hide file tree
Showing 2 changed files with 71 additions and 2 deletions.
71 changes: 71 additions & 0 deletions boa_engine/src/builtins/string/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -123,6 +123,7 @@ impl IntrinsicObject for String {
.method(Self::ends_with, "endsWith", 1)
.method(Self::includes, "includes", 1)
.method(Self::index_of, "indexOf", 1)
.method(Self::is_well_formed, "isWellFormed", 0)
.method(Self::last_index_of, "lastIndexOf", 1)
.method(Self::locale_compare, "localeCompare", 1)
.method(Self::r#match, "match", 1)
Expand All @@ -132,6 +133,7 @@ impl IntrinsicObject for String {
.method(Self::trim, "trim", 0)
.method(Self::to_case::<false>, "toLowerCase", 0)
.method(Self::to_case::<true>, "toUpperCase", 0)
.method(Self::to_well_formed, "toWellFormed", 0)
.method(Self::to_locale_case::<false>, "toLocaleLowerCase", 0)
.method(Self::to_locale_case::<true>, "toLocaleUpperCase", 0)
.method(Self::substring, "substring", 2)
Expand Down Expand Up @@ -1289,6 +1291,34 @@ impl String {
.into())
}

/// `String.prototype.isWellFormed ( )`
///
/// More information:
/// - [ECMAScript reference][spec]
/// - [MDN documentation][mdn]
///
/// [spec]: https://tc39.es/ecma262/#sec-string.prototype.iswellformed
/// [mdn]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/isWellFormed
pub(crate) fn is_well_formed(
this: &JsValue,
_: &[JsValue],
context: &mut Context<'_>,
) -> JsResult<JsValue> {
// 1. Let O be ? RequireObjectCoercible(this value).
let o = this.require_object_coercible()?;

// 2. Let S be ? ToString(O).
let s = o.to_string(context)?;

// 3. Return IsStringWellFormedUnicode(S).
for code_point in s.code_points() {
if let CodePoint::UnpairedSurrogate(_) = code_point {
return Ok(false.into());
}
}
Ok(true.into())
}

/// `String.prototype.lastIndexOf( searchValue[, fromIndex] )`
///
/// The `lastIndexOf()` method returns the index within the calling `String` object of the last occurrence
Expand Down Expand Up @@ -1772,6 +1802,47 @@ impl String {
}
}

/// `String.prototype.toWellFormed ( )`
///
/// More information:
/// - [ECMAScript reference][spec]
/// - [MDN documentation][mdn]
///
/// [spec]: https://tc39.es/ecma262/#sec-string.prototype.towellformed
/// [mdn]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/toWellFormed
pub(crate) fn to_well_formed(
this: &JsValue,
_: &[JsValue],
context: &mut Context<'_>,
) -> JsResult<JsValue> {
// 1. Let O be ? RequireObjectCoercible(this value).
let o = this.require_object_coercible()?;

// 2. Let S be ? ToString(O).
let s = o.to_string(context)?;

// 3. Let strLen be the length of S.
// 4. Let k be 0.
// 5. Let result be the empty String.
// 6. Repeat, while k < strLen,
// a. Let cp be CodePointAt(S, k).
// b. If cp.[[IsUnpairedSurrogate]] is true, then
// i. Set result to the string-concatenation of result and 0xFFFD (REPLACEMENT CHARACTER).
// c. Else,
// i. Set result to the string-concatenation of result and UTF16EncodeCodePoint(cp.[[CodePoint]]).
// d. Set k to k + cp.[[CodeUnitCount]].
let result = s
.code_points()
.map(|code_point| match code_point {
CodePoint::UnpairedSurrogate(_) => '\u{FFFD}',
CodePoint::Unicode(char) => char,
})
.collect::<std::string::String>();

// 7. Return result.
Ok(result.into())
}

/// `String.prototype.substring( indexStart[, indexEnd] )`
///
/// The `substring()` method returns the part of the `string` between the start and end indexes, or to the end of the string.
Expand Down
2 changes: 0 additions & 2 deletions test_ignore.toml
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,6 @@ features = [
"IsHTMLDDA",
"Atomics",
"change-array-by-copy",
"String.prototype.isWellFormed",
"String.prototype.toWellFormed",
"symbols-as-weakmap-keys",
"intl-normative-optional",
"Intl.DisplayNames",
Expand Down

0 comments on commit 4f9175b

Please sign in to comment.