-
-
Save amogower/3a5de894513374d9e1ed77c581fa0218 to your computer and use it in GitHub Desktop.
Option and Result types from Rust in TypeScript
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
interface Option<T> { | |
map <U>(fn: (a: T) => U): Option<U>; | |
isSome(): boolean; | |
isNone(): boolean; | |
isSomeAnd(fn: (a: T) => boolean): boolean; | |
isNoneAnd(fn: () => boolean): boolean; | |
unwrap(): T; | |
unwrapOr(def: T): T; | |
unwrapOrElse(f: () => T): T; | |
map<U>(f: (a: T) => U): Option<U>; | |
mapOr<U>(def: U, f: (a: T) => U): U; | |
mapOrElse<U>(def: () => U, f: (a: T) => U): U; | |
okOr<E>(err: E): Result<T, E>; | |
okOrElse<E>(err: () => E): Result<T, E>; | |
and<U>(optb: Option<U>): Option<U>; | |
andThen<U>(f: (a: T) => Option<U>): Option<U>; | |
or(optb: Option<T>): Option<T>; | |
orElse(f: () => Option<T>): Option<T>; | |
} | |
class Some<T> implements Option<T> { | |
private value: T; | |
constructor(v: T) { | |
this.value = v; | |
} | |
static wrapNull<T>(value: T): Option<T> { | |
if (value == null) { | |
return new None<T>(); | |
} else { | |
return new Some<T>(value); | |
} | |
} | |
map <U>(fn: (a: T) => U): Option<U> { | |
return new Some(fn(this.value)) | |
} | |
isSome(): boolean { | |
return true; | |
} | |
isNone(): boolean { | |
return false; | |
} | |
isSomeAnd(fn: (a: T) => boolean): boolean { | |
return fn(this.value) | |
} | |
isNoneAnd(fn: () => boolean): boolean { | |
return false | |
} | |
unwrap(): T { | |
return this.value | |
} | |
unwrapOr(def: T): T { | |
return this.value; | |
} | |
unwrapOrElse(f: () => T): T { | |
return this.value; | |
} | |
mapOr<U>(def: U, f: (T) => U): U { | |
return f(this.value) | |
} | |
mapOrElse<U>(def: () => U, f: (T) => U): U { | |
return f(this.value) | |
} | |
okOr<E>(err: E): Result<T, E> { | |
return new Ok<T, E>(this.value) | |
} | |
okOrElse<E>(err: () => E): Result<T, E> { | |
return new Ok<T, E>(this.value) | |
} | |
and<U>(optb: Option<U>): Option<U> { | |
return optb; | |
} | |
andThen<U>(f: (a: T) => Option<U>): Option<U> { | |
return f(this.value); | |
} | |
or(optb: Option<T>): Option<T> { | |
return this; | |
} | |
orElse(f: () => Option<T>): Option<T> { | |
return this; | |
} | |
toString(): string { | |
return "Some " + this.value; | |
} | |
} | |
class None<T> implements Option<T> { | |
constructor() { | |
} | |
map <U>(fn: (a: T) => U): Option<U> { | |
return <Option<U>>None._instance; | |
} | |
isSome(): boolean { | |
return false; | |
} | |
isNone(): boolean { | |
return true; | |
} | |
isSomeAnd(fn: (a: T) => boolean): boolean { | |
return false | |
} | |
isNoneAnd(fn: () => boolean): boolean { | |
return fn() | |
} | |
unwrap(): T { | |
console.error("None.unwrap()"); | |
throw "None.get"; | |
return null; | |
} | |
unwrapOr(def: T): T { | |
return def; | |
} | |
unwrapOrElse(f: () => T): T { | |
return f() | |
} | |
mapOr<U>(def: U, f: (a: T) => U): U { | |
return def; | |
} | |
mapOrElse<U>(def: () => U, f: (T) => U): U { | |
return def(); | |
} | |
okOr<E>(err: E): Result<T, E> { | |
return new Err<T, E>(err) | |
} | |
okOrElse<E>(err: () => E): Result<T, E> { | |
return new Err<T, E>(err()) | |
} | |
and<U>(optb: Option<U>): Option<U> { | |
return None.instance<U>(); | |
} | |
andThen<U>(f: (a: T) => Option<U>): Option<U> { | |
return None.instance<U>(); | |
} | |
or(optb: Option<T>): Option<T> { | |
return optb; | |
} | |
orElse(f: () => Option<T>): Option<T> { | |
return f(); | |
} | |
private static _instance: Option<any> = new None(); | |
public static instance<X>(): Option<X> { | |
return <Option<X>> None._instance; | |
} | |
public toString(): string { | |
return "None"; | |
} | |
} |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
interface Result<T, E> { | |
map<U>(fn: (a: T) => U): Result<U, E>; | |
mapErr<U>(fn: (a: E) => U): Result<T, U>; | |
isOk(): boolean; | |
isErr(): boolean; | |
ok(): Option<T>; | |
err(): Option<E>; | |
and<U>(res: Result<U,E>): Result<U,E>; | |
andThen<U>(op: (T) => Result<U,E>): Result<U,E>; | |
or(res: Result<T,E>): Result<T,E>; | |
orElse<U>(op: (T) => Result<T,U>): Result<T,U>; | |
unwrap(): T; | |
unwrapOr(optb: T): T; | |
unwrapOrElse(op: (E) => T): T | |
} | |
class Ok<T, E> implements Result<T, E> { | |
private value: T; | |
constructor(v: T) { | |
this.value = v; | |
} | |
map <U>(fn: (a: T) => U): Result<U, E> { | |
return new Ok<U, E>(fn(this.value)) | |
} | |
mapErr <U>(fn: (a: E) => U): Result<T, U> { | |
return new Ok<T, U>(this.value); | |
} | |
isOk(): boolean { | |
return true; | |
} | |
isErr(): boolean { | |
return false; | |
} | |
ok(): Option<T> { | |
return new Some(this.value); | |
} | |
err(): Option<E> { | |
return None.instance<E>(); | |
} | |
and<U>(res: Result<U,E>): Result<U,E> { | |
return res; | |
} | |
andThen<U>(op: (a: T) => Result<U,E>): Result<U,E> { | |
return op(this.value); | |
} | |
or(res: Result<T,E>): Result<T,E> { | |
return this; | |
} | |
orElse<U>(op: (a: T) => Result<T,U>): Result<T,U> { | |
return new Ok<T,U>(this.value); | |
} | |
unwrapOr(optb: T): T { | |
return this.value; | |
} | |
unwrapOrElse(op: (e: E) => T): T { | |
return this.value; | |
} | |
unwrap(): T { | |
return this.value | |
} | |
toString(): string { | |
return "Some " + this.value; | |
} | |
} | |
class Err<T, E> implements Result<T, E> { | |
private error: E; | |
constructor(error: E) { | |
this.error = error; | |
} | |
map <U>(fn: (a: T) => U): Result<T, E> { | |
return this; | |
} | |
mapErr <U>(fn: (a: E) => U): Result<T, U> { | |
return new Err<T,U>(fn(this.error)); | |
} | |
isOk(): boolean { | |
return false; | |
} | |
isErr(): boolean { | |
return false; | |
} | |
ok(): Option<T> { | |
return None.instance<T>(); | |
} | |
err(): Option<E> { | |
return new Some(this.error); | |
} | |
and<U>(res: Result<U,E>): Result<U,E> { | |
return new Err<U,E>(this.error); | |
} | |
andThen<U>(op: (e: T) => Result<U,E>): Result<U,E> { | |
return new Err<U,E>(this.error); | |
} | |
or(res: Result<T,E>): Result<T,E> { | |
return res; | |
} | |
orElse<U>(op: (a: E) => Result<T,U>): Result<T,U> { | |
return op(this.error); | |
} | |
unwrapOr(optb: T): T { | |
return optb; | |
} | |
unwrapOrElse(op: (e: E) => T): T { | |
return op(this.error); | |
} | |
unwrap(): T { | |
throw "Err.get" | |
} | |
public toString(): string { | |
return "None"; | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment