Nullable

Enum Nullable 

Source
pub enum Nullable<T> {
    Null,
    NonNull(T),
}
Expand description

A value that can be null in JavaScript, but not undefined.

This is used to distinguish between values in JavaScript that can be null (i.e. Nullable<T>) vs. those of which that can be undefined (i.e. Option<T>).

Nullable<T> tries to be close in API surface to the standard Option<T>, though with a much smaller API scope, and can be transformed into and from Option<T> for free.

§Why a new type?

There are in the standard 2 different values that are “null coalescing”:

  1. undefined, which is its own type (typeof void 0 == "undefined"),
  2. null, which is different from undefined (undefined !== null) but is of type object (typeof null === "object") for legacy reasons.

Option<T> sets up the first case, so we needed to add a new type for the second case.

§Would it be bad to use Option::<T>::None as both undefined or null?

Many values in the standard can be null but not undefined, or the other way around. Coalescing these two into Option::None means that any conversions using boa’s traits (such as TryFromJs) would result in indistinguishable values. The only way to respect the standard in this case would be to get JsValue and manually verify it’s not undefined/null, then do the conversion. This new type makes this process much simpler.

It also means that there is an asymmetry between JsValue to Option<T> then Option<T> back to JsValue, leading to unintuitive errors. Having a type Nullable<T> that acts like Option<T> but (de-)serialize to null clarifies all usage.

§How can I do EitherNullOrUndefined<T>?

The best way is to use Nullable<Option<T>> and convert into Option<T> using Nullable::flatten(). Please note that the reverse (Nullable<Option<T>>) results in the same deserializing behaviour but does not implement flatten().

Please note that JavaScript cannot make a distinction between Option<T> and Option<Option<T>>. This cannot be resolved using this type, as it suffers from the same limitation. There is no way to distinguish between Null and NonNull(Null). Nullable<Nullable<T>> does not provide additional information.

§Examples

let maybe_10: Nullable<u8> = JsValue::new(10).try_js_into(context).unwrap();
assert_eq!(maybe_10, Nullable::NonNull(10u8));

let maybe_not: Nullable<u8> = JsValue::null().try_js_into(context).unwrap();
assert_eq!(maybe_not, Nullable::Null);
let mut v: JsResult<Nullable<Option<u8>>> =
    JsValue::undefined().try_js_into(context);
assert_eq!(v, Ok(Nullable::NonNull(None)));

v = JsValue::null().try_js_into(context);
assert_eq!(v, Ok(Nullable::Null));

v = JsValue::from(42).try_js_into(context);
assert_eq!(v, Ok(Nullable::NonNull(Some(42))));
assert_eq!(v.unwrap().flatten(), Some(42));

Variants§

§

Null

The value was not defined in JavaScript.

§

NonNull(T)

The value was defined in JavaScript.

Implementations§

Source§

impl<T> Nullable<T>

Source

pub const fn is_null(&self) -> bool

Returns true if this value is Nullable::Null.

Source

pub const fn is_not_null(&self) -> bool

Returns true if this value is Nullable::NotNull.

Source

pub fn iter(&self) -> Iter<'_, T>

Returns an iterator over the possibly contained value.

§Examples
let x = Nullable::NonNull(4);
assert_eq!(x.iter().next(), Some(&4));

let x: Nullable<u32> = Nullable::Null;
assert_eq!(x.iter().next(), None);
Source

pub const fn as_ref(&self) -> Nullable<&T>

Converts from &Nullable<T> to Nullable<&T>.

§Examples
let text: Nullable<String> = Nullable::NonNull("Hello, world!".to_string());

// First, cast `Nullable<String>` to `Nullable<&String>` with `as_ref`,
// then consume *that* with `map`, leaving `text` on the stack.
let text_length: Nullable<usize> = text.as_ref().map(|s| s.len());
println!("still can print text: {text:?}");
Source

pub fn map<U, F>(self, f: F) -> Nullable<U>
where F: FnOnce(T) -> U,

Maps a Nullable<T> to Nullable<U> by applying a function to a contained value.

