Created
November 30, 2014 12:15
-
-
Save s-panferov/575da5a7131c285c0539 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