diff --git a/404.html b/404.html index d4b5206..f9de052 100644 --- a/404.html +++ b/404.html @@ -5,13 +5,13 @@ Page Not Found | Boxed - +
Skip to main content

Page Not Found

We could not find what you were looking for.

Please contact the owner of the site that linked you to the original URL and let them know their link is broken.

- + \ No newline at end of file diff --git a/array/index.html b/array/index.html index 984d9d5..bfc33e6 100644 --- a/array/index.html +++ b/array/index.html @@ -5,13 +5,13 @@ Array | Boxed - +
Skip to main content

Array

import { Array } from "@swan-io/boxed";

Array.filterMap(array, func)

Returns an array containing the Option.Some values returned by func for each array item.

This function can be useful to refine the types of an array.

Examples
Array.filterMap([1, 2, 3], (x) =>
isEven(x) === 0 ? Option.Some(x) : Option.None(),
); // [2]

Array.findMap(array, func)

Returns the first Option.Some value returned by func for each array item.

Examples
Array.findMap([1, 2, 3], (x) =>
isEven(x) === 0 ? Option.Some(x) : Option.None(),
); // Option.Some(2)

Array.find(array, predicate)

Return the first item in the array for which predicate returns true.

The function returns an Option so that we can distinguish between a found nullish value and a not found value.

Examples
Array.find(array, (x) => x === undefined);
// Some(undefined) if found
// None if not found

Array.findIndex(array, predicate)

Return the first index in the array for which predicate returns true.

The function returns an Option.

Examples
Array.findIndex(array, (x) => x === undefined);
// Some(index) if found
// None if not found

Array.binarySearchBy(sortedArray, item, compare)

Performs a binary search on the array.

Returns the index of the item if there's an exact match, return the index of the first superior value if not. Return -1 if the array is empty.

Examples
const index = Array.binarySearchBy(array, "my value");

Array.zip(arrayA, arrayB)

Create an array of pairs from two arrays.

Examples
Array.zip([1, 2, 3], ["one", "two", "three"]);
// [[1, "one"], [2, "two"], [3, "three"]]

Array.unzip(arrayOfPairs)

Turns an array of pairs into two arrays.

Examples
Array.unzip([
[1, "one"],
[2, "two"],
[3, "three"],
]);
// [[1, 2, 3], ["one", "two", "three"]]

Array.from(arrayLike)

Array.from, reexported for convenience when Boxed Array shadows the Array constructor in scope.

Examples
Array.from({ length: 3 }, (_, key) => key); // [0, 1, 2]

Array.of(...items)

Array.of, reexported for convenience when Boxed Array shadows the Array constructor in scope.

Examples
Array.of(1, 2, 3); // [1, 2, 3]

Array.isArray(value)

Array.isArray, reexported for convenience when Boxed Array shadows the Array constructor in scope.

Examples
Array.isArray("");
// false

Array.isArray([1, 2, 3]);
// true
- + \ No newline at end of file diff --git a/assets/js/28b9547a.8b9149cd.js b/assets/js/28b9547a.8b9149cd.js new file mode 100644 index 0000000..1846420 --- /dev/null +++ b/assets/js/28b9547a.8b9149cd.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdocs=self.webpackChunkdocs||[]).push([[498],{3905:(e,t,r)=>{r.d(t,{Zo:()=>k,kt:()=>N});var a=r(7294);function n(e,t,r){return t in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function l(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),r.push.apply(r,a)}return r}function u(e){for(var t=1;t=0||(n[r]=e[r]);return n}(e,t);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,r)&&(n[r]=e[r])}return n}var i=a.createContext({}),o=function(e){var t=a.useContext(i),r=t;return e&&(r="function"==typeof e?e(t):u(u({},t),e)),r},k=function(e){var t=o(e.components);return a.createElement(i.Provider,{value:t},e.children)},m="mdxType",s={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},d=a.forwardRef((function(e,t){var r=e.components,n=e.mdxType,l=e.originalType,i=e.parentName,k=p(e,["components","mdxType","originalType","parentName"]),m=o(r),d=n,N=m["".concat(i,".").concat(d)]||m[d]||s[d]||l;return r?a.createElement(N,u(u({ref:t},k),{},{components:r})):a.createElement(N,u({ref:t},k))}));function N(e,t){var r=arguments,n=t&&t.mdxType;if("string"==typeof e||n){var l=r.length,u=new Array(l);u[0]=d;var p={};for(var i in t)hasOwnProperty.call(t,i)&&(p[i]=t[i]);p.originalType=e,p[m]="string"==typeof e?e:n,u[1]=p;for(var o=2;o{r.r(t),r.d(t,{assets:()=>i,contentTitle:()=>u,default:()=>s,frontMatter:()=>l,metadata:()=>p,toc:()=>o});var a=r(7462),n=(r(7294),r(3905));const l={title:"Future>",sidebar_label:"Future helpers"},u=void 0,p={unversionedId:"future-result",id:"future-result",title:"Future>",description:"A Future can contain a Result (e.g. to represent an asynchronous value that can fail). We provide some utility functions to deal with that case without having to unwrap the Future result.",source:"@site/docs/future-result.md",sourceDirName:".",slug:"/future-result",permalink:"/boxed/future-result",draft:!1,editUrl:"https://github.com/swan-io/boxed/edit/main/docs/docs/future-result.md",tags:[],version:"current",frontMatter:{title:"Future>",sidebar_label:"Future helpers"},sidebar:"docs",previous:{title:"Future",permalink:"/boxed/future"},next:{title:"Deferred",permalink:"/boxed/deferred"}},i={},o=[{value:"Methods",id:"methods",level:2},{value:".mapOkToResult(f)",id:"mapoktoresultf",level:3},{value:".mapErrorToResult(f)",id:"maperrortoresultf",level:3},{value:".mapOk(f)",id:"mapokf",level:3},{value:".mapError(f)",id:"maperrorf",level:3},{value:".flatMapOk(f)",id:"flatmapokf",level:3},{value:".flatMapError(f)",id:"flatmaperrorf",level:3},{value:".tapOk(f)",id:"tapokf",level:3},{value:".tapError(f)",id:"taperrorf",level:3},{value:".resultToPromise()",id:"resulttopromise",level:3},{value:"Statics",id:"statics",level:2},{value:"Future.all(resultFutures)",id:"futureallresultfutures",level:3},{value:"Future.allFromDict(resultFutures)",id:"futureallfromdictresultfutures",level:3},{value:"Future.retry(getFuture)",id:"futureretrygetfuture",level:3},{value:"Cheatsheet",id:"cheatsheet",level:2}],k={toc:o},m="wrapper";function s(e){let{components:t,...r}=e;return(0,n.kt)(m,(0,a.Z)({},k,r,{components:t,mdxType:"MDXLayout"}),(0,n.kt)("p",null,"A ",(0,n.kt)("a",{parentName:"p",href:"./future"},"Future")," can contain a ",(0,n.kt)("inlineCode",{parentName:"p"},"Result")," (e.g. to represent an asynchronous value that can fail). We provide some utility functions to deal with that case without having to unwrap the Future result."),(0,n.kt)("admonition",{type:"note"},(0,n.kt)("p",{parentName:"admonition"},"You can still use all the regular ",(0,n.kt)("a",{parentName:"p",href:"./future"},"Future")," methods. The following helpers simply removes the need to unwrap the contained result.")),(0,n.kt)("h2",{id:"methods"},"Methods"),(0,n.kt)("h3",{id:"mapoktoresultf"},".mapOkToResult(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.mapOkToResult(\n func: (value: A) => Result,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Ok")," and returning ",(0,n.kt)("inlineCode",{parentName:"p"},"Result")," and returns a new ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Ok(3)).mapOkToResult((ok) => {\n return Result.Ok(ok * 2);\n});\n// Future>\n\nFuture.value(Result.Ok(3)).mapOkToResult((ok) =>\n isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");\n);\n// Future>\n')),(0,n.kt)("h3",{id:"maperrortoresultf"},".mapErrorToResult(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.mapErrorToResult(\n func: (value: E) => Result,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," and returning ",(0,n.kt)("inlineCode",{parentName:"p"},"Result")," and returns a new ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Error(3)).mapErrorToResult((ok) => {\n return Result.Ok(ok * 2);\n});\n// Future>\n\nFuture.value(Result.Error(3)).mapErrorToResult((ok) =>\n isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");\n);\n// Future>\n')),(0,n.kt)("h3",{id:"mapokf"},".mapOk(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.mapOk(\n func: (value: A) => B,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Ok")," and returning ",(0,n.kt)("inlineCode",{parentName:"p"},"ReturnValue")," and returns a new ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Ok(3)).mapOk((ok) => {\n return ok * 2;\n});\n// Future>\n\nFuture.value(Result.Error("something")).mapOk((ok) => {\n return ok * 2;\n});\n// Future>\n')),(0,n.kt)("h3",{id:"maperrorf"},".mapError(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.mapError(\n func: (value: E) => F,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," and returning ",(0,n.kt)("inlineCode",{parentName:"p"},"ReturnValue")," and returns a new ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Error(3)).mapError((error) => {\n return error * 2;\n});\n// Future>\n\nFuture.value(Result.Ok("something")).mapError((ok) => {\n return ok * 2;\n});\n// Future>\n')),(0,n.kt)("h3",{id:"flatmapokf"},".flatMapOk(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.flatMapOk(\n func: (value: A) => Future>,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Ok")," returning a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Ok(3)).flatMapOk((ok) => Future.value(Result.Ok(ok * 2)));\n// Future>\n\nFuture.value(Result.Ok(3)).flatMapOk((ok) =>\n Future.value(Result.Error("Nope")),\n);\n// Future>\n\nFuture.value(Result.Error("Error")).flatMapOk((ok) =>\n Future.value(Result.Ok(ok * 2)),\n);\n// Future>\n')),(0,n.kt)("h3",{id:"flatmaperrorf"},".flatMapError(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.flatMapError(\n func: (value: E) => Future>,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," returning a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Ok(3)).flatMapError((error) =>\n Future.value(Result.Ok(ok * 2)),\n);\n// Future>\n\nFuture.value(Result.Error("Error")).flatMapError((error) =>\n Future.value(Result.Error("Nope")),\n);\n// Future>\n\nFuture.value(Result.Error("Error")).flatMapError((error) =>\n Future.value(Result.Ok(1)),\n);\n// Future>\n')),(0,n.kt)("h3",{id:"tapokf"},".tapOk(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.tapOk(func: (value: A) => unknown): Future>\n")),(0,n.kt)("p",null,"Runs ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," if value is ",(0,n.kt)("inlineCode",{parentName:"p"},"Ok")," with the future value, and returns the original future. Useful for debugging."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"future.tapOk(console.log);\n")),(0,n.kt)("h3",{id:"taperrorf"},".tapError(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.tapError(func: (value: E) => unknown): Future>\n")),(0,n.kt)("p",null,"Runs ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," if value is ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," with the future value, and returns the original future. Useful for debugging."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"future.tapError(console.log);\n")),(0,n.kt)("h3",{id:"resulttopromise"},".resultToPromise()"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.resultToPromise(): Promise\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and returns a ",(0,n.kt)("inlineCode",{parentName:"p"},"Promise"),", rejecting the promise with ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," in this state."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"Future.value(Result.Ok(1)).resultToPromise();\n// Promise<1>\n\nFuture.value(Result.Reject(1)).resultToPromise();\n// Promise (rejected with 1)\n")),(0,n.kt)("h2",{id:"statics"},"Statics"),(0,n.kt)("h3",{id:"futureallresultfutures"},"Future.all(resultFutures)"),(0,n.kt)("p",null,"You can combine the ",(0,n.kt)("inlineCode",{parentName:"p"},"all")," helpers from ",(0,n.kt)("inlineCode",{parentName:"p"},"Future")," and ",(0,n.kt)("inlineCode",{parentName:"p"},"Result"),":"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"const futures = [\n Future.value(Result.Ok(1)),\n Future.value(Result.Ok(2)),\n Future.value(Result.Ok(3)),\n];\n\nFuture.all(futures).map(Result.all);\n// Future>\n")),(0,n.kt)("p",null,"Let's see the types at each step:"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"// Array>>\n// -> [Future>, Future>, Future>]\nconst input = [\n Future.value(Result.Ok(1)),\n Future.value(Result.Ok(2)),\n Future.value(Result.Ok(3)),\n];\n\n// Future>>\n// -> Future<[Result.Ok<1>>, Result.Ok<2>>, Result.Ok<3>]>\nconst step1 = Future.all(input);\n\n// Future, never>>\n// -> Future<[Result.Ok<[1, 2, 3]>>\nconst step2 = step1.map(Result.all);\n")),(0,n.kt)("h3",{id:"futureallfromdictresultfutures"},"Future.allFromDict(resultFutures)"),(0,n.kt)("p",null,"Like as ",(0,n.kt)("inlineCode",{parentName:"p"},"all"),", you can combine the ",(0,n.kt)("inlineCode",{parentName:"p"},"allFromDict"),":"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"const futures = {\n a: Future.value(Result.Ok(1)),\n b: Future.value(Result.Ok(2)),\n c: Future.value(Result.Ok(3)),\n};\n\nFuture.allFromDict(futures).map(Result.allFromDict);\n// Future<[Result.Ok<{a: 1, b: 2, c: 3}>>\n")),(0,n.kt)("p",null,"Let's see the types at each step:"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"// Dict>>\n// -> {a: Future>, b: Future>, c: Future>\u2014\nconst input = {\n a: Future.value(Result.Ok(1)),\n b: Future.value(Result.Ok(2)),\n c: Future.value(Result.Ok(3)),\n};\n\n// Future>>\n// -> Future<{a: Result.Ok<1>>, b: Result.Ok<2>>, c: Result.Ok<3>}>\nconst step1 = Future.all(input);\n\n// Future, never>>\n// -> Future<[Result.Ok<{a: 1, b: 2, c: 3}>>\nconst step2 = step1.map(Result.all);\n")),(0,n.kt)("h3",{id:"futureretrygetfuture"},"Future.retry(getFuture)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"retry(getFuture: () => Future>, {max: number}): Future>\n")),(0,n.kt)("p",null,"Runs the future getter, if the future resolves with a ",(0,n.kt)("inlineCode",{parentName:"p"},"Result.Error"),", retries until hitting ",(0,n.kt)("inlineCode",{parentName:"p"},"max")," attempts."),(0,n.kt)("p",null,"The function provides a 0-based ",(0,n.kt)("inlineCode",{parentName:"p"},"attempt")," count to the function if you need to implement delay logic."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"// retry immediately after failure\nFuture.retry(\n (attempt) => {\n return getUserById(userId);\n },\n { max: 3 },\n);\n// Future>\n\n// adding delay\nFuture.retry(\n (attempt) => {\n return Future.wait(attempt * 100).flatMap(() => getUserById(userId));\n },\n { max: 10 },\n);\n// Future>\n")),(0,n.kt)("h2",{id:"cheatsheet"},"Cheatsheet"),(0,n.kt)("table",null,(0,n.kt)("thead",{parentName:"table"},(0,n.kt)("tr",{parentName:"thead"},(0,n.kt)("th",{parentName:"tr",align:null},"Method"),(0,n.kt)("th",{parentName:"tr",align:null},"Input"),(0,n.kt)("th",{parentName:"tr",align:null},"Function input"),(0,n.kt)("th",{parentName:"tr",align:null},"Function output"),(0,n.kt)("th",{parentName:"tr",align:null},"Returned value"))),(0,n.kt)("tbody",{parentName:"table"},(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapoktoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOkToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Ok(y)")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapoktoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOkToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Error(f)")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapoktoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOkToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrortoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapErrorToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Ok(y)")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrortoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapErrorToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Error(f)")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrortoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapErrorToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"y")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"f")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmaperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmaperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmaperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))))))}s.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/28b9547a.e6eb6c9c.js b/assets/js/28b9547a.e6eb6c9c.js deleted file mode 100644 index 20baca9..0000000 --- a/assets/js/28b9547a.e6eb6c9c.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdocs=self.webpackChunkdocs||[]).push([[498],{3905:(e,t,r)=>{r.d(t,{Zo:()=>k,kt:()=>N});var a=r(7294);function n(e,t,r){return t in e?Object.defineProperty(e,t,{value:r,enumerable:!0,configurable:!0,writable:!0}):e[t]=r,e}function l(e,t){var r=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),r.push.apply(r,a)}return r}function u(e){for(var t=1;t=0||(n[r]=e[r]);return n}(e,t);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,r)&&(n[r]=e[r])}return n}var i=a.createContext({}),o=function(e){var t=a.useContext(i),r=t;return e&&(r="function"==typeof e?e(t):u(u({},t),e)),r},k=function(e){var t=o(e.components);return a.createElement(i.Provider,{value:t},e.children)},m="mdxType",s={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},d=a.forwardRef((function(e,t){var r=e.components,n=e.mdxType,l=e.originalType,i=e.parentName,k=p(e,["components","mdxType","originalType","parentName"]),m=o(r),d=n,N=m["".concat(i,".").concat(d)]||m[d]||s[d]||l;return r?a.createElement(N,u(u({ref:t},k),{},{components:r})):a.createElement(N,u({ref:t},k))}));function N(e,t){var r=arguments,n=t&&t.mdxType;if("string"==typeof e||n){var l=r.length,u=new Array(l);u[0]=d;var p={};for(var i in t)hasOwnProperty.call(t,i)&&(p[i]=t[i]);p.originalType=e,p[m]="string"==typeof e?e:n,u[1]=p;for(var o=2;o{r.r(t),r.d(t,{assets:()=>i,contentTitle:()=>u,default:()=>s,frontMatter:()=>l,metadata:()=>p,toc:()=>o});var a=r(7462),n=(r(7294),r(3905));const l={title:"Future>",sidebar_label:"Future helpers"},u=void 0,p={unversionedId:"future-result",id:"future-result",title:"Future>",description:"A Future can contain a Result (e.g. to represent an asynchronous value that can fail). We provide some utility functions to deal with that case without having to unwrap the Future result.",source:"@site/docs/future-result.md",sourceDirName:".",slug:"/future-result",permalink:"/boxed/future-result",draft:!1,editUrl:"https://github.com/swan-io/boxed/edit/main/docs/docs/future-result.md",tags:[],version:"current",frontMatter:{title:"Future>",sidebar_label:"Future helpers"},sidebar:"docs",previous:{title:"Future",permalink:"/boxed/future"},next:{title:"Deferred",permalink:"/boxed/deferred"}},i={},o=[{value:"Methods",id:"methods",level:2},{value:".mapOkToResult(f)",id:"mapoktoresultf",level:3},{value:".mapErrorToResult(f)",id:"maperrortoresultf",level:3},{value:".mapOk(f)",id:"mapokf",level:3},{value:".mapError(f)",id:"maperrorf",level:3},{value:".flatMapOk(f)",id:"flatmapokf",level:3},{value:".flatMapError(f)",id:"flatmaperrorf",level:3},{value:".tapOk(f)",id:"tapokf",level:3},{value:".tapError(f)",id:"taperrorf",level:3},{value:".resultToPromise()",id:"resulttopromise",level:3},{value:"Statics",id:"statics",level:2},{value:"Future.all(resultFutures)",id:"futureallresultfutures",level:3},{value:"Future.allFromDict(resultFutures)",id:"futureallfromdictresultfutures",level:3},{value:"Future.retry(getFuture)",id:"futureretrygetfuture",level:3},{value:"Cheatsheet",id:"cheatsheet",level:2}],k={toc:o},m="wrapper";function s(e){let{components:t,...r}=e;return(0,n.kt)(m,(0,a.Z)({},k,r,{components:t,mdxType:"MDXLayout"}),(0,n.kt)("p",null,"A ",(0,n.kt)("a",{parentName:"p",href:"./future"},"Future")," can contain a ",(0,n.kt)("inlineCode",{parentName:"p"},"Result")," (e.g. to represent an asynchronous value that can fail). We provide some utility functions to deal with that case without having to unwrap the Future result."),(0,n.kt)("admonition",{type:"note"},(0,n.kt)("p",{parentName:"admonition"},"You can still use all the regular ",(0,n.kt)("a",{parentName:"p",href:"./future"},"Future")," methods. The following helpers simply removes the need to unwrap the contained result.")),(0,n.kt)("h2",{id:"methods"},"Methods"),(0,n.kt)("h3",{id:"mapoktoresultf"},".mapOkToResult(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.mapOkToResult(\n func: (value: A) => Result,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Ok")," and returning ",(0,n.kt)("inlineCode",{parentName:"p"},"Result")," and returns a new ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Ok(3)).mapOkToResult((ok) => {\n return Result.Ok(ok * 2);\n});\n// Future>\n\nFuture.value(Result.Ok(3)).mapOkToResult((ok) =>\n isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");\n);\n// Future>\n')),(0,n.kt)("h3",{id:"maperrortoresultf"},".mapErrorToResult(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.mapErrorToResult(\n func: (value: E) => Result,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," and returning ",(0,n.kt)("inlineCode",{parentName:"p"},"Result")," and returns a new ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Error(3)).mapErrorToResult((ok) => {\n return Result.Ok(ok * 2);\n});\n// Future>\n\nFuture.value(Result.Error(3)).mapErrorToResult((ok) =>\n isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");\n);\n// Future>\n')),(0,n.kt)("h3",{id:"mapokf"},".mapOk(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.mapOk(\n func: (value: A) => B,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Ok")," and returning ",(0,n.kt)("inlineCode",{parentName:"p"},"ReturnValue")," and returns a new ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Ok(3)).mapOk((ok) => {\n return ok * 2;\n});\n// Future>\n\nFuture.value(Result.Error("something")).mapOk((ok) => {\n return ok * 2;\n});\n// Future>\n')),(0,n.kt)("h3",{id:"maperrorf"},".mapError(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.mapError(\n func: (value: E) => F,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," and returning ",(0,n.kt)("inlineCode",{parentName:"p"},"ReturnValue")," and returns a new ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Error(3)).mapError((error) => {\n return error * 2;\n});\n// Future>\n\nFuture.value(Result.Ok("something")).mapError((ok) => {\n return ok * 2;\n});\n// Future>\n')),(0,n.kt)("h3",{id:"flatmapokf"},".flatMapOk(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.flatMapOk(\n func: (value: A) => Future>,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Ok")," returning a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Ok(3)).flatMapOk((ok) => Future.value(Result.Ok(ok * 2)));\n// Future>\n\nFuture.value(Result.Ok(3)).flatMapOk((ok) =>\n Future.value(Result.Error("Nope")),\n);\n// Future>\n\nFuture.value(Result.Error("Error")).flatMapOk((ok) =>\n Future.value(Result.Ok(ok * 2)),\n);\n// Future>\n')),(0,n.kt)("h3",{id:"flatmaperrorf"},".flatMapError(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.flatMapError(\n func: (value: E) => Future>,\n propagateCancel?: boolean\n): Future>\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and a ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," function taking ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," returning a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},'Future.value(Result.Ok(3)).flatMapError((error) =>\n Future.value(Result.Ok(ok * 2)),\n);\n// Future>\n\nFuture.value(Result.Error("Error")).flatMapError((error) =>\n Future.value(Result.Error("Nope")),\n);\n// Future>\n\nFuture.value(Result.Error("Error")).flatMapError((error) =>\n Future.value(Result.Ok(1)),\n);\n// Future>\n')),(0,n.kt)("h3",{id:"tapokf"},".tapOk(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.tapOk(func: (value: A) => unknown): Future>\n")),(0,n.kt)("p",null,"Runs ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," if value is ",(0,n.kt)("inlineCode",{parentName:"p"},"Ok")," with the future value, and returns the original future. Useful for debugging."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"future.tapOk(console.log);\n")),(0,n.kt)("h3",{id:"taperrorf"},".tapError(f)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.tapError(func: (value: E) => unknown): Future>\n")),(0,n.kt)("p",null,"Runs ",(0,n.kt)("inlineCode",{parentName:"p"},"f")," if value is ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," with the future value, and returns the original future. Useful for debugging."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"future.tapError(console.log);\n")),(0,n.kt)("h3",{id:"resulttopromise"},".resultToPromise()"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"Future>.resultToPromise(): Promise\n")),(0,n.kt)("p",null,"Takes a ",(0,n.kt)("inlineCode",{parentName:"p"},"Future>")," and returns a ",(0,n.kt)("inlineCode",{parentName:"p"},"Promise"),", rejecting the promise with ",(0,n.kt)("inlineCode",{parentName:"p"},"Error")," in this state."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"Future.value(Result.Ok(1)).resultToPromise();\n// Promise<1>\n\nFuture.value(Result.Reject(1)).resultToPromise();\n// Promise (rejected with 1)\n")),(0,n.kt)("h2",{id:"statics"},"Statics"),(0,n.kt)("h3",{id:"futureallresultfutures"},"Future.all(resultFutures)"),(0,n.kt)("p",null,"You can combine the ",(0,n.kt)("inlineCode",{parentName:"p"},"all")," helpers from ",(0,n.kt)("inlineCode",{parentName:"p"},"Future")," and ",(0,n.kt)("inlineCode",{parentName:"p"},"Result"),":"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"const futures = [\n Future.value(Result.Ok(1)),\n Future.value(Result.Ok(2)),\n Future.value(Result.Ok(3)),\n];\n\nFuture.all(futures).map(Result.all);\n// Future>\n")),(0,n.kt)("p",null,"Let's see the types at each step:"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"// Array>>\n// -> [Future>, Future>, Future>]\nconst input = [\n Future.value(Result.Ok(1)),\n Future.value(Result.Ok(2)),\n Future.value(Result.Ok(3)),\n];\n\n// Future>>\n// -> Future<[Result.Ok<1>>, Result.Ok<2>>, Result.Ok<3>]>\nconst step1 = Future.all(input);\n\n// Future, never>>\n// -> Future<[Result.Ok<[1, 2, 3]>>\nconst step2 = step1.map(Result.all);\n")),(0,n.kt)("h3",{id:"futureallfromdictresultfutures"},"Future.allFromDict(resultFutures)"),(0,n.kt)("p",null,"Like as ",(0,n.kt)("inlineCode",{parentName:"p"},"all"),", you can combine the ",(0,n.kt)("inlineCode",{parentName:"p"},"allFromDict"),":"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"const futures = {\n a: Future.value(Result.Ok(1)),\n b: Future.value(Result.Ok(2)),\n c: Future.value(Result.Ok(3)),\n};\n\nFuture.allFromDict(futures).map(Result.allFromDict);\n// Future<[Result.Ok<{a: 1, b: 2, c: 3}>>\n")),(0,n.kt)("p",null,"Let's see the types at each step:"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"// Dict>>\n// -> {a: Future>, b: Future>, c: Future>\u2014\nconst input = {\n a: Future.value(Result.Ok(1)),\n b: Future.value(Result.Ok(2)),\n c: Future.value(Result.Ok(3)),\n};\n\n// Future>>\n// -> Future<{a: Result.Ok<1>>, b: Result.Ok<2>>, c: Result.Ok<3>}>\nconst step1 = Future.all(input);\n\n// Future, never>>\n// -> Future<[Result.Ok<{a: 1, b: 2, c: 3}>>\nconst step2 = step1.map(Result.all);\n")),(0,n.kt)("h3",{id:"futureretrygetfuture"},"Future.retry(getFuture)"),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts"},"retry(getFuture: () => Future>, {max: number}): Future>\n")),(0,n.kt)("p",null,"Runs the future getter, if the future resolves with a ",(0,n.kt)("inlineCode",{parentName:"p"},"Result.Error"),", retries until hitting ",(0,n.kt)("inlineCode",{parentName:"p"},"max")," attempts."),(0,n.kt)("p",null,"The function provides a 0-based ",(0,n.kt)("inlineCode",{parentName:"p"},"attempt")," count to the function if you need to implement delay logic."),(0,n.kt)("pre",null,(0,n.kt)("code",{parentName:"pre",className:"language-ts",metastring:'title="Examples"',title:'"Examples"'},"// retry immediately after failure\nFuture.retry(\n (attempt) => {\n return getUserById(userId);\n },\n { max: 3 },\n);\n// Future>\n\n// adding delay\nFuture.retry(\n (attempt) => {\n return Future.wait(attempt * 100).flatMap(() => getUserById(userId));\n },\n { max: 10 },\n);\n// Future>\n")),(0,n.kt)("h2",{id:"cheatsheet"},"Cheatsheet"),(0,n.kt)("table",null,(0,n.kt)("thead",{parentName:"table"},(0,n.kt)("tr",{parentName:"thead"},(0,n.kt)("th",{parentName:"tr",align:null},"Method"),(0,n.kt)("th",{parentName:"tr",align:null},"Input"),(0,n.kt)("th",{parentName:"tr",align:null},"Function input"),(0,n.kt)("th",{parentName:"tr",align:null},"Function output"),(0,n.kt)("th",{parentName:"tr",align:null},"Returned value"))),(0,n.kt)("tbody",{parentName:"table"},(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapoktoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOkToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Ok(y)")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapoktoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOkToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Error(f)")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapoktoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOkToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrortoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapErrorToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Ok(y)")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrortoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapErrorToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Error(f)")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrortoresultf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapErrorToResult"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"y")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#mapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#maperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"mapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"f")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"x")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmapokf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapOk"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmaperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not provided")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("em",{parentName:"td"},"not executed")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(x))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmaperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Ok(y))"))),(0,n.kt)("tr",{parentName:"tbody"},(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("a",{parentName:"td",href:"#flatmaperrorf"},(0,n.kt)("inlineCode",{parentName:"a"},"flatMapError"))),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(e))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"e")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))")),(0,n.kt)("td",{parentName:"tr",align:null},(0,n.kt)("inlineCode",{parentName:"td"},"Future(Error(f))"))))))}s.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/runtime~main.c9e18d27.js b/assets/js/runtime~main.6daa7e3e.js similarity index 98% rename from assets/js/runtime~main.c9e18d27.js rename to assets/js/runtime~main.6daa7e3e.js index 54d211c..e7fcad6 100644 --- a/assets/js/runtime~main.c9e18d27.js +++ b/assets/js/runtime~main.6daa7e3e.js @@ -1 +1 @@ -(()=>{"use strict";var e,t,r,a,o,d={},f={};function n(e){var t=f[e];if(void 0!==t)return t.exports;var r=f[e]={id:e,loaded:!1,exports:{}};return d[e].call(r.exports,r,r.exports,n),r.loaded=!0,r.exports}n.m=d,n.c=f,e=[],n.O=(t,r,a,o)=>{if(!r){var d=1/0;for(i=0;i=o)&&Object.keys(n.O).every((e=>n.O[e](r[c])))?r.splice(c--,1):(f=!1,o0&&e[i-1][2]>o;i--)e[i]=e[i-1];e[i]=[r,a,o]},n.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return n.d(t,{a:t}),t},r=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,n.t=function(e,a){if(1&a&&(e=this(e)),8&a)return e;if("object"==typeof e&&e){if(4&a&&e.__esModule)return e;if(16&a&&"function"==typeof e.then)return e}var o=Object.create(null);n.r(o);var d={};t=t||[null,r({}),r([]),r(r)];for(var f=2&a&&e;"object"==typeof f&&!~t.indexOf(f);f=r(f))Object.getOwnPropertyNames(f).forEach((t=>d[t]=()=>e[t]));return d.default=()=>e,n.d(o,d),o},n.d=(e,t)=>{for(var r in t)n.o(t,r)&&!n.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},n.f={},n.e=e=>Promise.all(Object.keys(n.f).reduce(((t,r)=>(n.f[r](e,t),t)),[])),n.u=e=>"assets/js/"+({20:"977fb8cd",53:"935f2afb",61:"44ef08bc",78:"168e900d",109:"75a55335",162:"d589d3a7",195:"c4f5d8e4",217:"3b8c55ea",235:"f38b294d",241:"d8fa0e62",245:"3d955d88",271:"1d90b0b1",315:"e9c55326",385:"974039d5",437:"600fab28",476:"618f8fb4",498:"28b9547a",514:"1be78505",527:"ace312b1",571:"7958baa0",611:"a7da88a7",620:"927d9381",658:"43cd4838",675:"984db706",887:"d5efc89d",918:"17896441",920:"1a4e3797",960:"414a3798"}[e]||e)+"."+{20:"48a989ec",53:"2c4fc095",61:"9f344e37",78:"3c6dcb38",109:"70de1b18",162:"e6f49f4f",195:"a3dda12c",217:"6aee7597",235:"1111032d",241:"fdc8a617",245:"7c2822bc",271:"a2e9df6a",315:"bf7580ff",385:"69facaf0",437:"48886f91",476:"80c6c655",498:"e6eb6c9c",514:"eda58c99",527:"a5b8073b",571:"7ccf143a",611:"b5f2b6f0",620:"ec232123",658:"10b358ab",675:"ef09508a",780:"7f861c8c",814:"bb50acf4",887:"cc162c4a",894:"662af41b",918:"4c0895fa",920:"ecc9196e",945:"e8a8f394",960:"3626043e",972:"af7814cb"}[e]+".js",n.miniCssF=e=>{},n.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),n.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),a={},o="docs:",n.l=(e,t,r,d)=>{if(a[e])a[e].push(t);else{var f,c;if(void 0!==r)for(var b=document.getElementsByTagName("script"),i=0;i{f.onerror=f.onload=null,clearTimeout(s);var o=a[e];if(delete a[e],f.parentNode&&f.parentNode.removeChild(f),o&&o.forEach((e=>e(r))),t)return t(r)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:f}),12e4);f.onerror=l.bind(null,f.onerror),f.onload=l.bind(null,f.onload),c&&document.head.appendChild(f)}},n.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},n.p="/boxed/",n.gca=function(e){return e={17896441:"918","977fb8cd":"20","935f2afb":"53","44ef08bc":"61","168e900d":"78","75a55335":"109",d589d3a7:"162",c4f5d8e4:"195","3b8c55ea":"217",f38b294d:"235",d8fa0e62:"241","3d955d88":"245","1d90b0b1":"271",e9c55326:"315","974039d5":"385","600fab28":"437","618f8fb4":"476","28b9547a":"498","1be78505":"514",ace312b1:"527","7958baa0":"571",a7da88a7:"611","927d9381":"620","43cd4838":"658","984db706":"675",d5efc89d:"887","1a4e3797":"920","414a3798":"960"}[e]||e,n.p+n.u(e)},(()=>{var e={303:0,532:0};n.f.j=(t,r)=>{var a=n.o(e,t)?e[t]:void 0;if(0!==a)if(a)r.push(a[2]);else if(/^(303|532)$/.test(t))e[t]=0;else{var o=new Promise(((r,o)=>a=e[t]=[r,o]));r.push(a[2]=o);var d=n.p+n.u(t),f=new Error;n.l(d,(r=>{if(n.o(e,t)&&(0!==(a=e[t])&&(e[t]=void 0),a)){var o=r&&("load"===r.type?"missing":r.type),d=r&&r.target&&r.target.src;f.message="Loading chunk "+t+" failed.\n("+o+": "+d+")",f.name="ChunkLoadError",f.type=o,f.request=d,a[1](f)}}),"chunk-"+t,t)}},n.O.j=t=>0===e[t];var t=(t,r)=>{var a,o,d=r[0],f=r[1],c=r[2],b=0;if(d.some((t=>0!==e[t]))){for(a in f)n.o(f,a)&&(n.m[a]=f[a]);if(c)var i=c(n)}for(t&&t(r);b{"use strict";var e,t,r,a,o,d={},f={};function n(e){var t=f[e];if(void 0!==t)return t.exports;var r=f[e]={id:e,loaded:!1,exports:{}};return d[e].call(r.exports,r,r.exports,n),r.loaded=!0,r.exports}n.m=d,n.c=f,e=[],n.O=(t,r,a,o)=>{if(!r){var d=1/0;for(i=0;i=o)&&Object.keys(n.O).every((e=>n.O[e](r[c])))?r.splice(c--,1):(f=!1,o0&&e[i-1][2]>o;i--)e[i]=e[i-1];e[i]=[r,a,o]},n.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return n.d(t,{a:t}),t},r=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,n.t=function(e,a){if(1&a&&(e=this(e)),8&a)return e;if("object"==typeof e&&e){if(4&a&&e.__esModule)return e;if(16&a&&"function"==typeof e.then)return e}var o=Object.create(null);n.r(o);var d={};t=t||[null,r({}),r([]),r(r)];for(var f=2&a&&e;"object"==typeof f&&!~t.indexOf(f);f=r(f))Object.getOwnPropertyNames(f).forEach((t=>d[t]=()=>e[t]));return d.default=()=>e,n.d(o,d),o},n.d=(e,t)=>{for(var r in t)n.o(t,r)&&!n.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},n.f={},n.e=e=>Promise.all(Object.keys(n.f).reduce(((t,r)=>(n.f[r](e,t),t)),[])),n.u=e=>"assets/js/"+({20:"977fb8cd",53:"935f2afb",61:"44ef08bc",78:"168e900d",109:"75a55335",162:"d589d3a7",195:"c4f5d8e4",217:"3b8c55ea",235:"f38b294d",241:"d8fa0e62",245:"3d955d88",271:"1d90b0b1",315:"e9c55326",385:"974039d5",437:"600fab28",476:"618f8fb4",498:"28b9547a",514:"1be78505",527:"ace312b1",571:"7958baa0",611:"a7da88a7",620:"927d9381",658:"43cd4838",675:"984db706",887:"d5efc89d",918:"17896441",920:"1a4e3797",960:"414a3798"}[e]||e)+"."+{20:"48a989ec",53:"2c4fc095",61:"9f344e37",78:"3c6dcb38",109:"70de1b18",162:"e6f49f4f",195:"a3dda12c",217:"6aee7597",235:"1111032d",241:"fdc8a617",245:"7c2822bc",271:"a2e9df6a",315:"bf7580ff",385:"69facaf0",437:"48886f91",476:"80c6c655",498:"8b9149cd",514:"eda58c99",527:"a5b8073b",571:"7ccf143a",611:"b5f2b6f0",620:"ec232123",658:"10b358ab",675:"ef09508a",780:"7f861c8c",814:"bb50acf4",887:"cc162c4a",894:"662af41b",918:"4c0895fa",920:"ecc9196e",945:"e8a8f394",960:"3626043e",972:"af7814cb"}[e]+".js",n.miniCssF=e=>{},n.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),n.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),a={},o="docs:",n.l=(e,t,r,d)=>{if(a[e])a[e].push(t);else{var f,c;if(void 0!==r)for(var b=document.getElementsByTagName("script"),i=0;i{f.onerror=f.onload=null,clearTimeout(s);var o=a[e];if(delete a[e],f.parentNode&&f.parentNode.removeChild(f),o&&o.forEach((e=>e(r))),t)return t(r)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:f}),12e4);f.onerror=l.bind(null,f.onerror),f.onload=l.bind(null,f.onload),c&&document.head.appendChild(f)}},n.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},n.p="/boxed/",n.gca=function(e){return e={17896441:"918","977fb8cd":"20","935f2afb":"53","44ef08bc":"61","168e900d":"78","75a55335":"109",d589d3a7:"162",c4f5d8e4:"195","3b8c55ea":"217",f38b294d:"235",d8fa0e62:"241","3d955d88":"245","1d90b0b1":"271",e9c55326:"315","974039d5":"385","600fab28":"437","618f8fb4":"476","28b9547a":"498","1be78505":"514",ace312b1:"527","7958baa0":"571",a7da88a7:"611","927d9381":"620","43cd4838":"658","984db706":"675",d5efc89d:"887","1a4e3797":"920","414a3798":"960"}[e]||e,n.p+n.u(e)},(()=>{var e={303:0,532:0};n.f.j=(t,r)=>{var a=n.o(e,t)?e[t]:void 0;if(0!==a)if(a)r.push(a[2]);else if(/^(303|532)$/.test(t))e[t]=0;else{var o=new Promise(((r,o)=>a=e[t]=[r,o]));r.push(a[2]=o);var d=n.p+n.u(t),f=new Error;n.l(d,(r=>{if(n.o(e,t)&&(0!==(a=e[t])&&(e[t]=void 0),a)){var o=r&&("load"===r.type?"missing":r.type),d=r&&r.target&&r.target.src;f.message="Loading chunk "+t+" failed.\n("+o+": "+d+")",f.name="ChunkLoadError",f.type=o,f.request=d,a[1](f)}}),"chunk-"+t,t)}},n.O.j=t=>0===e[t];var t=(t,r)=>{var a,o,d=r[0],f=r[1],c=r[2],b=0;if(d.some((t=>0!==e[t]))){for(a in f)n.o(f,a)&&(n.m[a]=f[a]);if(c)var i=c(n)}for(t&&t(r);b AsyncData<Result<Ok, Error>> | Boxed - +

AsyncData<Result<Ok, Error>>

A AsyncData can contain a Result (e.g. to represent an asynchronous value that can fail). We provide some utility functions to deal with that case without having to unwrap the AsyncData result.

note

You can still use all the regular AsyncData methods. The following helpers simply removes the need to unwrap the contained result.

Methods

.mapOkToResult(f)

AsyncData<Result<A, E>>.mapOkToResult<B, F>(
func: (value: A) => Result<B, F>,
): AsyncData<Result<B, E | F>>

Takes a AsyncData<Result<Ok, Error>> and a f function taking Ok and returning Result<ReturnValue, Error> and returns a new AsyncData<Result<ReturnValue, Error>>

Examples
AsyncData.Done(Result.Ok(3)).mapOkToResult((ok) => {
return Result.Ok(ok * 2);
});
// AsyncData<Result.Ok<6>>

AsyncData.Done(Result.Ok(3)).mapOkToResult((ok) =>
isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");
);
// AsyncData<Result.Error<"Odd number">>

.mapErrorToResult(f)

AsyncData<Result<A, E>>.mapErrorToResult<B, F>(
func: (value: E) => Result<B, F>,
): AsyncData<Result<A | B, F>>

Takes a AsyncData<Result<Ok, Error>> and a f function taking Error and returning Result<ReturnValue, Error> and returns a new AsyncData<Result<ReturnValue, Error>>

Examples
AsyncData.Done(Result.Error(3)).mapErrorToResult((error) => {
return Result.Ok(ok * 2);
});
// AsyncData<Result.Ok<6>>

AsyncData.Done(Result.Error(3)).mapErrorToResult((error) =>
isEven(error) ? Result.Ok(error) : Result.Error("Odd number");
);
// AsyncData<Result.Error<"Odd number">>

.mapOk(f)

AsyncData<Result<A, E>>.mapOk<B>(
func: (value: A) => B,
): AsyncData<Result<B, E>>

Takes a AsyncData<Result<Ok, Error>> and a f function taking Ok and returning ReturnValue and returns a new AsyncData<Result<ReturnValue, Error>>

Examples
AsyncData.Done(Result.Ok(3)).mapOk((ok) => {
return ok * 2;
});
// AsyncData<Result.Ok<6>>

AsyncData.Done(Result.Error("something")).mapOk((ok) => {
return ok * 2;
});
// AsyncData<Result.Error<"something">>

.mapError(f)

AsyncData<Result<A, E>>.mapError<F>(
func: (value: E) => F,
): AsyncData<Result<A, F>>

Takes a AsyncData<Result<Ok, Error>> and a f function taking Error and returning ReturnValue and returns a new AsyncData<Result<Ok, ReturnValue>>

Examples
AsyncData.Done(Result.Error(3)).mapError((error) => {
return error * 2;
});
// AsyncData<Result.Error<6>>

AsyncData.Done(Result.Ok("something")).mapError((ok) => {
return ok * 2;
});
// AsyncData<Result.Ok<"something">>

.flatMapOk(f)

AsyncData<Result<A, E>>.flatMapOk<B, F>(
func: (value: A) => AsyncData<Result<B, F>>,
): AsyncData<Result<B, E | F>>

Takes a AsyncData<Result<Ok, Error>> and a f function taking Ok returning a AsyncData<Result<ReturnValue, Error>>

Examples
AsyncData.Done(Result.Ok(3)).flatMapOk((ok) =>
AsyncData.Done(Result.Ok(ok * 2)),
);
// AsyncData<Result.Ok<6>>

AsyncData.Done(Result.Ok(3)).flatMapOk((ok) =>
AsyncData.Done(Result.Error("Nope")),
);
// AsyncData<Result.Error<"Nope">>

AsyncData.Done(Result.Error("Error")).flatMapOk((ok) =>
AsyncData.Done(Result.Ok(ok * 2)),
);
// AsyncData<Result.Error<"Error">>

.flatMapError(f)

AsyncData<Result<A, E>>.flatMapError<B, F>(
func: (value: E) => AsyncData<Result<B, F>>,
): AsyncData<Result<A | B, F>>

Takes a AsyncData<Result<Ok, Error>> and a f function taking Error returning a AsyncData<Result<Ok, ReturnValue>>

Examples
AsyncData.Done(Result.Ok(3)).flatMapError((error) =>
AsyncData.Done(Result.Ok(ok * 2)),
);
// AsyncData<Result.Ok<3>>

AsyncData.Done(Result.Error("Error")).flatMapError((error) =>
AsyncData.Done(Result.Error("Nope")),
);
// AsyncData<Result.Error<"Nope">>

AsyncData.Done(Result.Error("Error")).flatMapError((error) =>
AsyncData.Done(Result.Ok(1)),
);
// AsyncData<Result.Ok<1>>

Cheatsheet

MethodInputFunction inputFunction outputReturned value
mapOkToResultAsyncData(Ok(x))xOk(y)AsyncData(Ok(y))
mapOkToResultAsyncData(Ok(x))xError(f)AsyncData(Error(f))
mapOkToResultAsyncData(Error(e))not providednot executedAsyncData(Error(e))
mapErrorToResultAsyncData(Error(e))eOk(y)AsyncData(Ok(y))
mapErrorToResultAsyncData(Error(e))eError(f)AsyncData(Error(f))
mapErrorToResultAsyncData(Ok(x))not providednot executedAsyncData(Ok(x))
mapOkAsyncData(Ok(x))xyAsyncData(Ok(y))
mapOkAsyncData(Error(e))not providednot executedAsyncData(Error(e))
mapErrorAsyncData(Ok(x))not providednot executedAsyncData(Ok(x))
mapErrorAsyncData(Error(e))efAsyncData(Error(f))
flatMapOkAsyncData(Ok(x))xAsyncData(Ok(y))AsyncData(Ok(y))
flatMapOkAsyncData(Ok(x))xAsyncData(Error(f))AsyncData(Error(f))
flatMapOkAsyncData(Error(e))not providednot executedAsyncData(Error(e))
flatMapErrorAsyncData(Ok(x))not providednot executedAsyncData(Ok(x))
flatMapErrorAsyncData(Error(e))eAsyncData(Ok(y))AsyncData(Ok(y))
flatMapErrorAsyncData(Error(e))eAsyncData(Error(f))AsyncData(Error(f))
- + \ No newline at end of file diff --git a/async-data/index.html b/async-data/index.html index 57ae527..b75c091 100644 --- a/async-data/index.html +++ b/async-data/index.html @@ -5,13 +5,13 @@ AsyncData<Value> | Boxed - +

AsyncData<Value>

The AsyncData type enables representing asynchronous flows (e.g. requests). The type represents the state as a discriminating union, avoiding manual management for loading flows.

AsyncData can have three possible states:

  • NotAsked
  • Loading
  • Done(value)

Create an AsyncData value

To create an async data, use the NotAsked, Loading and Done constructors:

Examples
import { AsyncData } from "@swan-io/boxed";

const notAsked = AsyncData.NotAsked();

const loading = AsyncData.Loading();

const done = AsyncData.Done(1);

Methods

The async data type provides a few manipulation functions:

.map(f)

AsyncData<A>.map<B>(f: (value: A) => B): AsyncData<B>

If the asyncData is Done(value) returns Done(f(value)), otherwise returns the async data.

Examples
AsyncData.Done(2).map((x) => x * 2);
// AsyncData.Done<4>

AsyncData.Loading().map((x) => x * 2);
// AsyncData.Loading

AsyncData.NotAsked().map((x) => x * 2);
// AsyncData.NotAsked

.flatMap(f)

AsyncData<A>.flatMap<B>(f: (value: A) => AsyncData<B>): AsyncData<B>

If the asyncData is Done(value) returns f(value), otherwise returns the async data.

Examples
AsyncData.Done(3).flatMap((x) =>
x > 2 ? AsyncData.NotAsked() : AsyncData.Done(2),
);
// AsyncData.NotAsked

AsyncData.Done(1).flatMap((x) =>
x > 2 ? AsyncData.NotAsked() : AsyncData.Done(2),
);
// AsyncData.Done<2>

AsyncData.NotAsked().flatMap((x) =>
x > 2 ? AsyncData.NotAsked() : AsyncData.Done(2),
);
// AsyncData.NotAsked

AsyncData.Loading().flatMap((x) =>
x > 2 ? AsyncData.NotAsked() : AsyncData.Done(2),
);
// AsyncData.Loading

.getOr(defaultValue)

Alias: getWithDefault

AsyncData<A>.getOr(defaultValue: A): A

If the async data is Done(value) returns value, otherwise returns defaultValue.

Examples
AsyncData.Done(2).getOr(1);
// 2

AsyncData.Loading().getOr(1);
// 1

AsyncData.NotAsked().getOr(1);
// 1

.get()

AsyncData<A>.get(): A

Returns the value contained in Done(value). Only usable within a isDone() check.

Examples
const value = asyncData.get();
// does not compile

if (asyncData.isDone()) {
const value = asyncData.get();
// value
}

.isDone()

AsyncData<A>.isDone(): boolean

Type guard. Checks if the option is Done(value)

Examples
AsyncData.Done(2).isDone();
// true

AsyncData.Loading().isDone();
// false

AsyncData.NotAsked().isDone();
// false

if (asyncData.isDone()) {
const value = asyncData.get();
}

.isLoading()

AsyncData<A>.isLoading(): boolean

Type guard. Checks if the option is Loading

Examples
AsyncData.Done(2).isLoading();
// false

AsyncData.Loading().isLoading();
// true

AsyncData.NotAsked().isLoading();
// false

.isNotAsked()

AsyncData<A>.isNotAsked(): boolean

Type guard. Checks if the option is NotAsked

Examples
AsyncData.Done(2).isNotAsked();
// false

AsyncData.Loading().isNotAsked();
// false

AsyncData.NotAsked().isNotAsked();
// true

.toOption()

AsyncData<A>.toOption(): Option<A>

If the result is Done(value) returns Some(value), otherwise returns None.

Examples
Result.Done(2).toOption();
// Option.Some<2>

Result.Loading().toOption();
// Option.None

Result.NotAsked().toOption();
// Option.None

.match()

AsyncData<A>.match<B>(config: {
Done: (value: A) => B;
Loading: () => B;
NotAsked: () => B;
}): B;

Match the async data state

Examples
const valueToDisplay = result.match({
Done: (value) => value,
Loading: () => "Loading ...",
NotAsked: () => "",
});

.tap(func)

AsyncData<A>.tap(func: (asyncData: AsyncData<A>) => unknown): AsyncData<A>

Executes func with asyncData, and returns asyncData. Useful for logging and debugging.

Examples
asyncData.tap(console.log).map((x) => x * 2);

Statics

AsyncData.isAsyncData(value)

isAsyncData(value: unknown): boolean

Type guard, checks if the provided value is an asyncData.

Examples
AsyncData.isAsyncData(AsyncData.Done(1));
// true

AsyncData.isAsyncData([]);
// false

AsyncData.all(asyncDatas)

all(asyncDatas: Array<AsyncData<A>>): AsyncData<Array<A>>

Turns an "array of asyncDatas of value" into a "asyncData of array of value".

Examples
AsyncData.all([AsyncData.Done(1), AsyncData.Done(2), AsyncData.Done(3)]);
// AsyncData.Done<[1, 2, 3]>

AsyncData.all([Result.NotAsked(), AsyncData.Done(2), AsyncData.Done(3)]);
// AsyncData.NotAsked

AsyncData.all([Result.Loading(), AsyncData.Done(2), AsyncData.Done(3)]);
// AsyncData.Loading

AsyncData.allFromDict(asyncDatas)

allFromDict(asyncDatas: Dict<AsyncData<A>>): AsyncData<Dict<A>>

Turns a "dict of asyncDatas of value" into a "asyncData of dict of value".

Examples
AsyncData.allFromDict({
a: AsyncData.Done(1),
b: AsyncData.Done(2),
c: AsyncData.Done(3),
});
// AsyncData.Done<{a: 1, b: 2, c: 3}>

AsyncData.allFromDict({
a: Result.NotAsked(),
b: AsyncData.Done(2),
c: AsyncData.Done(3),
});
// AsyncData.NotAsked

AsyncData.allFromDict({
a: Result.Loading(),
b: AsyncData.Done(2),
c: AsyncData.Done(3),
});
// AsyncData.Loading

TS Pattern interop

Examples
import { match, P } from "ts-pattern";
import { AsyncData } from "@swan-io/boxed";

match(asyncData)
.with(AsyncData.P.Done(P.select()), (value) => console.log(value))
.with(AsyncData.P.Loading, () => "Loading ...")
.with(AsyncData.P.NotAsked, () => "")
.exhaustive();

Cheatsheet

MethodInputFunction inputFunction outputReturned value
mapDone(x)xyDone(y)
mapLoading()not providednot executedLoading()
mapNotAsked()not providednot executedNotAsked()
flatMapDone(x)xDone(y)Done(y)
flatMapDone(x)xLoading()Loading()
flatMapDone(x)xNotAsked()NotAsked()
flatMapLoading()not providednot executedLoading()
flatMapNotAsked()not providednot executedNotAsked()
- + \ No newline at end of file diff --git a/cancellable-request/index.html b/cancellable-request/index.html index 9b1f2aa..069707d 100644 --- a/cancellable-request/index.html +++ b/cancellable-request/index.html @@ -5,13 +5,13 @@ Cancellable Request | Boxed - +

Cancellable Request

When using the naive fetch, cancelling a request can be inelegant:

import { useEffect } from "react";
import { Result } from "@swan-io/boxed";

const callMyApi = (url, { signal }) => {
return fetch(url, { signal })
.then((res) => res.json())
.then((json) => Result.Ok(res))
.catch((error) => Result.Error(error));
};

// ...
useEffect(() => {
// Implementation details leak to your components
const controller = new AbortController();
callMyApi("/users", { signal: controller.signal })
.then((res) => res.json())
.then((json) => setState(Result.Ok(json)))
.catch((error) => setState(Result.Error(error)));
return () => controller.abort();
}, []);

Using Future can make this easier:

import { useEffect } from "react";
import { Future, Result } from "@swan-io/boxed";

const callMyApi = (url: string) =>
Future.make((resolve) => {
const controller = new AbortController();

fetch(url, { signal: controller.signal })
.then((res) => res.json())
.then((json) => resolve(Result.Ok(json)))
.catch((error) => resolve(Result.Error(error)));

// Here, the implementation detail is managed in place
return () => controller.abort();
});

// And the noise dissapears from your components!
useEffect(() => {
const request = callMyApi("/api").tap(setState);
return () => request.cancel();
}, []);
- + \ No newline at end of file diff --git a/core-concepts/index.html b/core-concepts/index.html index c5f8dca..9b28ba1 100644 --- a/core-concepts/index.html +++ b/core-concepts/index.html @@ -5,13 +5,13 @@ Core Concepts | Boxed - +

Core Concepts

The Boxed approach takes root in typed functional paradigms. We know that these concepts can be overwhelming, especially with their jargon and mathematical concepts, and therefore want to make them more accessible.

As beautiful and powerful these concepts are, they come with a huge learning curve that we want to avoid. We also want your code to be simple to read, write and reason about without having to know the full theory behind.

danger

If you have a strong opinion on what metaphor to use to describe monads, please stop reading this page immediately and visit the API reference.

Schrödinger's cat

When physicists discovered that particules do weird stuff, they decided that while we don't look at them, they're in a superposition of states.

Erwin Schrödinger didn't like that one bit and decided to show them how utterly stupid it was. For that, he created a thought experiment now called the Schrödinger's cat experiment, in which a cat is put into a box (roll credits) with a cat-killing device reacting to some quantum stuff. While we don't open the box, the cat is both alive and dead as the same time, when we open the box, we fix an outcome.

Schrödinger&#39;s cat experiment

info

We love cats and the boxes used in the following explanations will only contain JavaScript values.

Boxes

The way we like to think of the data-structures we expose are that they're boxes (think of it as containers) that may or may not contain a value.

Here's a visual example using the Option type. The Option represents an optional value, it can have two possible states: either Some(value) or None().

Option of blue circle, it can be either a box containing a blue circle, which we call Some blue circle, or an empty box, which we call None

Option is a generic type, meaning you can define what type of value it holds: Option<User>, Option<string> (just like an array: Array<User>, Array<string>).

In your program flow, you don't know what's inside, as if the box was closed.

When you extract the value from the box, you have a few options:

// Let's assument we have `option` be of type `Option<number>`

// Returns the value if present or the fallback otherwise
const a = option.getOr(0);

// Explode the box
const b = option.match({
Some: (value) => value,
None: () => 0,
});

Data-manipulation basics

Most of the data-manipulation you'll do comes down to two function: map and flatMap.

Here's a visual explanation:

The map function transforms the value with a callback, the flatMap function returns an box itself

The map and flatMap functions allow you to transform data in a typesafe way:

const some = Option.Some(1);
// Option.Some<1>

const none = Option.None();
// Option.None

const doubledSome = some.map((x) => x * 2);
// Option.Some<2>

const doubledNone = none.map((x) => x * 2);
// Option.None -> Nothing to transform!

The flatMap lets you return another option, which can be useful for nested optional values:

type UserInfo = {
name: Option<string>;
};

type User = {
id: string;
info: Option<UserInfo>;
};

const name = user
.flatMap((user) => user.info) // Returns the Option<UserInfo>
.flatMap((info) => info.name) // Returns the Option<string>
.getOr("Anonymous user");

The main kind of boxes

Option<Value>

Represents optional values:

  • Replaces undefined and null
  • Makes it possible to differentiate nested optionality (Some(None()) vs None())
  • Reduces the number of codepaths needed to read and transform such values

Result<Ok, Error>

Represents a computation that can either succeed or fail:

  • Replaces exceptions
  • Allows you to have a single codepath instead or "return or throw".
  • Makes it easy to aggregate all possible errors a stack can generate

AsyncData<Value>

Represents a value with an asynchronous lifecycle:

  • Eliminates impossibles cases in your state
  • Avoids inconsistent states induced by traditional modeling
  • Allows you to tie the lifecycle information with the value itself

Future<Value>

Represents an asynchronous value:

  • Replaces promises
  • Supports cancellation
  • Delegates success/failure to the Result type
  • Exposes a map & flatMap API
- + \ No newline at end of file diff --git a/deferred/index.html b/deferred/index.html index 7c12345..4bffb1c 100644 --- a/deferred/index.html +++ b/deferred/index.html @@ -5,13 +5,13 @@ Deferred<Value> | Boxed - +

Deferred<Value>

Deferred.make()

Returns a future and its resolver:

import { Deferred } from "@swan-io/boxed";

const [future, resolve] = Deferred.make<string>();

// subscribe to the promise
future.onResolve(console.log);

// resolve from elsewhere
resolve("Hello!");
- + \ No newline at end of file diff --git a/design-choices/index.html b/design-choices/index.html index cfc30de..5a2b992 100644 --- a/design-choices/index.html +++ b/design-choices/index.html @@ -5,13 +5,13 @@ Design choices | Boxed - +

Design choices

Chaining API

Most functional programming libraries in JavaScript provide a data-last API (map(func)(value)) along with a pipe function to chain operations.

While this approach works well in languages that include some kind of native pipe operator, we consider that it doesn't provide a good enough developer experience in your TypeScript code editor: it makes it tedious to inspect a value in a pipe chain and doesn't provide autocomplete natively.

For this reason, we decided to use good ol'chaining, reducing the number of imports, making your code more expressive, easier to read and inspect.

Eager over lazy

More often than not, these functional programming libraries use lazy initialisation, meaning your code won't execute until you tell it to explicitely (e.g. with a run call at the end).

In the vast majority of cases we've seen, this is not something that's wanted. For the few cases where laziness is wanted, making a function that returns the data-structure or using a Deferred does the job pretty well.

const eagerFuture = Future.make((resolve) => resolve(1));

const lazyFuture = () => Future.make((resolve) => resolve(1));

const [deferred, resolve] = Deferred.make<number>();

Naming

Rather than using naming from abstract theory, if a concept exists in JavaScript built-ins, Boxed will provide similar naming (e.g. we provide Future.all to mimic Promise.all rather than Future.sequenceArray).

- + \ No newline at end of file diff --git a/dict/index.html b/dict/index.html index d919d7b..c2cb355 100644 --- a/dict/index.html +++ b/dict/index.html @@ -5,13 +5,13 @@ Dict | Boxed - +

Dict

import { Dict } from "@swan-io/boxed";

Dict.entries(dict)

Returns the entries in the dict.

Contrary to the TS bindings for Object.entries, the types are refined.

Examples
const index = Dict.entries({ foo: 1, bar: 2, baz: 3 });
// [["foo", 1], ["bar", 2], ["baz", 3]];

Dict.fromEntries(entries)

Returns a dict from the provided [key, value] pairs.

Examples
const dict = Dict.fromEntries([
["foo", 1],
["bar", 2],
["baz", 3],
]);
// { foo: 1, bar: 2, baz: 3 };

Dict.keys(dict)

Returns the keys in the dict.

Contrary to the TS bindings for Object.keys, the types are refined.

Examples
const index = Dict.keys({ foo: 1, bar: 2, baz: 3 });
// ["foo", "bar", "baz"];

Dict.values(dict)

Returns the values in the dict.

Contrary to the TS bindings for Object.values, the types are refined.

Examples
const index = Dict.values({ foo: 1, bar: 2, baz: 3 });
// [1, 2, 3];

Dict.fromOptional(dictOfOptions)

Takes a dict whose values are Option<unknown> and returns a dict containing only the values contained in Some.

Examples
Dict.fromOptional({
foo: Option.Some(1),
bar: Option.None(),
baz: Option.None(),
});
// {foo: 1}

Dict.fromOptional({
foo: Option.Some(1),
bar: Option.Some(2),
baz: Option.None(),
});
// {foo: 1, bar: 2}

Dict.fromOptional({
foo: Option.None(),
bar: Option.None(),
baz: Option.None(),
});
// {}
- + \ No newline at end of file diff --git a/form-validation/index.html b/form-validation/index.html index b2e0cbd..21365f2 100644 --- a/form-validation/index.html +++ b/form-validation/index.html @@ -5,13 +5,13 @@ Form Validation | Boxed - +

Form Validation

A common need in applications is to validate user-input before sending it to the server.

Let's assume we have a form with the following fields:

type FormInput = {
id: string;
amount: number;
};

One can use exceptions:

const validate = (input: FormInput) => {
if (input.id.trim().length !== 24) {
throw new Error("Input ID is invalid");
}
if (input.amount <= 0) {
throw new Error("Invalid amount");
}
};

In that case, we'd use a try statement:

try {
const sanitized = sanitize(input);
validate(sanitized);
setValidation(null);

// send to the server
} catch (err) {
setValidation(err);
}

Or one can return errors from the validate function:

const validate = (input: FormInput) => {
const errors = [];
if (input.id.trim().length !== 24) {
errors.push("Input ID is invalid");
}
if (input.amount <= 0) {
errors.push("Invalid amount");
}
return errors;
};

Which would be consumed like the following:

const sanitized = sanitize(input);
const errors = validate(sanitized);
if (errors.length) {
setValidation(errors);
// show the errors
} else {
setValidation(null);
// send to the server
}

In both cases, we are required to have handle the validation state manually, which increases complexity and can lead to UI inconsistencies. Let's see how we can leverage the Result type for such patterns:

import { Result } from "@swan-io/boxed";

const validate = (input: FormInput): Result<FormInput, Array<string>> => {
const errors = [];
const id = input.id.trim();
if (id.length !== 24) {
errors.push("Input ID is invalid");
}
if (input.amount <= 0) {
errors.push("Invalid amount");
}

// We can directly return a sanitized version if the validation passed
return errors.length === 0
? Result.Ok({ ...input, id })
: Result.Error(errors);
};

Here, the validate return value can directly give you the sanitized input or the validation errors, depending on which case you're in.

We can then store the Result directly:

// A single codepath for handling the validation
const validation = validate(input);

setValidation(validation);

validation.match({
Error: () => {} // do nothing
Ok: (sanitizedInput) => {
sendToServer(sanitizedInput)
}
})

// and pattern match in the UI code
<Input
name="id"
value={input.id}
onChange={id => setInput({...input, id}))}
hasError={validation.match({
Ok: () => false,
Error: (errors) => errors.includes("Input ID is invalid"),
})}
/>;
- + \ No newline at end of file diff --git a/future-result/index.html b/future-result/index.html index f9846e3..cabb9bf 100644 --- a/future-result/index.html +++ b/future-result/index.html @@ -5,13 +5,13 @@ Future<Result<Ok, Error>> | Boxed - +
-

Future<Result<Ok, Error>>

A Future can contain a Result (e.g. to represent an asynchronous value that can fail). We provide some utility functions to deal with that case without having to unwrap the Future result.

note

You can still use all the regular Future methods. The following helpers simply removes the need to unwrap the contained result.

Methods

.mapOkToResult(f)

Future<Result<A, E>>.mapOkToResult<B, F>(
func: (value: A) => Result<B, F>,
propagateCancel?: boolean
): Future<Result<B, E | F>>

Takes a Future<Result<Ok, Error>> and a f function taking Ok and returning Result<ReturnValue, Error> and returns a new Future<Result<ReturnValue, Error>>

Examples
Future.value(Result.Ok(3)).mapOkToResult((ok) => {
return Result.Ok(ok * 2);
});
// Future<Result.Ok<6>>

Future.value(Result.Ok(3)).mapOkToResult((ok) =>
isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");
);
// Future<Result.Error<"Odd number">>

.mapErrorToResult(f)

Future<Result<A, E>>.mapErrorToResult<B, F>(
func: (value: E) => Result<B, F>,
propagateCancel?: boolean
): Future<Result<A | B, F>>

Takes a Future<Result<Ok, Error>> and a f function taking Error and returning Result<ReturnValue, Error> and returns a new Future<Result<ReturnValue, Error>>

Examples
Future.value(Result.Error(3)).mapErrorToResult((ok) => {
return Result.Ok(ok * 2);
});
// Future<Result.Ok<6>>

Future.value(Result.Error(3)).mapErrorToResult((ok) =>
isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");
);
// Future<Result.Error<"Odd number">>

.mapOk(f)

Future<Result<A, E>>.mapOk<B>(
func: (value: A) => B,
propagateCancel?: boolean
): Future<Result<B, E>>

Takes a Future<Result<Ok, Error>> and a f function taking Ok and returning ReturnValue and returns a new Future<Result<ReturnValue, Error>>

Examples
Future.value(Result.Ok(3)).mapOk((ok) => {
return ok * 2;
});
// Future<Result.Ok<6>>

Future.value(Result.Error("something")).mapOk((ok) => {
return ok * 2;
});
// Future<Result.Error<"something">>

.mapError(f)

Future<Result<A, E>>.mapError<F>(
func: (value: E) => F,
propagateCancel?: boolean
): Future<Result<A, F>>

Takes a Future<Result<Ok, Error>> and a f function taking Error and returning ReturnValue and returns a new Future<Result<Ok, ReturnValue>>

Examples
Future.value(Result.Error(3)).mapError((error) => {
return error * 2;
});
// Future<Result.Error<6>>

Future.value(Result.Ok("something")).mapError((ok) => {
return ok * 2;
});
// Future<Result.Ok<"something">>

.flatMapOk(f)

Future<Result<A, E>>.flatMapOk<B, F>(
func: (value: A) => Future<Result<B, F>>,
propagateCancel?: boolean
): Future<Result<B, E | F>>

Takes a Future<Result<Ok, Error>> and a f function taking Ok returning a Future<Result<ReturnValue, Error>>

Examples
Future.value(Result.Ok(3)).flatMapOk((ok) => Future.value(Result.Ok(ok * 2)));
// Future<Result.Ok<6>>

Future.value(Result.Ok(3)).flatMapOk((ok) =>
Future.value(Result.Error("Nope")),
);
// Future<Result.Error<"Nope">>

Future.value(Result.Error("Error")).flatMapOk((ok) =>
Future.value(Result.Ok(ok * 2)),
);
// Future<Result.Error<"Error">>

.flatMapError(f)

Future<Result<A, E>>.flatMapError<B, F>(
func: (value: E) => Future<Result<B, F>>,
propagateCancel?: boolean
): Future<Result<A | B, F>>

Takes a Future<Result<Ok, Error>> and a f function taking Error returning a Future<Result<Ok, ReturnValue>>

Examples
Future.value(Result.Ok(3)).flatMapError((error) =>
Future.value(Result.Ok(ok * 2)),
);
// Future<Result.Ok<3>>

Future.value(Result.Error("Error")).flatMapError((error) =>
Future.value(Result.Error("Nope")),
);
// Future<Result.Error<"Nope">>

Future.value(Result.Error("Error")).flatMapError((error) =>
Future.value(Result.Ok(1)),
);
// Future<Result.Ok<1>>

.tapOk(f)

Future<Result<A, E>>.tapOk(func: (value: A) => unknown): Future<Result<A, E>>

Runs f if value is Ok with the future value, and returns the original future. Useful for debugging.

Examples
future.tapOk(console.log);

.tapError(f)

Future<Result<A, E>>.tapError(func: (value: E) => unknown): Future<Result<A, E>>

Runs f if value is Error with the future value, and returns the original future. Useful for debugging.

Examples
future.tapError(console.log);

.resultToPromise()

Future<Result<A, E>>.resultToPromise(): Promise<A>

Takes a Future<Result<Ok, Error>> and returns a Promise<Ok>, rejecting the promise with Error in this state.

Examples
Future.value(Result.Ok(1)).resultToPromise();
// Promise<1>

Future.value(Result.Reject(1)).resultToPromise();
// Promise (rejected with 1)

Statics

Future.all(resultFutures)

You can combine the all helpers from Future and Result:

Examples
const futures = [
Future.value(Result.Ok(1)),
Future.value(Result.Ok(2)),
Future.value(Result.Ok(3)),
];

Future.all(futures).map(Result.all);
// Future<Result.Ok<[1, 2, 3]>>

Let's see the types at each step:

Examples
// Array<Future<Result<number, never>>>
// -> [Future<Result.Ok<1>>, Future<Result.Ok<2>>, Future<Result.Ok<3>>]
const input = [
Future.value(Result.Ok(1)),
Future.value(Result.Ok(2)),
Future.value(Result.Ok(3)),
];

// Future<Array<Result<number, never>>>
// -> Future<[Result.Ok<1>>, Result.Ok<2>>, Result.Ok<3>]>
const step1 = Future.all(input);

// Future<Result<Array<number>, never>>
// -> Future<[Result.Ok<[1, 2, 3]>>
const step2 = step1.map(Result.all);

Future.allFromDict(resultFutures)

Like as all, you can combine the allFromDict:

Examples
const futures = {
a: Future.value(Result.Ok(1)),
b: Future.value(Result.Ok(2)),
c: Future.value(Result.Ok(3)),
};

Future.allFromDict(futures).map(Result.allFromDict);
// Future<[Result.Ok<{a: 1, b: 2, c: 3}>>

Let's see the types at each step:

Examples
// Dict<Future<Result<number, never>>>
// -> {a: Future<Result.Ok<1>>, b: Future<Result.Ok<2>>, c: Future<Result.Ok<3>>—
const input = {
a: Future.value(Result.Ok(1)),
b: Future.value(Result.Ok(2)),
c: Future.value(Result.Ok(3)),
};

// Future<Dict<Result<number, never>>>
// -> Future<{a: Result.Ok<1>>, b: Result.Ok<2>>, c: Result.Ok<3>}>
const step1 = Future.all(input);

// Future<Result<Array<number>, never>>
// -> Future<[Result.Ok<{a: 1, b: 2, c: 3}>>
const step2 = step1.map(Result.all);

Future.retry(getFuture)

retry(getFuture: () => Future<Result<A, E>>, {max: number}): Future<Result<A, E>>

Runs the future getter, if the future resolves with a Result.Error, retries until hitting max attempts.

The function provides a 0-based attempt count to the function if you need to implement delay logic.

Examples
// retry immediately after failure
Future.retry(
(attempt) => {
return getUserById(userId);
},
{ max: 3 },
);
// Future<Result<...>>

// adding delay
Future.retry(
(attempt) => {
return Future.wait(attempt * 100).flatMap(() => getUserById(userId));
},
{ max: 10 },
);
// Future<Result<...>>

Cheatsheet

MethodInputFunction inputFunction outputReturned value
mapOkToResultFuture(Ok(x))xOk(y)Future(Ok(y))
mapOkToResultFuture(Ok(x))xError(f)Future(Error(f))
mapOkToResultFuture(Error(e))not providednot executedFuture(Error(e))
mapErrorToResultFuture(Error(e))eOk(y)Future(Ok(y))
mapErrorToResultFuture(Error(e))eError(f)Future(Error(f))
mapErrorToResultFuture(Ok(x))not providednot executedFuture(Ok(x))
mapOkFuture(Ok(x))xyFuture(Ok(y))
mapOkFuture(Error(e))not providednot executedFuture(Error(e))
mapErrorFuture(Ok(x))not providednot executedFuture(Ok(x))
mapErrorFuture(Error(e))efFuture(Error(f))
flatMapOkFuture(Ok(x))xFuture(Ok(y))Future(Ok(y))
flatMapOkFuture(Ok(x))xFuture(Error(f))Future(Error(f))
flatMapOkFuture(Error(e))not providednot executedFuture(Error(e))
flatMapErrorFuture(Ok(x))not providednot executedFuture(Ok(x))
flatMapErrorFuture(Error(e))eFuture(Ok(y))Future(Ok(y))
flatMapErrorFuture(Error(e))eFuture(Error(f))Future(Error(f))
- +

Future<Result<Ok, Error>>

A Future can contain a Result (e.g. to represent an asynchronous value that can fail). We provide some utility functions to deal with that case without having to unwrap the Future result.

note

You can still use all the regular Future methods. The following helpers simply removes the need to unwrap the contained result.

Methods

.mapOkToResult(f)

Future<Result<A, E>>.mapOkToResult<B, F>(
func: (value: A) => Result<B, F>,
propagateCancel?: boolean
): Future<Result<B, E | F>>

Takes a Future<Result<Ok, Error>> and a f function taking Ok and returning Result<ReturnValue, Error> and returns a new Future<Result<ReturnValue, Error>>

Examples
Future.value(Result.Ok(3)).mapOkToResult((ok) => {
return Result.Ok(ok * 2);
});
// Future<Result.Ok<6>>

Future.value(Result.Ok(3)).mapOkToResult((ok) =>
isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");
);
// Future<Result.Error<"Odd number">>

.mapErrorToResult(f)

Future<Result<A, E>>.mapErrorToResult<B, F>(
func: (value: E) => Result<B, F>,
propagateCancel?: boolean
): Future<Result<A | B, F>>

Takes a Future<Result<Ok, Error>> and a f function taking Error and returning Result<Ok, ReturnError> and returns a new Future<Result<Ok, ReturnError>>

Examples
Future.value(Result.Error(3)).mapErrorToResult((ok) => {
return Result.Ok(ok * 2);
});
// Future<Result.Ok<6>>

Future.value(Result.Error(3)).mapErrorToResult((ok) =>
isEven(ok) ? Result.Ok(ok) : Result.Error("Odd number");
);
// Future<Result.Error<"Odd number">>

.mapOk(f)

Future<Result<A, E>>.mapOk<B>(
func: (value: A) => B,
propagateCancel?: boolean
): Future<Result<B, E>>

Takes a Future<Result<Ok, Error>> and a f function taking Ok and returning ReturnValue and returns a new Future<Result<ReturnValue, Error>>

Examples
Future.value(Result.Ok(3)).mapOk((ok) => {
return ok * 2;
});
// Future<Result.Ok<6>>

Future.value(Result.Error("something")).mapOk((ok) => {
return ok * 2;
});
// Future<Result.Error<"something">>

.mapError(f)

Future<Result<A, E>>.mapError<F>(
func: (value: E) => F,
propagateCancel?: boolean
): Future<Result<A, F>>

Takes a Future<Result<Ok, Error>> and a f function taking Error and returning ReturnValue and returns a new Future<Result<Ok, ReturnValue>>

Examples
Future.value(Result.Error(3)).mapError((error) => {
return error * 2;
});
// Future<Result.Error<6>>

Future.value(Result.Ok("something")).mapError((ok) => {
return ok * 2;
});
// Future<Result.Ok<"something">>

.flatMapOk(f)

Future<Result<A, E>>.flatMapOk<B, F>(
func: (value: A) => Future<Result<B, F>>,
propagateCancel?: boolean
): Future<Result<B, E | F>>

Takes a Future<Result<Ok, Error>> and a f function taking Ok returning a Future<Result<ReturnValue, Error>>

Examples
Future.value(Result.Ok(3)).flatMapOk((ok) => Future.value(Result.Ok(ok * 2)));
// Future<Result.Ok<6>>

Future.value(Result.Ok(3)).flatMapOk((ok) =>
Future.value(Result.Error("Nope")),
);
// Future<Result.Error<"Nope">>

Future.value(Result.Error("Error")).flatMapOk((ok) =>
Future.value(Result.Ok(ok * 2)),
);
// Future<Result.Error<"Error">>

.flatMapError(f)

Future<Result<A, E>>.flatMapError<B, F>(
func: (value: E) => Future<Result<B, F>>,
propagateCancel?: boolean
): Future<Result<A | B, F>>

Takes a Future<Result<Ok, Error>> and a f function taking Error returning a Future<Result<Ok, ReturnValue>>

Examples
Future.value(Result.Ok(3)).flatMapError((error) =>
Future.value(Result.Ok(ok * 2)),
);
// Future<Result.Ok<3>>

Future.value(Result.Error("Error")).flatMapError((error) =>
Future.value(Result.Error("Nope")),
);
// Future<Result.Error<"Nope">>

Future.value(Result.Error("Error")).flatMapError((error) =>
Future.value(Result.Ok(1)),
);
// Future<Result.Ok<1>>

.tapOk(f)

Future<Result<A, E>>.tapOk(func: (value: A) => unknown): Future<Result<A, E>>

Runs f if value is Ok with the future value, and returns the original future. Useful for debugging.

Examples
future.tapOk(console.log);

.tapError(f)

Future<Result<A, E>>.tapError(func: (value: E) => unknown): Future<Result<A, E>>

Runs f if value is Error with the future value, and returns the original future. Useful for debugging.

Examples
future.tapError(console.log);

.resultToPromise()

Future<Result<A, E>>.resultToPromise(): Promise<A>

Takes a Future<Result<Ok, Error>> and returns a Promise<Ok>, rejecting the promise with Error in this state.

Examples
Future.value(Result.Ok(1)).resultToPromise();
// Promise<1>

Future.value(Result.Reject(1)).resultToPromise();
// Promise (rejected with 1)

Statics

Future.all(resultFutures)

You can combine the all helpers from Future and Result:

Examples
const futures = [
Future.value(Result.Ok(1)),
Future.value(Result.Ok(2)),
Future.value(Result.Ok(3)),
];

Future.all(futures).map(Result.all);
// Future<Result.Ok<[1, 2, 3]>>

Let's see the types at each step:

Examples
// Array<Future<Result<number, never>>>
// -> [Future<Result.Ok<1>>, Future<Result.Ok<2>>, Future<Result.Ok<3>>]
const input = [
Future.value(Result.Ok(1)),
Future.value(Result.Ok(2)),
Future.value(Result.Ok(3)),
];

// Future<Array<Result<number, never>>>
// -> Future<[Result.Ok<1>>, Result.Ok<2>>, Result.Ok<3>]>
const step1 = Future.all(input);

// Future<Result<Array<number>, never>>
// -> Future<[Result.Ok<[1, 2, 3]>>
const step2 = step1.map(Result.all);

Future.allFromDict(resultFutures)

Like as all, you can combine the allFromDict:

Examples
const futures = {
a: Future.value(Result.Ok(1)),
b: Future.value(Result.Ok(2)),
c: Future.value(Result.Ok(3)),
};

Future.allFromDict(futures).map(Result.allFromDict);
// Future<[Result.Ok<{a: 1, b: 2, c: 3}>>

Let's see the types at each step:

Examples
// Dict<Future<Result<number, never>>>
// -> {a: Future<Result.Ok<1>>, b: Future<Result.Ok<2>>, c: Future<Result.Ok<3>>—
const input = {
a: Future.value(Result.Ok(1)),
b: Future.value(Result.Ok(2)),
c: Future.value(Result.Ok(3)),
};

// Future<Dict<Result<number, never>>>
// -> Future<{a: Result.Ok<1>>, b: Result.Ok<2>>, c: Result.Ok<3>}>
const step1 = Future.all(input);

// Future<Result<Array<number>, never>>
// -> Future<[Result.Ok<{a: 1, b: 2, c: 3}>>
const step2 = step1.map(Result.all);

Future.retry(getFuture)

retry(getFuture: () => Future<Result<A, E>>, {max: number}): Future<Result<A, E>>

Runs the future getter, if the future resolves with a Result.Error, retries until hitting max attempts.

The function provides a 0-based attempt count to the function if you need to implement delay logic.

Examples
// retry immediately after failure
Future.retry(
(attempt) => {
return getUserById(userId);
},
{ max: 3 },
);
// Future<Result<...>>

// adding delay
Future.retry(
(attempt) => {
return Future.wait(attempt * 100).flatMap(() => getUserById(userId));
},
{ max: 10 },
);
// Future<Result<...>>

Cheatsheet

MethodInputFunction inputFunction outputReturned value
mapOkToResultFuture(Ok(x))xOk(y)Future(Ok(y))
mapOkToResultFuture(Ok(x))xError(f)Future(Error(f))
mapOkToResultFuture(Error(e))not providednot executedFuture(Error(e))
mapErrorToResultFuture(Error(e))eOk(y)Future(Ok(y))
mapErrorToResultFuture(Error(e))eError(f)Future(Error(f))
mapErrorToResultFuture(Ok(x))not providednot executedFuture(Ok(x))
mapOkFuture(Ok(x))xyFuture(Ok(y))
mapOkFuture(Error(e))not providednot executedFuture(Error(e))
mapErrorFuture(Ok(x))not providednot executedFuture(Ok(x))
mapErrorFuture(Error(e))efFuture(Error(f))
flatMapOkFuture(Ok(x))xFuture(Ok(y))Future(Ok(y))
flatMapOkFuture(Ok(x))xFuture(Error(f))Future(Error(f))
flatMapOkFuture(Error(e))not providednot executedFuture(Error(e))
flatMapErrorFuture(Ok(x))not providednot executedFuture(Ok(x))
flatMapErrorFuture(Error(e))eFuture(Ok(y))Future(Ok(y))
flatMapErrorFuture(Error(e))eFuture(Error(f))Future(Error(f))
+ \ No newline at end of file diff --git a/future/index.html b/future/index.html index 951374a..f51783b 100644 --- a/future/index.html +++ b/future/index.html @@ -5,13 +5,13 @@ Future<Value> | Boxed - +

Future<Value>

The Future is a replacement for Promise.

Main differences with Promises

  • Futures don't handle rejection state, instead leaving it to a contained Result
  • Futures have built-in cancellation (and don't reject like the fetch signal API does)
  • Futures don't "swallow" futures that are returned from map and flatMap
  • Future callbacks run synchronously
info

Even though we're diverging from Promise, you can await a Future.

Create a Future

Examples
import { Future } from "@swan-io/boxed";

// Value
const future = Future.value(1);

// Simple future
const otherFuture = Future.make((resolve) => {
resolve(1);
});

// Future with cancellation effect
const otherFuture = Future.make((resolve) => {
const timeoutId = setTimeout(() => {
resolve(1);
}, 1000);
return () => clearTimeout(timeoutId);
});

Methods

.onResolve(f)

Future<A>.onResolve(func: (value: A) => void): void

Runs f with the future value as argument when available.

Examples
Future.value(1).onResolve(console.log);
// Log: 1

.onCancel(f)

Future<A>.onCancel(func: () => void): void

Runs f when the future is cancelled.

Examples
future.onCancel(() => {
// do something
});

.map(f)

Future<A>.map<B>(func: (value: A) => B, propagateCancel?: boolean): Future<B>

Takes a Future<A> and returns a new Future<f<A>>

Examples
Future.value(3).map((x) => x * 2);
// Future<6>

.flatMap(f)

Future<A>.flatMap<B>(func: (value: A) => Future<B>, propagateCancel?: boolean): Future<B>

Takes a Future<A>, and returns a new future taking the value of the future returned by f(A)

Examples
Future.value(3).flatMap((x) => Future.value(x * 2));
// Future<6>

.tap(f)

Future<A>.tap(func: (value: A) => unknown): Future<A>

Runs f with the future value, and returns the original future. Useful for debugging.

Examples
Future.value(3).tap(console.log);
// Log: 3
// Future<3>

.toPromise()

Future<A>.toPromise(): Promise<A>

Takes a Future<T> and returns a Promise<T>

Examples
Future.value(1).toPromise();
// Promise<1>

Future<Result<Ok, Error>>

We provide some special helpers for Futures containing a Result.

Statics

Future.isFuture(value)

isFuture(value: unknown): boolean

Type guard, checks if the provided value is a future.

Examples
Future.isFuture(Future.value(1));
// true

Future.isFuture([]);
// false

Future.all(futures)

all(futures: Array<Future<A>>): Future<Array<A>>

Turns an "array of futures of values" into a "future of array of value".

Examples
Future.all([Future.value(1), Future.value(2), Future.value(3)]);
// Future<[1, 2, 3]>

Future.concurrent(futureGetters, options)

all(futures: Array<() => Future<A>>, {concurrency: number}): Future<Array<A>>

Like Future.all with a max concurrency, and in order to control the flow, provided with functions returning futures.

Examples
Future.concurrent(
userIds.map((userId) => {
// notice we return a function
return () => getUserById(userId);
}),
{ concurrency: 10 },
);
// Future<[...]>

Future.wait(ms)

wait(ms: number): Future<void>

Helper to create a future that resolves after ms (in milliseconds).

Examples
Future.wait(1000).tap(() => console.log("Hey"));
// Logs "Hey" after 1s

Future.allFromDict(futures)

allFromDict(futures: Dict<Future<A>>): Future<Dict<A>>

Turns a "dict of futures of values" into a "future of dict of value".

Examples
Future.allFromDict({
a: Future.value(1),
b: Future.value(2),
c: Future.value(3),
});
// Future<{a: 1, b: 2, c: 3}>

Future.fromPromise(promise)

fromPromise<A>(promise: Promise<A>): Future<Result<A, unknown>>

Takes a Promise<T> and returns a Future<Result<T, Error>>

Examples
Future.fromPromise(Promise.resolve(1));
// Future<Result.Ok<1>>

Future.fromPromise(Promise.reject(1));
// Future<Result.Error<1>>

Cancellation

Basics

In JavaScript, Promises are not cancellable.

That can be limiting at times, especially when using React's useEffect, that let's you return a cancellation effect in order to prevent unwanted side-effects.

You can return a cleanup effect from the future init function:

const future = Future.make((resolve) => {
const timeoutId = setTimeout(() => {
resolve(1);
}, 1000);
// will run on cancellation
return () => clearTimeout(timeoutId);
});

To cancel a future, call future.cancel().

future.cancel();
note

You can only cancel a pending future.

Calling cancel on a resolved future is a no-op, meaning the future will keep its resolved state.

A cancelled future will automatically cancel any future created from it (e.g. from .map or .flatMap):

const future = Future.make((resolve) => {
const timeoutId = setTimeout(() => {
resolve(1);
}, 1000);
// will run on cancellation
return () => clearTimeout(timeoutId);
});

const future2 = future.map((x) => x * 2);

future.cancel(); // Both `future` and `future2` are cancelled

Bubbling cancellation

All .map* and .flatMap* methods take an extra parameter called propagateCancel, it enables the returned future cancel to bubble up cancellation to its depedencies:

// disabled by default: cancelling `future2` will not cancel `future`
const future2 = future.map((x) => x * 2);

// optin: cancelling `future2` will cancel `future`
const future2 = future.map((x) => x * 2, true);

This can be useful at call site:

const request = apiCall().map(parse, true);

request.cancel(); // will run the cleanup effect in `apiCall`

Cheatsheet

MethodInputFunction inputFunction outputReturned value
mapFuture(x)xyFuture(y)
flatMapFuture(x)xFuture(y)Future(y)
- + \ No newline at end of file diff --git a/getting-started/index.html b/getting-started/index.html index 36bcf43..1c1ee98 100644 --- a/getting-started/index.html +++ b/getting-started/index.html @@ -5,13 +5,13 @@ Getting started | Boxed - +

Getting started

Boxed provides essential building-blocks to solve common issues you can run into in your application or library development.

What does Boxed solve?

Virtually any application has to deal with the following states:

  • optionality (a value being there or not)
  • success (a value that can be computed or fails to be)
  • completion (a value that is available or not)

If we use the default way JavaScript (and by extension TypeScript) provides to handle these, we generally end up with code that's growing more complex over time, and introduce subtle bugs:

A subtle bug

How does Boxed solve these bugs?

Boxed provides useful data-structures designed in way that completely eliminates these issues using properties from maths™ (don't worry, you don't need to know the full theory, you can just enjoy the benefits it provides).

Monads &amp; functors

Now, let's get into the details with the core concepts.

- + \ No newline at end of file diff --git a/index.html b/index.html index 841a12f..474e8e8 100644 --- a/index.html +++ b/index.html @@ -5,13 +5,13 @@ Boxed: Essential building-blocks for functional & safe TypeScript code | Boxed - +
Boxed logo

Boxed

Essential building-blocks for functional & safe TypeScript code

import { AsyncData } from "@swan-io/boxed";

const UserCard = ({user}: {user: AsyncData<User>}) => {
return user.match({
NotAsked: () => null,
Loading: () => `Loading`,
Done: (user) => {
const name = user.name.getOr("anonymous");
return `Hello ${name}!`;
},
});
};

Avoid accidental complexity

Boxed provides functional building blocks that make your code more maintainable, more expressive, and safer.

Focused on DX

We provide a very small API surface. With easy interop, and compatiblity with the ecosystem (like ts-pattern)

Easy to reason about

The concepts exposed by Boxed are simple and accessible: you don't need a CS degree to get started.

Example of a request lifecycle management in a React component using Boxed.AsyncData

Build with the right tools

By using functional type-safe constructs like Option, Result and AsyncData, you can eliminate bugs right from the modeling.

Your code will be simpler, safer,and easier to reason about than with regular null-checks, exception flows and manual value tracking.

All of that for less that 3KBs when gzipped!

Tailored for your IDE

Thanks to our chaining API, you get a nice autocomplete right from the value and can easily name intermediate variables.

Boxed leverages the JavaScript class API so that you don't need to import any module to work with a Boxed value: it's all available as a method. On top of that, the Boxed API is minimal, so that your tooling doesn't feel overwhelming.

Cheatsheet table for the types of the map and flatMap functions

Get productive immediately

Boxed gives you the tools you need without requiring loads of theoretical knowledge.

We provide simple naming, documentation and escape hatches so that you don't get stuck. You get to learn as you use the library instead of getting frustrated over complex concepts.

- + \ No newline at end of file diff --git a/inspirations/index.html b/inspirations/index.html index 8bce7b6..e19a944 100644 --- a/inspirations/index.html +++ b/inspirations/index.html @@ -5,13 +5,13 @@ Inspirations | Boxed - + - + \ No newline at end of file diff --git a/installation/index.html b/installation/index.html index 9d0bf16..f69e501 100644 --- a/installation/index.html +++ b/installation/index.html @@ -5,13 +5,13 @@ Installation | Boxed - +

Installation

Prerequisites

Even though you can use Boxed without TypeScript and still leverage some benefits, we recommend to use it to get all of the benefits Boxed can provide.

Installation

In your console

$ yarn add @swan-io/boxed

or

$ npm install @swan-io/boxed
- + \ No newline at end of file diff --git a/lazy/index.html b/lazy/index.html index 6f7c1e1..b8c394c 100644 --- a/lazy/index.html +++ b/lazy/index.html @@ -5,13 +5,13 @@ Lazy | Boxed - +

Lazy

Lazy(f)

Creates a lazy value. The computation won't happen until the first access.

A lazy type exposes a get method that'll return the result from the computation.

import { Lazy } from "@swan-io/boxed";

const lazy = Lazy(() => {
return myComputation();
});

lazy.get()

Computes the value once and returns it.

lazy.get(); // value is computed and return here
- + \ No newline at end of file diff --git a/nested-optional-values/index.html b/nested-optional-values/index.html index 6aa81c7..1f2a752 100644 --- a/nested-optional-values/index.html +++ b/nested-optional-values/index.html @@ -5,13 +5,13 @@ Nested optional values | Boxed - +

Nested optional values

Managing optionality with undefined and null can lead to tedious code, especially when dealing with default values.

Let's assume that we have the following values in scope:

declare var input: string | undefined;
declare function parseInput(input: string): Array<string>;
declare function transform(input: Array<string>): Array<string> | undefined;
declare function print(input: Array<string>): string;
declare function prettify(input: string): string;

Here, parse always returns an Array<string>, and transform can return either an Array<string> or undefined.

Handling this using null or undefined values would lead to code like the following:

const parsed = input != undefined ? parseInput(input) : undefined;
// Keep the `parsed` value if `transform` doesn't output
const transformed =
parsed != undefined ? transform(parsed) ?? parsed : undefined;
// Fallback at the end
const printed = transformed != undefined ? print(transformed) : undefined;
const value = printed != undefined ? prettify(printed) : "fallback";

We lose a lot of the code intent, as we're distracted with some unnecessary complexity.

Now, let's tweak our values so that we use the Option type instead of undefined:

declare var input: Option<string>;
declare function parseInput(input: string): Array<string>;
declare function transform(input: Array<string>): Option<Array<string>>;
declare function print(input: Array<string>): string;
declare function prettify(input: string): string;

Using Option, the same code as above can be written as follows:

input
.map(parseInput)
.flatMap(transform)
.map(print)
.map(prettify)
.getOr("fallback");

Here, the intent of the code is clearly represented, making it much easier to follow.

If we need quick interop with existing code returning undefined or null values, Boxed provides transformers:

If we were to assume again that we have:

declare var input: string | undefined;
declare function parseInput(input: string): Array<string>;
declare function transform(input: Array<string>): Array<string> | undefined;
declare function print(input: Array<string>): string;
declare function prettify(input: string): string;

We'd simplfy need to write the following:

Option.fromNullable(input)
.map(parseInput)
.flatMap((input) => Option.fromNullable(transform(input)))
.map(print)
.map(prettify)
.getOr("fallback");
- + \ No newline at end of file diff --git a/option/index.html b/option/index.html index 1931604..14597ff 100644 --- a/option/index.html +++ b/option/index.html @@ -5,13 +5,13 @@ Option<Value> | Boxed - +

Option<Value>

The Option type can be used as a replacement for null and undefined when manipulating optional data. Contrary to null and undefined, an option is kind of like a box, that contains a value or not.

It can be useful to distinguish values between each other: you can represent Some(None) with options, whereas undefined or null replace the value they intend to make optional.

An option can have two possible states:

  • Some(value)
  • None

Create an Option value

To create an option, use the Some and None constructors:

import { Option } from "@swan-io/boxed";

const aName = Option.Some("John");
const bName = Option.None();

// You can enforce the type using a type parameter
Option.Some<string>("John");
Option.None<string>();

You get interop with null and undefined:

// `value` being `null` or `undefined` makes a `None`
const a = Option.fromNullable(value);

// `value` being `null` makes a `None`
const b = Option.fromNull(value);

// `value` being `undefined` makes a `None`
const c = Option.fromUndefined(value);

Methods

The option type provides a few manipulation functions:

.map(f)

Option<A>.map<B>(f: (value: A) => B): Option<B>

If the option is Some(value) returns Some(f(value)), otherwise returns None.

Examples
Option.Some(2).map((x) => x * 2);
// Option.Some<4>

Option.None().map((x) => x * 2);
// Option.None

.flatMap(f)

Option<A>.flatMap<B>(f: (value: A) => Option<B>): Option<B>

If the option is Some(value) returns f(value), otherwise returns None.

Examples
Option.Some(3).flatMap((x) => (x > 2 ? Option.None() : Option.Some(2)));
// Option.None

Option.Some(1).flatMap((x) => (x > 2 ? Option.None() : Option.Some(2)));
// Option.Some<2>

option.flatMap((value) => value.optionalProperty);
// Option<optionalProperty>

.filter(f)

Option<A>.filter(f: (value: A) => boolean): Option<A>

If the option is Some(value) and that f(value) is true, returns Some(value), otherwise returns None.

Examples
Option.Some(3).filter((x) => x > 2);
// Option.Some(3)

Option.Some(1).filter((x) => x > 2);
// Option.None

.getOr(defaultValue)

Alias: getWithDefault

Option<A>.getOr(defaultValue: A): A

If the option is Some(value) returns value, otherwise returns defaultValue.

Examples
Option.Some(2).getOr(1);
// 2

Option.None().getOr(1);
// 1

.get()

Option<A>.get(): A

Returns the value contained in Some(value). Only usable within a isSome() check.

Examples
const value = option.get();
// does not compile

if (option.isSome()) {
const value = option.get();
// value
}

.isSome()

Option<A>.isSome(): boolean

Type guard. Checks if the option is Some(value)

Examples
Option.Some(2).isSome();
// true

Option.None().isSome();
// false

if (option.isSome()) {
const value = option.get();
}

.isNone()

Option<A>.isNone(): boolean

Type guard. Checks if the option is None

Examples
Option.Some(2).isNone();
// false

Option.None().isNone();
// true

.toNull()

Option<A>.toNull(): A | null

Returns null if the option is None, returns the value otherwise

Examples
Option.Some(2).toNull();
// 2

Option.None().toNull();
// null

.toUndefined()

Option<A>.toUndefined(): A | undefined

Returns undefined if the option is None, returns the value otherwise

Examples
Option.Some(2).toUndefined();
// 2

Option.None().toUndefined();
// undefined

.toResult(errorWhenNone)

Option<A>.toResult<E>(valueWhenNone: E): Result<A, E>

Returns Ok if the option is Some, returns Error otherwise

Examples
const a = Option.Some(1).toResult("NotFound");
// Ok<1>

const b = Option.None().toResult("NotFound");
// Error<"NotFound">

.match()

Option<A>.match<B>(config: {
Some: (value: A) => B;
None: () => B;
}): B

Match the option state

Examples
const valueToDisplay = option.match({
Some: (value) => value,
None: () => "No value",
});
// value | "No value"

.tap(func)

Option<A>.tap(func: (option: Option<A>) => unknown): Option<A>

Executes func with option, and returns option. Useful for logging and debugging.

Examples
option.tap(console.log).map((x) => x * 2);

.tapSome(func)

Option<A>.tapSome(func: (option: A) => unknown): Option<A>

Executes func with option's value if Some, and returns option. Useful for logging and debugging.

Examples
option.tapSome(console.log).map((x) => x * 2);

Statics

Option.isOption(value)

isOption(value: unknown): boolean

Type guard, checks if the provided value is an option.

Examples
Option.isOption(Option.Some(1));
// true

Option.isOption([]);
// false

Option.all(options)

all(options: Array<Option<A>>): Option<Array<A>>

Turns an "array of options of value" into a "option of array of value".

Examples
Option.all([Option.Some(1), Option.Some(2), Option.Some(3)]);
// Some([1, 2, 3])

Option.all([Option.None(), Option.Some(2), Option.Some(3)]);
// None

Option.allFromDict(options)

allFromDict(options: Dict<Option<A>>): Option<Dict<A>>

Turns a "dict of options of value" into a "option of dict of value".

Examples
Option.allFromDict({ a: Option.Some(1), b: Option.Some(2), c: Option.Some(3) });
// Some({a: 1, b: 2, c: 3})

Option.allFromDict({ a: Option.None(), b: Option.Some(2), c: Option.Some(3) });
// None

TS Pattern interop

Examples
import { match, P } from "ts-pattern";
import { Option } from "@swan-io/boxed";

match(myOption)
.with(Option.P.Some(P.select()), (value) => console.log(value))
.with(Option.P.None, () => "No value")
.exhaustive();

Cheatsheet

MethodInputFunction inputFunction outputReturned value
mapSome(x)xySome(y)
mapNone()not providednot executedNone()
flatMapSome(x)xSome(y)Some(y)
flatMapSome(x)xNone()None()
flatMapNone()not providednot executedNone()
- + \ No newline at end of file diff --git a/react-request/index.html b/react-request/index.html index 5abc484..58c7a86 100644 --- a/react-request/index.html +++ b/react-request/index.html @@ -5,13 +5,13 @@ React Request | Boxed - +

React Request

The AsyncData type removes the need for manual request modeling.

Instead of having to maintain a state like the following, you can store the AsyncData value directly.

type UserQuery = {
isLoading: boolean;
error: Error;
data: User;
};

The problem with this representation is that it can represent impossible states, and require additional work to make it safe. It will also encourage nested conditions, which decreases code readability.

Here's how we can represent this using the AsyncData type.

import { useState, useEffect } from "react";
import { AsyncData } from "@swan-io/boxed";
import { queryUser, User } from "./api";

type Props = {
userId: string;
};

const UserPage = ({ userId }: Props) => {
// Initially, the request hasn't performed
const [user, setUser] = useState(() => AsyncData.NotAsked<User>());

useEffect(() => {
// Indicate that we started loading
setUser(AsyncData.Loading());
const cancel = queryUser({ userId }, (user) => {
// Then, set the received value
setUser(AsyncData.Done(user));
});
return cancel;
}, [userId]);

// We can then match on the value, in a flat way
return user.match({
NotAsked: () => null,
Loading: () => `Loading`,
Done: (user) => `Hello ${user.name}!`,
});
};
- + \ No newline at end of file diff --git a/result/index.html b/result/index.html index f7415dd..28a5d7a 100644 --- a/result/index.html +++ b/result/index.html @@ -5,13 +5,13 @@ Result<Ok, Error> | Boxed - +

Result<Ok, Error>

The Result can replace exception flows.

Exceptions can be tricky to handle: there's nothing in the type system that tracks if an error has been handled, which is error prone, and adds to your mental overhead. Result helps as it makes the value hold the success state, making it dead-simple to track with a type-system.

Just like the Option type, the Result type is a box that can have two states:

  • Ok(value)
  • Error(error)

Create a Result value

To create a result, use the Ok and Error constructors:

import { Result } from "@swan-io/boxed";

const ok = Result.Ok(1);

const notOk = Result.Error("something happened");

You can convert an option to a Result:

import { Result, Option } from "@swan-io/boxed";

const a = Result.fromOption(Option.Some(1), "NotFound");
// Ok<1>

const b = Result.fromOption(Option.None(), "NotFound");
// Error<"NotFound">

You get interop with exceptions and promises:

// Let's say you have some function that throws an error
const init = (id: string) => {
if (id.length !== 24) {
throw new Error();
}
return new Client({ id });
};

const result = Result.fromExecution(() => init(id));
// Here, result will either be:
// - Ok(client)
// - Error(error)

// It works with promises too:

const value = await Result.fromPromise(() => fetch("/api"));
// `value` will either be:
// - Ok(res)
// - Error(error)

Methods

The result type provides a few manipulation functions:

.map(f)

Result<A, E>.map<B>(f: (value: A) => B): Result<B, E>

If the result is Ok(value) returns Ok(f(value)), otherwise returns Error(error).

Examples
Result.Ok(2).map((x) => x * 2);
// Result.Ok<4>

Result.Ok(2).map((x) => Result.Ok(x * 2));
// Result.Ok<Result.Ok<4>>

.mapError(f)

Result<A, E>.mapError<F>(f: (value: E) => F): Result<A, F>

If the result is Error(error) returns Error(f(error)), otherwise returns Ok(value).

Examples
Result.Error(2).mapError((x) => x * 2);
// Result.Error<4>

Result.Error(2).mapError((x) => Result.Ok(x * 2));
// Result.Error<Result.Ok<4>>

.flatMap(f)

Result<A, E>.flatMap<B, F>(f: (value: A) => Result<B, F>): Result<B, F | E>

If the result is Ok(value) returns f(value), otherwise returns Error(error).

Examples
Result.Ok(1).flatMap((x) =>
x > 2 ? Result.Error("some error") : Result.Ok(2),
);
// Result.Ok<2>

Result.Ok(3).flatMap((x) =>
x > 2 ? Result.Error("some error") : Result.Ok(2),
);
// Result.Error<"some error">

Result.Error("initial error").flatMap((x) =>
x > 2 ? Result.Error("some error") : Result.Ok(2),
);
// Result.Error<"initial error">

.flatMapError(f)

Result<A, E>.flatMapError<B, F>(f: (value: E) => Result<B, F>): Result<A | B, F>

If the result is Error(error) returns f(error), otherwise returns Ok(value).

Examples
Result.Error(3).flatMapError((x) =>
x > 2 ? Result.Error("some error") : Result.Ok(2),
);
// Result.Error<"some error">

Result.Error(1).flatMapError((x) =>
x > 2 ? Result.Error("some error") : Result.Ok(2),
);
// Result.Ok<2>

Result.Ok("ok").flatMapError((x) =>
x > 2 ? Result.Error("some error") : Result.Ok(2),
);
// Result.Ok<"ok">

.getOr(defaultValue)

Alias: getWithDefault

Result<A, E>.getOr(defaultValue: A): A

If the result is Ok(value) returns value, otherwise returns defaultValue.

Examples
Result.Ok(2).getOr(1);
// 2

Result.Error(2).getOr(1);
// 1

.get()

Result<A, E>.get(): A

Returns the value contained in Ok(value). Only usable within a isOk() check.

Examples
const value = result.get();
// does not compile

if (result.isOk()) {
const value = result.get();
// value
}

.getError()

Result<A, E>.getError(): E

Returns the error contained in Error(error). Only usable within a isError() check.

Examples
const error = result.getError();
// does not compile

if (result.isError()) {
const error = result.getError();
// error
}

.isOk()

Result<A, E>.isOk(): boolean

Type guard. Checks if the result is Ok(value)

Examples
Result.Ok(2).isOk();
// true

Result.Error(2).isOk();
// false

if (result.isOk()) {
const value = result.get();
}

.isError()

Result<A, E>.isError(): boolean

Type guard. Checks if the result is Error(error)

Examples
Result.Ok(2).isError();
// false

Result.Error().isError();
// true

if (result.isError()) {
const value = result.getError();
}

.toOption()

Result<A, E>.toOption(): Option<A>

If the result is Ok(value) returns Some(value), otherwise returns None.

Examples
Result.Ok(2).toOption();
// Option.Some<2>

Result.Error(2).toOption();
// Option.None

.match()

Result<A, E>.match<B>(config: {
Ok: (value: A) => B;
Error: (error: E) => B;
}): B

Match the result state

Examples
const valueToDisplay = result.match({
Ok: (value) => value,
Error: (error) => {
console.error(error);
return "fallback";
},
});

.tap(func)

Result<A, E>.tap(func: (result: Result<A, E>) => unknown): Result<A, E>

Executes func with result, and returns result. Useful for logging and debugging.

Examples
result.tap(console.log).map((x) => x * 2);

.tapOk(func)

Result<A, E>.tapOk(func: (value: A) => unknown): Result<A, E>

Executes func with ok, and returns result. Useful for logging and debugging. No-op if result is an error.

Examples
result.tapOk(console.log).map((x) => x * 2);

.tapError(func)

Result<A, E>.tapError(func: (error: E) => unknown): Result<A, E>

Executes func with error, and returns result. Useful for logging and debugging. No-op if result is ok.

Examples
result.tapError(console.log).map((x) => x * 2);

Statics

Result.isResult(value)

isResult(value: unknown): boolean

Type guard, checks if the provided value is a result.

Examples
Result.isResult(Result.Ok(1));
// true

Result.isResult([]);
// false

Result.all(results)

all(options: Array<Result<A, E>>): Result<Array<A>, E>

Turns an "array of results of value" into a "result of array of value".

Examples
Result.all([Result.Ok(1), Result.Ok(2), Result.Ok(3)]);
// Result.Ok<[1, 2, 3]>

Result.all([Result.Error("error"), Result.Ok(2), Result.Ok(3)]);
// Result.Error<"error">

Result.allFromDict(results)

allFromDict(options: Dict<Result<A, E>>): Result<Dict<A>, E>

Turns a "dict of results of value" into a "result of dict of value".

Examples
Result.allFromDict({ a: Result.Ok(1), b: Result.Ok(2), c: Result.Ok(3) });
// Result.Ok<{a: 1, b: 2, c: 3}>

Result.allFromDict({
a: Result.Error("error"),
b: Result.Ok(2),
c: Result.Ok(3),
});
// Result.Error<"error">

Result.fromExecution(() => value)

fromExecution<A, E>(func: () => A) => Result<A, E>

Takes a function returning Value that can throw an Error and returns a Result<Value, Error>

Examples
Result.fromExecution(() => 1);
// Result.Ok<1>

Result.fromExecution(() => {
throw "Something went wrong";
});
// Result.Error<"Something went wrong">

Result.fromPromise(promise)

fromPromise<A, E>(promise: Promise<A>) => Promise<Result<A, E>>

Takes a Promise<Value> that can fail with Error and returns a Promise<Result<Value, Error>>

Examples
await Result.fromPromise(Promise.resolve(1));
// Result.Ok<1>

await Result.fromPromise(Promise.reject(1));
// Result.Error<1>

Result.fromOption(option, valueIfNone)

fromOption<A, E>(option: Option<A>, valueWhenNone: E): Result<A, E>

Takes a function returning Value that can throw an Error and returns a Result<Value, Error>

Examples
const a = Result.fromOption(Option.Some(1), "NotFound");
// Result.Ok<1>

const b = Result.fromOption(Option.None(), "NotFound");
// Result.Error<"NotFound">

TS Pattern interop

import { match, P } from "ts-pattern";
import { Result } from "@swan-io/boxed";

match(myResult)
.with(Result.P.Ok(P.select()), (value) => console.log(value))
.with(Result.P.Error(P.select()), (error) => {
console.error(error);
return "fallback";
})
.exhaustive();

Cheatsheet

MethodInputFunction inputFunction outputReturned value
mapOk(x)xyOk(y)
mapError(e)not providednot executedError(e)
mapErrorOk(x)not providednot executedOk(x)
mapErrorError(e)efError(f)
flatMapOk(x)xOk(y)Ok(y)
flatMapOk(x)xError(f)Error(f)
flatMapError(e)not providednot executedError(e)
flatMapErrorOk(x)not providednot executedOk(x)
flatMapErrorError(e)eOk(y)Ok(y)
flatMapErrorError(e)eError(f)Error(f)
- + \ No newline at end of file diff --git a/search/index.html b/search/index.html index 163cbe4..a173721 100644 --- a/search/index.html +++ b/search/index.html @@ -5,13 +5,13 @@ Search the documentation | Boxed - +

Search the documentation

- + \ No newline at end of file diff --git a/serializer/index.html b/serializer/index.html index 905af2c..3a58c8a 100644 --- a/serializer/index.html +++ b/serializer/index.html @@ -5,13 +5,13 @@ Serializer | Boxed - +

Serializer

The serializer enables you to serialize some Boxed values (e.g. to store in LocalStorage, or to hydrate data from SSR).

import { Serializer } from "@swan-io/boxed";

Serializer.encode(value)

Stringifies the input to JSON, managing the AsyncData, Option and Result types properly.

Serializer.encode({
data: AsyncData.Done({
name: Option.None(),
}),
});
// {"data":{"__boxed_type__":"AsyncData","tag":"Done","value":{"name":{"__boxed_type__":"Option","tag":"None"}}}}

Serializer.decode(value)

Parse the JSON input, reviving the AsyncData, Option and Result types properly.

Serializer.decode(`{"__boxed_type__":"Option","tag":"None"}`); // Option.None();
- + \ No newline at end of file