§Examples
let maybe_some_string = Nullable::NonNull(String::from("Hello, World!"));
// `Nullable::map` takes self *by value*, consuming `maybe_some_string`
let maybe_some_len = maybe_some_string.map(|s| s.len());
assert_eq!(maybe_some_len, Nullable::NonNull(13));

let x: Nullable<&str> = Nullable::Null;
assert_eq!(x.map(|s| s.len()), Nullable::Null);
Source

pub fn unwrap_or_default(self) -> T
where T: Default,

Returns the contained Nullable::NonNull value or a default.

Consumes the self argument then, if Nullable::NonNull, returns the contained value, otherwise if Nullable::Null, returns the default value for that type.

§Examples
let x: Nullable<u32> = Nullable::Null;
let y: Nullable<u32> = Nullable::NonNull(12);

assert_eq!(x.unwrap_or_default(), 0);
assert_eq!(y.unwrap_or_default(), 12);
Source§

impl<T> Nullable<Option<T>>

Source

pub fn flatten(self) -> Option<T>

Converts from Nullable<Option<T>> to Option<T>.

Trait Implementations§

Source§

impl<T> Clone for Nullable<T>
where T: Clone,

Source§

fn clone(&self) -> Nullable<T>

Returns a duplicate of the value. Read more
1.0.0 · Source§

fn clone_from(&mut self, source: &Self)

Performs copy-assignment from source. Read more
Source§

impl<T> Debug for Nullable<T>
where T: Debug,

Source§

fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error>

Formats the value using the given formatter. Read more
Source§

impl<T> From<Nullable<T>> for Option<T>

Source§

fn from(value: Nullable<T>) -> Option<T>

Converts to this type from the input type.
Source§

impl<T> From<Option<T>> for Nullable<T>

Source§

fn from(value: Option<T>) -> Nullable<T>

Converts to this type from the input type.
Source§

impl<T> Hash for Nullable<T>
where T: Hash,

Source§

fn hash<__H>(&self, state: &mut __H)
where __H: Hasher,

Feeds this value into the given Hasher. Read more
1.3.0 · Source§

fn hash_slice<H>(data: &[Self], state: &mut H)
where H: Hasher, Self: Sized,

Feeds a slice of this type into the given Hasher. Read more
Source§

impl<'a, T> IntoIterator for &'a Nullable<T>

Source§

type Item = &'a T

The type of the elements being iterated over.
Source§

type IntoIter = Iter<'a, T>

Which kind of iterator are we turning this into?
Source§

fn into_iter(self) -> Iter<'a, T>

Creates an iterator from a value. Read more
Source§

impl<T> IntoIterator for Nullable<T>

Source§

fn into_iter(self) -> IntoIter<T>

Returns a consuming iterator over the possibly contained value.

§Examples
let x = Nullable::NonNull("string");
let v: Vec<&str> = x.into_iter().collect();
assert_eq!(v, ["string"]);

let x = Nullable::Null;
let v: Vec<&str> = x.into_iter().collect();
assert!(v.is_empty());
Source§

type Item = T

The type of the elements being iterated over.
Source§

type IntoIter = IntoIter<T>

Which kind of iterator are we turning this into?
Source§

impl<T> Ord for Nullable<T>
where T: Ord,

Source§

fn cmp(&self, other: &Nullable<T>) -> Ordering

This method returns an Ordering between self and other. Read more
1.21.0 · Source§

fn max(self, other: Self) -> Self
where Self: Sized,

Compares and returns the maximum of two values. Read more
1.21.0 · Source§

fn min(self, other: Self) -> Self
where Self: Sized,

Compares and returns the minimum of two values. Read more
1.50.0 · Source§

fn clamp(self, min: Self, max: Self) -> Self
where Self: Sized,

Restrict a value to a certain interval. Read more
Source§

impl<T> PartialEq for Nullable<T>
where T: PartialEq,

Source§

fn eq(&self, other: &Nullable<T>) -> bool

Tests for self and other values to be equal, and is used by ==.
1.0.0 · Source§

