mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-05-18 17:12:54 +00:00
Note that as of this commit, there aren't any such throwers, and the call site in Heap::allocate will drop exceptions on the floor. This commit only serves to change the declaration of the overrides, make sure they return an empty value, and to propagate OOM errors frm their base initialize invocations.
241 lines
10 KiB
C++
241 lines
10 KiB
C++
/*
|
||
* Copyright (c) 2021-2023, Linus Groh <linusg@serenityos.org>
|
||
*
|
||
* SPDX-License-Identifier: BSD-2-Clause
|
||
*/
|
||
|
||
#include <AK/Time.h>
|
||
#include <LibCrypto/BigInt/SignedBigInteger.h>
|
||
#include <LibJS/Runtime/Completion.h>
|
||
#include <LibJS/Runtime/Date.h>
|
||
#include <LibJS/Runtime/GlobalObject.h>
|
||
#include <LibJS/Runtime/Temporal/Calendar.h>
|
||
#include <LibJS/Runtime/Temporal/Instant.h>
|
||
#include <LibJS/Runtime/Temporal/Now.h>
|
||
#include <LibJS/Runtime/Temporal/PlainDate.h>
|
||
#include <LibJS/Runtime/Temporal/PlainDateTime.h>
|
||
#include <LibJS/Runtime/Temporal/PlainTime.h>
|
||
#include <LibJS/Runtime/Temporal/TimeZone.h>
|
||
#include <LibJS/Runtime/Temporal/ZonedDateTime.h>
|
||
|
||
namespace JS::Temporal {
|
||
|
||
// 2 The Temporal.Now Object, https://tc39.es/proposal-temporal/#sec-temporal-now-object
|
||
Now::Now(Realm& realm)
|
||
: Object(ConstructWithPrototypeTag::Tag, *realm.intrinsics().object_prototype())
|
||
{
|
||
}
|
||
|
||
ThrowCompletionOr<void> Now::initialize(Realm& realm)
|
||
{
|
||
MUST_OR_THROW_OOM(Base::initialize(realm));
|
||
|
||
auto& vm = this->vm();
|
||
|
||
// 2.1.1 Temporal.Now [ @@toStringTag ], https://tc39.es/proposal-temporal/#sec-temporal-now-@@tostringtag
|
||
define_direct_property(*vm.well_known_symbol_to_string_tag(), PrimitiveString::create(vm, "Temporal.Now"), Attribute::Configurable);
|
||
|
||
u8 attr = Attribute::Writable | Attribute::Configurable;
|
||
define_native_function(realm, vm.names.timeZone, time_zone, 0, attr);
|
||
define_native_function(realm, vm.names.instant, instant, 0, attr);
|
||
define_native_function(realm, vm.names.plainDateTime, plain_date_time, 1, attr);
|
||
define_native_function(realm, vm.names.plainDateTimeISO, plain_date_time_iso, 0, attr);
|
||
define_native_function(realm, vm.names.zonedDateTime, zoned_date_time, 1, attr);
|
||
define_native_function(realm, vm.names.zonedDateTimeISO, zoned_date_time_iso, 0, attr);
|
||
define_native_function(realm, vm.names.plainDate, plain_date, 1, attr);
|
||
define_native_function(realm, vm.names.plainDateISO, plain_date_iso, 0, attr);
|
||
define_native_function(realm, vm.names.plainTimeISO, plain_time_iso, 0, attr);
|
||
|
||
return {};
|
||
}
|
||
|
||
// 2.2.1 Temporal.Now.timeZone ( ), https://tc39.es/proposal-temporal/#sec-temporal.now.timezone
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::time_zone)
|
||
{
|
||
// 1. Return ! SystemTimeZone().
|
||
return system_time_zone(vm);
|
||
}
|
||
|
||
// 2.2.2 Temporal.Now.instant ( ), https://tc39.es/proposal-temporal/#sec-temporal.now.instant
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::instant)
|
||
{
|
||
// 1. Return ! SystemInstant().
|
||
return system_instant(vm);
|
||
}
|
||
|
||
// 2.2.3 Temporal.Now.plainDateTime ( calendarLike [ , temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaindatetime
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_date_time)
|
||
{
|
||
auto calendar_like = vm.argument(0);
|
||
auto temporal_time_zone_like = vm.argument(1);
|
||
|
||
// 1. Return ? SystemDateTime(temporalTimeZoneLike, calendarLike).
|
||
return TRY(system_date_time(vm, temporal_time_zone_like, calendar_like));
|
||
}
|
||
|
||
// 2.2.4 Temporal.Now.plainDateTimeISO ( [ temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaindatetimeiso
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_date_time_iso)
|
||
{
|
||
auto temporal_time_zone_like = vm.argument(0);
|
||
|
||
// 1, Let calendar be ! GetISO8601Calendar().
|
||
auto* calendar = get_iso8601_calendar(vm);
|
||
|
||
// 2. Return ? SystemDateTime(temporalTimeZoneLike, calendar).
|
||
return TRY(system_date_time(vm, temporal_time_zone_like, calendar));
|
||
}
|
||
|
||
// 2.2.5 Temporal.Now.zonedDateTime ( calendarLike [ , temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.zoneddatetime
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::zoned_date_time)
|
||
{
|
||
auto calendar_like = vm.argument(0);
|
||
auto temporal_time_zone_like = vm.argument(1);
|
||
|
||
// 1. Return ? SystemZonedDateTime(temporalTimeZoneLike, calendarLike).
|
||
return TRY(system_zoned_date_time(vm, temporal_time_zone_like, calendar_like));
|
||
}
|
||
|
||
// 2.2.6 Temporal.Now.zonedDateTimeISO ( [ temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.zoneddatetimeiso
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::zoned_date_time_iso)
|
||
{
|
||
auto temporal_time_zone_like = vm.argument(0);
|
||
|
||
// 1, Let calendar be ! GetISO8601Calendar().
|
||
auto* calendar = get_iso8601_calendar(vm);
|
||
|
||
// 2. Return ? SystemZonedDateTime(temporalTimeZoneLike, calendar).
|
||
return TRY(system_zoned_date_time(vm, temporal_time_zone_like, calendar));
|
||
}
|
||
|
||
// 2.2.7 Temporal.Now.plainDate ( calendarLike [ , temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaindate
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_date)
|
||
{
|
||
auto calendar_like = vm.argument(0);
|
||
auto temporal_time_zone_like = vm.argument(1);
|
||
|
||
// 1. Let dateTime be ? SystemDateTime(temporalTimeZoneLike, calendarLike).
|
||
auto* date_time = TRY(system_date_time(vm, temporal_time_zone_like, calendar_like));
|
||
|
||
// 2. Return ! CreateTemporalDate(dateTime.[[ISOYear]], dateTime.[[ISOMonth]], dateTime.[[ISODay]], dateTime.[[Calendar]]).
|
||
return MUST(create_temporal_date(vm, date_time->iso_year(), date_time->iso_month(), date_time->iso_day(), date_time->calendar()));
|
||
}
|
||
|
||
// 2.2.8 Temporal.Now.plainDateISO ( [ temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaindateiso
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_date_iso)
|
||
{
|
||
auto temporal_time_zone_like = vm.argument(0);
|
||
|
||
// 1. Let calendar be ! GetISO8601Calendar().
|
||
auto* calendar = get_iso8601_calendar(vm);
|
||
|
||
// 2. Let dateTime be ? SystemDateTime(temporalTimeZoneLike, calendar).
|
||
auto* date_time = TRY(system_date_time(vm, temporal_time_zone_like, calendar));
|
||
|
||
// 3. Return ! CreateTemporalDate(dateTime.[[ISOYear]], dateTime.[[ISOMonth]], dateTime.[[ISODay]], dateTime.[[Calendar]]).
|
||
return MUST(create_temporal_date(vm, date_time->iso_year(), date_time->iso_month(), date_time->iso_day(), date_time->calendar()));
|
||
}
|
||
|
||
// 2.2.9 Temporal.Now.plainTimeISO ( [ temporalTimeZoneLike ] ), https://tc39.es/proposal-temporal/#sec-temporal.now.plaintimeiso
|
||
JS_DEFINE_NATIVE_FUNCTION(Now::plain_time_iso)
|
||
{
|
||
auto temporal_time_zone_like = vm.argument(0);
|
||
|
||
// 1. Let calendar be ! GetISO8601Calendar().
|
||
auto* calendar = get_iso8601_calendar(vm);
|
||
|
||
// 2. Let dateTime be ? SystemDateTime(temporalTimeZoneLike, calendar).
|
||
auto* date_time = TRY(system_date_time(vm, temporal_time_zone_like, calendar));
|
||
|
||
// 3. Return ! CreateTemporalTime(dateTime.[[ISOHour]], dateTime.[[ISOMinute]], dateTime.[[ISOSecond]], dateTime.[[ISOMillisecond]], dateTime.[[ISOMicrosecond]], dateTime.[[ISONanosecond]]).
|
||
return MUST(create_temporal_time(vm, date_time->iso_hour(), date_time->iso_minute(), date_time->iso_second(), date_time->iso_millisecond(), date_time->iso_microsecond(), date_time->iso_nanosecond()));
|
||
}
|
||
|
||
// 2.3.1 SystemTimeZone ( ), https://tc39.es/proposal-temporal/#sec-temporal-systemtimezone
|
||
TimeZone* system_time_zone(VM& vm)
|
||
{
|
||
// 1. Let identifier be ! DefaultTimeZone().
|
||
auto identifier = default_time_zone();
|
||
|
||
// 2. Return ! CreateTemporalTimeZone(identifier).
|
||
// FIXME: Propagate possible OOM error
|
||
return MUST(create_temporal_time_zone(vm, identifier));
|
||
}
|
||
|
||
// 2.3.2 SystemUTCEpochNanoseconds ( ), https://tc39.es/proposal-temporal/#sec-temporal-systemutcepochnanoseconds
|
||
BigInt* system_utc_epoch_nanoseconds(VM& vm)
|
||
{
|
||
// 1. Let ns be the approximate current UTC date and time, in nanoseconds since the epoch.
|
||
auto now = Time::now_realtime().to_nanoseconds();
|
||
auto ns = Crypto::SignedBigInteger { now };
|
||
|
||
// 2. Set ns to the result of clamping ns between nsMinInstant and nsMaxInstant.
|
||
// NOTE: Time::to_nanoseconds() already clamps between -(2^63) and 2^63 - 1, the range of an i64,
|
||
// if an overflow occurs during seconds -> nanoseconds conversion.
|
||
|
||
// 3. Return ℤ(ns).
|
||
return BigInt::create(vm, move(ns));
|
||
}
|
||
|
||
// 2.3.3 SystemInstant ( ), https://tc39.es/proposal-temporal/#sec-temporal-systeminstant
|
||
Instant* system_instant(VM& vm)
|
||
{
|
||
// 1. Let ns be ! SystemUTCEpochNanoseconds().
|
||
auto* ns = system_utc_epoch_nanoseconds(vm);
|
||
|
||
// 2. Return ! CreateTemporalInstant(ns).
|
||
return MUST(create_temporal_instant(vm, *ns));
|
||
}
|
||
|
||
// 2.3.4 SystemDateTime ( temporalTimeZoneLike, calendarLike ), https://tc39.es/proposal-temporal/#sec-temporal-systemdatetime
|
||
ThrowCompletionOr<PlainDateTime*> system_date_time(VM& vm, Value temporal_time_zone_like, Value calendar_like)
|
||
{
|
||
Object* time_zone;
|
||
|
||
// 1. If temporalTimeZoneLike is undefined, then
|
||
if (temporal_time_zone_like.is_undefined()) {
|
||
// a. Let timeZone be ! SystemTimeZone().
|
||
time_zone = system_time_zone(vm);
|
||
}
|
||
// 2. Else,
|
||
else {
|
||
// a. Let timeZone be ? ToTemporalTimeZone(temporalTimeZoneLike).
|
||
time_zone = TRY(to_temporal_time_zone(vm, temporal_time_zone_like));
|
||
}
|
||
|
||
// 3. Let calendar be ? ToTemporalCalendar(calendarLike).
|
||
auto* calendar = TRY(to_temporal_calendar(vm, calendar_like));
|
||
|
||
// 4. Let instant be ! SystemInstant().
|
||
auto* instant = system_instant(vm);
|
||
|
||
// 5. Return ? BuiltinTimeZoneGetPlainDateTimeFor(timeZone, instant, calendar).
|
||
return builtin_time_zone_get_plain_date_time_for(vm, time_zone, *instant, *calendar);
|
||
}
|
||
|
||
// 2.3.5 SystemZonedDateTime ( temporalTimeZoneLike, calendarLike ), https://tc39.es/proposal-temporal/#sec-temporal-systemzoneddatetime
|
||
ThrowCompletionOr<ZonedDateTime*> system_zoned_date_time(VM& vm, Value temporal_time_zone_like, Value calendar_like)
|
||
{
|
||
Object* time_zone;
|
||
|
||
// 1. If temporalTimeZoneLike is undefined, then
|
||
if (temporal_time_zone_like.is_undefined()) {
|
||
// a. Let timeZone be ! SystemTimeZone().
|
||
time_zone = system_time_zone(vm);
|
||
}
|
||
// 2. Else,
|
||
else {
|
||
// a. Let timeZone be ? ToTemporalTimeZone(temporalTimeZoneLike).
|
||
time_zone = TRY(to_temporal_time_zone(vm, temporal_time_zone_like));
|
||
}
|
||
|
||
// 3. Let calendar be ? ToTemporalCalendar(calendarLike).
|
||
auto* calendar = TRY(to_temporal_calendar(vm, calendar_like));
|
||
|
||
// 4. Let ns be ! SystemUTCEpochNanoseconds().
|
||
auto* ns = system_utc_epoch_nanoseconds(vm);
|
||
|
||
// 5. Return ? CreateTemporalZonedDateTime(ns, timeZone, calendar).
|
||
return create_temporal_zoned_date_time(vm, *ns, *time_zone, *calendar);
|
||
}
|
||
|
||
}
|