fn ne(&self, other: &Rhs) -> bool

Tests for !=. The default implementation is almost always sufficient, and should not be overridden without very good reason.
Source§

impl<T> PartialOrd for Nullable<T>
where T: PartialOrd,

Source§

fn partial_cmp(&self, other: &Nullable<T>) -> Option<Ordering>

This method returns an ordering between self and other values if one exists. Read more
1.0.0 · Source§

fn lt(&self, other: &Rhs) -> bool

Tests less than (for self and other) and is used by the < operator. Read more
1.0.0 · Source§

fn le(&self, other: &Rhs) -> bool

Tests less than or equal to (for self and other) and is used by the <= operator. Read more
1.0.0 · Source§

fn gt(&self, other: &Rhs) -> bool

Tests greater than (for self and other) and is used by the > operator. Read more
1.0.0 · Source§

fn ge(&self, other: &Rhs) -> bool

Tests greater than or equal to (for self and other) and is used by the >= operator. Read more
Source§

impl<T> TryFromJs for Nullable<T>
where T: TryFromJs,

Source§

fn try_from_js( value: &JsValue, context: &mut Context, ) -> Result<Nullable<T>, JsError>

This function tries to convert a JavaScript value into Self.
Source§

impl<T> TryIntoJs for Nullable<T>
where T: TryIntoJs,

Source§

fn try_into_js(&self, context: &mut Context) -> Result<JsValue, JsError>

This function tries to convert a Self into JsValue.
Source§

impl<T> Copy for Nullable<T>
where T: Copy,

Source§

impl<T> Eq for Nullable<T>
where T: Eq,

Source§

impl<T> StructuralPartialEq for Nullable<T>

Auto Trait Implementations§

§

impl<T> Freeze for Nullable<T>
where T: Freeze,

§

impl<T> RefUnwindSafe for Nullable<T>
where T: RefUnwindSafe,

§

impl<T> Send for Nullable<T>
where T: Send,

§

impl<T> Sync for Nullable<T>
where T: Sync,

§

impl<T> Unpin for Nullable<T>
where T: Unpin,

§

impl<T> UnwindSafe for Nullable<T>
where T: UnwindSafe,

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
Source§

impl<T> CloneToUninit for T
where T: Clone,

Source§

unsafe fn clone_to_uninit(&self, dest: *mut u8)

🔬This is a nightly-only experimental API. (clone_to_uninit)
Performs copy-assignment from self to dest. Read more
Source§

impl<Q, K> Comparable<K> for Q
where Q: Ord + ?Sized, K: Borrow<Q> + ?Sized,

Source§

fn compare(&self, key: &K) -> Ordering

Compare self to key and return their ordering.
Source§

impl<T> Conv for T

Source§

fn conv<T>(self) -> T
where Self: Into<T>,

Converts self into T using Into<T>. Read more
Source§

impl<Q, K> Equivalent<K> for Q
where Q: Eq + ?Sized, K: Borrow<Q> + ?Sized,

Source§

fn equivalent(&self, key: &K) -> bool

Compare self to key and return true if they are equal.
Source§

impl<Q, K> Equivalent<K> for Q
where Q: Eq + ?Sized, K: Borrow<Q> + ?Sized,

Source§

fn equivalent(&self, key: &K) -> bool

Checks if this value is equivalent to the given key. Read more
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

Source§

impl<T> Instrument for T

Source§

fn instrument(self, span: Span) -> Instrumented<Self>

Instruments this type with the provided Span, returning an Instrumented wrapper. Read more
Source§

fn in_current_span(self) -> Instrumented<Self>

Instruments this type with the current Span, returning an Instrumented wrapper. Read more
Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

Source§

impl<T> IntoEither for T

Source§

fn into_either(self, into_left: bool) -> Either<Self, Self>

Converts self into a Left variant of Either<Self, Self> if into_left is true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
Source§

fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
where F: FnOnce(&Self) -> bool,

Converts self into a Left variant of Either<Self, Self> if into_left(&self) returns true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
Source§

impl<T> IntoJsModule for T
where T: IntoIterator<Item = (JsString, NativeFunction)> + Clone,

Source§

fn into_js_module(self, context: &mut Context) -> Module

Converts the type into a JS module.
Source§

impl<T> Pipe for T
where T: ?Sized,

Source§

fn pipe<R>(self, func: impl FnOnce(Self) -> R) -> R
where Self: Sized,

Pipes by value. This is generally the method you want to use. Read more
Source§

fn pipe_ref<'a, R>(&'a self, func: impl FnOnce(&'a Self) -> R) -> R
where R: 'a,

Borrows self and passes that borrow into the pipe function. Read more
Source§

fn pipe_ref_mut<'a, R>(&'a mut self, func: impl FnOnce(&'a mut Self) -> R) -> R
where R: 'a,

Mutably borrows self and passes that borrow into the pipe function. Read more
Source§

fn pipe_borrow<'a, B, R>(&'a self, func: impl FnOnce(&'a B) -> R) -> R
where Self: Borrow<B>, B: 'a + ?Sized, R: 'a,

Borrows self, then passes self.borrow() into the pipe function. Read more
Source§

fn pipe_borrow_mut<'a, B, R>( &'a mut self, func: impl FnOnce(&'a mut B) -> R, ) -> R
where Self: BorrowMut<B>, B: 'a + ?Sized, R: 'a,

Mutably borrows self, then passes self.borrow_mut() into the pipe function. Read more
Source§

fn pipe_as_ref<'a, U, R>(&'a self, func: impl FnOnce(&'a U) -> R) -> R
where Self: AsRef<U>, U: 'a + ?Sized, R: 'a,

Borrows self, then passes self.as_ref() into the pipe function.
Source§

fn pipe_as_mut<'a, U, R>(&'a mut self, func: impl FnOnce(&'a mut U) -> R) -> R
where Self: AsMut<U>, U: 'a + ?Sized, R: 'a,

Mutably borrows self, then passes self.as_mut() into the pipe function.
Source§

fn pipe_deref<'a, T, R>(&'a self, func: impl FnOnce(&'a T) -> R) -> R
where Self: Deref<Target = T>, T: 'a + ?Sized, R: 'a,

Borrows self, then passes self.deref() into the pipe function.
Source§

fn pipe_deref_mut<'a, T, R>( &'a mut self, func: impl FnOnce(&'a mut T) -> R, ) -> R
where Self: DerefMut<Target = T> + Deref, T: 'a + ?Sized, R: 'a,

Mutably borrows self, then passes self.deref_mut() into the pipe function.
Source§

impl<T> PolicyExt for T
where T: ?Sized,

Source§

fn and<P, B, E>(self, other: P) -> And<T, P>
where T: Policy<B, E>, P: Policy<B, E>,

Create a new Policy that returns Action::Follow only if self and other return Action::Follow. Read more
Source§

fn or<P, B, E>(self, other: P) -> Or<T, P>
where T: Policy<B, E>, P: Policy<B, E>,

Create a new Policy that returns Action::Follow if either self or other returns Action::Follow. Read more
Source§

impl<T> Tap for T

Source§

fn tap(self, func: impl FnOnce(&Self)) -> Self

Immutable access to a value. Read more
Source§

fn tap_mut(self, func: impl FnOnce(&mut Self)) -> Self

Mutable access to a value. Read more
Source§

fn tap_borrow<B>(self, func: impl FnOnce(&B)) -> Self
where Self: Borrow<B>, B: ?Sized,

Immutable access to the Borrow<B> of a value. Read more
Source§

fn tap_borrow_mut<B>(self, func: impl FnOnce(&mut B)) -> Self
where Self: BorrowMut<B>, B: ?Sized,

Mutable access to the BorrowMut<B> of a value. Read more
Source§

fn tap_ref<R>(self, func: impl FnOnce(&R)) -> Self
where Self: AsRef<R>, R: ?Sized,

Immutable access to the AsRef<R> view of a value. Read more
Source§

fn tap_ref_mut<R>(self, func: impl FnOnce(&mut R)) -> Self
where Self: AsMut<R>, R: ?Sized,

Mutable access to the AsMut<R> view of a value. Read more
Source§

fn tap_deref<T>(self, func: impl FnOnce(&T)) -> Self
where Self: Deref<Target = T>, T: ?Sized,

Immutable access to the Deref::Target of a value. Read more
Source§

fn tap_deref_mut<T>(self, func: impl FnOnce(&mut T)) -> Self
where Self: DerefMut<Target = T> + Deref, T: ?Sized,

Mutable access to the Deref::Target of a value. Read more
Source§

fn tap_dbg(self, func: impl FnOnce(&Self)) -> Self

Calls .tap() only in debug builds, and is erased in release builds.
Source§

fn tap_mut_dbg(self, func: impl FnOnce(&mut Self)) -> Self

Calls .tap_mut() only in debug builds, and is erased in release builds.
Source§

fn tap_borrow_dbg<B>(self, func: impl FnOnce(&B)) -> Self
where Self: Borrow<B>, B: ?Sized,

Calls .tap_borrow() only in debug builds, and is erased in release builds.
Source§

fn tap_borrow_mut_dbg<B>(self, func: impl FnOnce(&mut B)) -> Self
where Self: BorrowMut<B>, B: ?Sized,

Calls .tap_borrow_mut() only in debug builds, and is erased in release builds.
Source§

fn tap_ref_dbg<R>(self, func: impl FnOnce(&R)) -> Self
where Self: AsRef<R>, R: ?Sized,

Calls .tap_ref() only in debug builds, and is erased in release builds.
Source§

fn tap_ref_mut_dbg<R>(self, func: impl FnOnce(&mut R)) -> Self
where Self: AsMut<R>, R: ?Sized,

Calls .tap_ref_mut() only in debug builds, and is erased in release builds.
Source§

fn tap_deref_dbg<T>(self, func: impl FnOnce(&T)) -> Self
where Self: Deref<Target = T>, T: ?Sized,

Calls .tap_deref() only in debug builds, and is erased in release builds.
Source§

fn tap_deref_mut_dbg<T>(self, func: impl FnOnce(&mut T)) -> Self
where Self: DerefMut<Target = T> + Deref, T: ?Sized,

Calls .tap_deref_mut() only in debug builds, and is erased in release builds.
Source§

impl<T> ToOwned for T
where T: Clone,

Source§

type Owned = T

The resulting type after obtaining ownership.
Source§

fn to_owned(&self) -> T

Creates owned data from borrowed data, usually by cloning. Read more
Source§

fn clone_into(&self, target: &mut T)

Uses borrowed data to replace owned data, usually by cloning. Read more
Source§

impl<T> TryConv for T

Source§

fn try_conv<T>(self) -> Result<T, Self::Error>
where Self: TryInto<T>,

Attempts to convert self into T using TryInto<T>. Read more
Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<'a, T> TryFromJsArgument<'a> for T
where T: TryFromJs,

Source§

fn try_from_js_argument( _: &'a JsValue, rest: &'a [JsValue], context: &mut Context, ) -> Result<(T, &'a [JsValue]), JsError>

Try to convert a JS argument into a Rust value, returning the value and the rest of the arguments to be parsed. Read more
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
Source§

impl<T> TryIntoJsResult for T
where T: TryIntoJs,

Source§

fn try_into_js_result(self, ctx: &mut Context) -> Result<JsValue, JsError>

Try to convert a Rust value into a JsResult<JsValue>. Read more
Source§

impl<V, T> VZip<V> for T
where V: MultiLane<T>,

Source§

fn vzip(self) -> V

Source§

impl<T> WithSubscriber for T

Source§

fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
where S: Into<Dispatch>,

Attaches the provided Subscriber to this type, returning a WithDispatch wrapper. Read more
Source§

fn with_current_subscriber(self) -> WithDispatch<Self>

Attaches the current default Subscriber to this type, returning a WithDispatch wrapper. Read more
Source§

impl<T> ErasedDestructor for T
where T: 'static,