@onrails/result 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (69) hide show
  1. package/DESIGN.md +119 -0
  2. package/LICENSE +21 -0
  3. package/README.md +323 -0
  4. package/RECIPES.md +367 -0
  5. package/dist/async-CCA1yK8q.d.cts +147 -0
  6. package/dist/async-DH_-dNIo.d.ts +147 -0
  7. package/dist/compat/neverthrow.cjs +446 -0
  8. package/dist/compat/neverthrow.cjs.map +1 -0
  9. package/dist/compat/neverthrow.d.cts +77 -0
  10. package/dist/compat/neverthrow.d.ts +77 -0
  11. package/dist/compat/neverthrow.js +435 -0
  12. package/dist/compat/neverthrow.js.map +1 -0
  13. package/dist/extra.cjs +37 -0
  14. package/dist/extra.cjs.map +1 -0
  15. package/dist/extra.d.cts +50 -0
  16. package/dist/extra.d.ts +50 -0
  17. package/dist/extra.js +31 -0
  18. package/dist/extra.js.map +1 -0
  19. package/dist/fluent.cjs +64 -0
  20. package/dist/fluent.cjs.map +1 -0
  21. package/dist/fluent.d.cts +28 -0
  22. package/dist/fluent.d.ts +28 -0
  23. package/dist/fluent.js +61 -0
  24. package/dist/fluent.js.map +1 -0
  25. package/dist/index.cjs +406 -0
  26. package/dist/index.cjs.map +1 -0
  27. package/dist/index.d.cts +227 -0
  28. package/dist/index.d.ts +227 -0
  29. package/dist/index.js +369 -0
  30. package/dist/index.js.map +1 -0
  31. package/dist/interop.cjs +248 -0
  32. package/dist/interop.cjs.map +1 -0
  33. package/dist/interop.d.cts +25 -0
  34. package/dist/interop.d.ts +25 -0
  35. package/dist/interop.js +244 -0
  36. package/dist/interop.js.map +1 -0
  37. package/dist/mcp.cjs +292 -0
  38. package/dist/mcp.cjs.map +1 -0
  39. package/dist/mcp.d.cts +62 -0
  40. package/dist/mcp.d.ts +62 -0
  41. package/dist/mcp.js +284 -0
  42. package/dist/mcp.js.map +1 -0
  43. package/dist/pipe.cjs +16 -0
  44. package/dist/pipe.cjs.map +1 -0
  45. package/dist/pipe.d.cts +26 -0
  46. package/dist/pipe.d.ts +26 -0
  47. package/dist/pipe.js +14 -0
  48. package/dist/pipe.js.map +1 -0
  49. package/dist/railway.cjs +443 -0
  50. package/dist/railway.cjs.map +1 -0
  51. package/dist/railway.d.cts +214 -0
  52. package/dist/railway.d.ts +214 -0
  53. package/dist/railway.js +431 -0
  54. package/dist/railway.js.map +1 -0
  55. package/dist/try-gen.cjs +40 -0
  56. package/dist/try-gen.cjs.map +1 -0
  57. package/dist/try-gen.d.cts +23 -0
  58. package/dist/try-gen.d.ts +23 -0
  59. package/dist/try-gen.js +36 -0
  60. package/dist/try-gen.js.map +1 -0
  61. package/dist/types-C2Dp1d5J.d.cts +21 -0
  62. package/dist/types-C2Dp1d5J.d.ts +21 -0
  63. package/dist/validation.cjs +70 -0
  64. package/dist/validation.cjs.map +1 -0
  65. package/dist/validation.d.cts +72 -0
  66. package/dist/validation.d.ts +72 -0
  67. package/dist/validation.js +65 -0
  68. package/dist/validation.js.map +1 -0
  69. package/package.json +114 -0
@@ -0,0 +1,23 @@
1
+ import { R as Result } from './types-C2Dp1d5J.cjs';
2
+
3
+ /**
4
+ * Unwrap a {@link Result} inside {@link tryGen} (Rust `?` for sync code).
5
+ * Prefer {@link flatMap} / {@link fluent} for long pipelines.
6
+ */
7
+ declare const yieldResult: <T, E>(result: Result<T, E>) => T;
8
+ declare const $: <T, E>(result: Result<T, E>) => T;
9
+ /**
10
+ * Run a block that uses {@link yieldResult}; returns the final `Result` or the first Err.
11
+ *
12
+ * @example
13
+ * ```ts
14
+ * const out = tryGen(() => {
15
+ * const a = $(parseA());
16
+ * const b = $(parseB());
17
+ * return ok(a + b);
18
+ * });
19
+ * ```
20
+ */
21
+ declare const tryGen: <T, E>(fn: () => Result<T, E>) => Result<T, E>;
22
+
23
+ export { $, tryGen, yieldResult };
@@ -0,0 +1,23 @@
1
+ import { R as Result } from './types-C2Dp1d5J.js';
2
+
3
+ /**
4
+ * Unwrap a {@link Result} inside {@link tryGen} (Rust `?` for sync code).
5
+ * Prefer {@link flatMap} / {@link fluent} for long pipelines.
6
+ */
7
+ declare const yieldResult: <T, E>(result: Result<T, E>) => T;
8
+ declare const $: <T, E>(result: Result<T, E>) => T;
9
+ /**
10
+ * Run a block that uses {@link yieldResult}; returns the final `Result` or the first Err.
11
+ *
12
+ * @example
13
+ * ```ts
14
+ * const out = tryGen(() => {
15
+ * const a = $(parseA());
16
+ * const b = $(parseB());
17
+ * return ok(a + b);
18
+ * });
19
+ * ```
20
+ */
21
+ declare const tryGen: <T, E>(fn: () => Result<T, E>) => Result<T, E>;
22
+
23
+ export { $, tryGen, yieldResult };
@@ -0,0 +1,36 @@
1
+ // src/result.ts
2
+ var err = (error) => ({
3
+ _tag: "Err",
4
+ error
5
+ });
6
+ var isErr = (result) => result._tag === "Err";
7
+
8
+ // src/try-gen.ts
9
+ var ErrSignal = class {
10
+ constructor(error) {
11
+ this.error = error;
12
+ }
13
+ error;
14
+ _tag = "ErrSignal";
15
+ };
16
+ var yieldResult = (result) => {
17
+ if (isErr(result)) {
18
+ throw new ErrSignal(result.error);
19
+ }
20
+ return result.value;
21
+ };
22
+ var $ = yieldResult;
23
+ var tryGen = (fn) => {
24
+ try {
25
+ return fn();
26
+ } catch (error) {
27
+ if (error instanceof ErrSignal) {
28
+ return err(error.error);
29
+ }
30
+ throw error;
31
+ }
32
+ };
33
+
34
+ export { $, tryGen, yieldResult };
35
+ //# sourceMappingURL=try-gen.js.map
36
+ //# sourceMappingURL=try-gen.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/result.ts","../src/try-gen.ts"],"names":[],"mappings":";AA2BO,IAAM,GAAA,GAAM,CAAyB,KAAA,MAA4B;AAAA,EACtE,IAAA,EAAM,KAAA;AAAA,EACN;AACF,CAAA,CAAA;AAwBO,IAAM,KAAA,GAAQ,CAAO,MAAA,KAA8C,MAAA,CAAO,IAAA,KAAS,KAAA;;;AClD1F,IAAM,YAAN,MAAmB;AAAA,EAEjB,YAAqB,KAAA,EAAU;AAAV,IAAA,IAAA,CAAA,KAAA,GAAA,KAAA;AAAA,EAAW;AAAA,EAAX,KAAA;AAAA,EADZ,IAAA,GAAO,WAAA;AAElB,CAAA;AAMO,IAAM,WAAA,GAAc,CAAO,MAAA,KAA4B;AAC5D,EAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,IAAA,MAAM,IAAI,SAAA,CAAU,MAAA,CAAO,KAAK,CAAA;AAAA,EAClC;AACA,EAAA,OAAO,MAAA,CAAO,KAAA;AAChB;AAEO,IAAM,CAAA,GAAI;AAcV,IAAM,MAAA,GAAS,CAAO,EAAA,KAAyC;AACpE,EAAA,IAAI;AACF,IAAA,OAAO,EAAA,EAAG;AAAA,EACZ,SAAS,KAAA,EAAO;AACd,IAAA,IAAI,iBAAiB,SAAA,EAAW;AAC9B,MAAA,OAAO,GAAA,CAAI,MAAM,KAAK,CAAA;AAAA,IACxB;AACA,IAAA,MAAM,KAAA;AAAA,EACR;AACF","file":"try-gen.js","sourcesContent":["import type { Err, Ok, Result } from \"./types.js\";\n\nexport type { Err, Ok, Result } from \"./types.js\";\n\n/**\n * Lifts a value into the success track.\n *\n * @example\n * ```ts\n * const r = ok(42); // Result<number, never>\n * const typed: Result<number, \"parse\"> = ok(1);\n * ```\n */\nexport const ok = <T, E = never>(value: T): Result<T, E> => ({\n _tag: \"Ok\",\n value,\n});\n\n/**\n * Lifts a value into the error track.\n *\n * @example\n * ```ts\n * const r = err({ kind: \"parse\", message: \"bad json\" });\n * // Result<never, { kind: \"parse\"; message: string }>\n * ```\n */\nexport const err = <T = never, E = unknown>(error: E): Result<T, E> => ({\n _tag: \"Err\",\n error,\n});\n\n/**\n * Type-narrowing predicate: returns `true` when the result is `Ok`.\n *\n * @example\n * ```ts\n * if (isOk(r)) {\n * console.log(r.value); // narrowed to Ok branch\n * }\n * ```\n */\nexport const isOk = <T, E>(result: Result<T, E>): result is Ok<T, E> => result._tag === \"Ok\";\n\n/**\n * Type-narrowing predicate: returns `true` when the result is `Err`.\n *\n * @example\n * ```ts\n * if (isErr(r)) {\n * metrics.inc(\"error\", { kind: r.error.kind });\n * }\n * ```\n */\nexport const isErr = <T, E>(result: Result<T, E>): result is Err<T, E> => result._tag === \"Err\";\n\n/** Fantasy Land `of` — alias of {@link ok}. */\nexport const of = ok;\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Dual-form helpers — each export accepts either shape:\n// data-first: `map(result, fn)`\n// curried: `map(fn)(result)`\n// Arity at the call site selects the overload.\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst mapImpl = <T, U, E>(result: Result<T, E>, fn: (value: T) => U): Result<U, E> =>\n isOk(result) ? ok(fn(result.value)) : err(result.error);\n\n/**\n * Transform the `Ok` value, passing `Err` through unchanged. Dual-form:\n * call data-first or curried (for use with {@link pipe}).\n *\n * @example\n * ```ts\n * map(ok(2), (n) => n * 3); // Ok 6 — data-first\n * pipe(ok(\"x\"), map((s) => s.length));// Ok 1 — curried\n * ```\n */\nexport function map<T, U, E>(result: Result<T, E>, fn: (value: T) => U): Result<U, E>;\nexport function map<T, U>(fn: (value: T) => U): <E>(result: Result<T, E>) => Result<U, E>;\nexport function map(\n ...args: [Result<unknown, unknown>, (value: unknown) => unknown] | [(value: unknown) => unknown]\n): unknown {\n if (args.length === 2) return mapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => mapImpl(result, fn);\n}\n\nconst mapErrImpl = <T, E, F>(result: Result<T, E>, fn: (error: E) => F): Result<T, F> =>\n isErr(result) ? err(fn(result.error)) : ok(result.value);\n\n/**\n * Transform the `Err` value, passing `Ok` through unchanged. Useful for\n * unifying heterogeneous failure types into one app-level union.\n *\n * @example\n * ```ts\n * type AppError = { kind: \"http\"; status: number } | { kind: \"parse\" };\n * pipe(\n * fetchSync(url), // Result<Body, { status: number }>\n * mapErr((e): AppError => ({ kind: \"http\", status: e.status })),\n * );\n * ```\n */\nexport function mapErr<T, E, F>(result: Result<T, E>, fn: (error: E) => F): Result<T, F>;\nexport function mapErr<E, F>(fn: (error: E) => F): <T>(result: Result<T, E>) => Result<T, F>;\nexport function mapErr(\n ...args: [Result<unknown, unknown>, (error: unknown) => unknown] | [(error: unknown) => unknown]\n): unknown {\n if (args.length === 2) return mapErrImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => mapErrImpl(result, fn);\n}\n\nconst bimapImpl = <T, U, E, F>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): Result<U, F> => (isOk(result) ? ok(onOk(result.value)) : err(onErr(result.error)));\n\n/**\n * Transform both tracks at once — `Ok` via `onOk`, `Err` via `onErr`.\n * Equivalent to `mapErr(onErr)(map(onOk)(result))` but in one pass.\n *\n * @example\n * ```ts\n * bimap(parsed, (cfg) => cfg.name, (e) => ({ kind: \"input\", cause: e }));\n * ```\n */\nexport function bimap<T, U, E, F>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): Result<U, F>;\nexport function bimap<T, U, E, F>(\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): (result: Result<T, E>) => Result<U, F>;\nexport function bimap(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => unknown, (error: unknown) => unknown]\n | [(value: unknown) => unknown, (error: unknown) => unknown]\n): unknown {\n if (args.length === 3) return bimapImpl(args[0], args[1], args[2]);\n const [onOk, onErr] = args;\n return (result: Result<unknown, unknown>) => bimapImpl(result, onOk, onErr);\n}\n\nconst flatMapImpl = <T, U, E, F>(\n result: Result<T, E>,\n fn: (value: T) => Result<U, F>,\n): Result<U, E | F> => (isOk(result) ? fn(result.value) : err(result.error));\n\n/**\n * Canonical bind (Fantasy Land `chain`). Chains a Result-returning step,\n * widening the error union to `E | F`. Short-circuits on `Err`.\n *\n * @example\n * ```ts\n * flatMap(parseInput(raw), (data) =>\n * data.id != null ? ok(data) : err({ kind: \"missing_id\" as const }),\n * );\n * // Result<Data, ParseError | { kind: \"missing_id\" }>\n * ```\n */\nexport function flatMap<T, U, E, F>(\n result: Result<T, E>,\n fn: (value: T) => Result<U, F>,\n): Result<U, E | F>;\nexport function flatMap<T, U, F>(\n fn: (value: T) => Result<U, F>,\n): <E>(result: Result<T, E>) => Result<U, E | F>;\nexport function flatMap(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => Result<unknown, unknown>]\n | [(value: unknown) => Result<unknown, unknown>]\n): unknown {\n if (args.length === 2) return flatMapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => flatMapImpl(result, fn);\n}\n\nconst recoverImpl = <T, E, F>(\n result: Result<T, E>,\n fn: (error: E) => Result<T, F>,\n): Result<T, F> => (isErr(result) ? fn(result.error) : ok(result.value));\n\n/**\n * Error-track bind — runs `fn` only when the result is `Err`, allowing\n * a failed workflow to recover to `Ok` or remap the failure. Mirror of\n * {@link flatMap} on the error channel.\n *\n * @example\n * ```ts\n * recover(networkResult, (e) =>\n * e.kind === \"rate_limit\" ? ok(cachedBody) : err(e),\n * );\n * ```\n */\nexport function recover<T, E, F>(\n result: Result<T, E>,\n fn: (error: E) => Result<T, F>,\n): Result<T, F>;\nexport function recover<T, E, F>(\n fn: (error: E) => Result<T, F>,\n): (result: Result<T, E>) => Result<T, F>;\nexport function recover(\n ...args:\n | [Result<unknown, unknown>, (error: unknown) => Result<unknown, unknown>]\n | [(error: unknown) => Result<unknown, unknown>]\n): unknown {\n if (args.length === 2) return recoverImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => recoverImpl(result, fn);\n}\n\nconst tapImpl = <T, E>(result: Result<T, E>, fn: (value: T) => void): Result<T, E> => {\n if (isOk(result)) fn(result.value);\n return result;\n};\n\n/**\n * Observe the `Ok` value for side effects (logging, metrics) without\n * changing the carried value. Passes `Err` through untouched.\n *\n * @example\n * ```ts\n * pipe(\n * parseConfig(raw),\n * tap((cfg) => log.info({ msg: \"parsed\", name: cfg.name })),\n * flatMap(validate),\n * );\n * ```\n */\nexport function tap<T, E>(result: Result<T, E>, fn: (value: T) => void): Result<T, E>;\nexport function tap<T>(fn: (value: T) => void): <E>(result: Result<T, E>) => Result<T, E>;\nexport function tap(\n ...args: [Result<unknown, unknown>, (value: unknown) => void] | [(value: unknown) => void]\n): unknown {\n if (args.length === 2) return tapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => tapImpl(result, fn);\n}\n\nconst tapErrImpl = <T, E>(result: Result<T, E>, fn: (error: E) => void): Result<T, E> => {\n if (isErr(result)) fn(result.error);\n return result;\n};\n\n/**\n * Observe the `Err` value for side effects (logging, metrics) without\n * changing the carried error. Passes `Ok` through untouched.\n *\n * @example\n * ```ts\n * pipe(\n * loadUser(id),\n * tapErr((e) => metrics.inc(\"user.load.fail\", { kind: e.kind })),\n * );\n * ```\n */\nexport function tapErr<T, E>(result: Result<T, E>, fn: (error: E) => void): Result<T, E>;\nexport function tapErr<E>(fn: (error: E) => void): <T>(result: Result<T, E>) => Result<T, E>;\nexport function tapErr(\n ...args: [Result<unknown, unknown>, (error: unknown) => void] | [(error: unknown) => void]\n): unknown {\n if (args.length === 2) return tapErrImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => tapErrImpl(result, fn);\n}\n\nconst matchImpl = <T, E, U>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): U => (isOk(result) ? onOk(result.value) : onErr(result.error));\n\n/**\n * Terminal collapse — fold both tracks into a single value. Dual-form:\n * 3-args data-first, 2-args curried for {@link pipe}. Returns whatever\n * the handlers return.\n *\n * For files that also import `match` from `ts-pattern`, the\n * collision-free alias {@link matchResult} is identical.\n *\n * @example\n * ```ts\n * const html = match(parsed, (cfg) => render(cfg), (e) => renderError(e));\n * ```\n */\nexport function match<T, E, U>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): U;\nexport function match<T, E, U>(\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): (result: Result<T, E>) => U;\nexport function match(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => unknown, (error: unknown) => unknown]\n | [(value: unknown) => unknown, (error: unknown) => unknown]\n): unknown {\n if (args.length === 3) return matchImpl(args[0], args[1], args[2]);\n const [onOk, onErr] = args;\n return (result: Result<unknown, unknown>) => matchImpl(result, onOk, onErr);\n}\n\n/** Collision-free alias for files that also import `match` from ts-pattern. */\nexport const matchResult = match;\n\n/**\n * Curried collapse with named slots — escape valve when positional `match`\n * order is unclear at the call site (e.g. when both handlers return the same\n * type and a transposed `match(r, onErr, onOk)` would silently compile).\n */\nexport const fold =\n <T, E, U>(handlers: { readonly ok: (value: T) => U; readonly err: (error: E) => U }) =>\n (result: Result<T, E>): U =>\n matchImpl(result, handlers.ok, handlers.err);\n\n/**\n * Returns the `Ok` value, or `defaultValue` when the result is `Err`.\n *\n * @example\n * ```ts\n * unwrapOr(parsedSetting, \"default-value\");\n * ```\n */\nexport const unwrapOr = <T, E>(result: Result<T, E>, defaultValue: T): T =>\n isOk(result) ? result.value : defaultValue;\n\n/** Test/assert helper — throws the original Err value when called on Err. */\nexport function unwrapOk<T, E>(result: Result<T, E>): T {\n if (isErr(result)) throw result.error;\n return result.value;\n}\n\n/** Test/assert helper — throws TypeError when called on Ok. */\nexport function unwrapErr<T, E>(result: Result<T, E>): E {\n if (isOk(result)) throw new TypeError(\"unwrapErr called on Ok\");\n return result.error;\n}\n\n/** Wrap a throwing sync function — neverthrow `Result.fromThrowable` */\nexport function trySync<A extends readonly unknown[], T, E>(\n fn: (...args: A) => T,\n onThrow: (error: unknown) => E,\n): (...args: A) => Result<T, E>;\nexport function trySync<F extends (...args: never) => unknown, E>(\n fn: F,\n onThrow: (error: unknown) => E,\n): (...args: Parameters<F>) => Result<ReturnType<F>, E>;\nexport function trySync(\n fn: (...args: never) => unknown,\n onThrow: (error: unknown) => unknown,\n): (...args: never) => Result<unknown, unknown> {\n return (...args: never) => {\n try {\n return ok(fn(...args));\n } catch (error) {\n return err(onThrow(error));\n }\n };\n}\n\n/** First failure wins; otherwise collects values in order */\nexport const combine = <T, E>(results: readonly Result<T, E>[]): Result<T[], E> => {\n const values: T[] = [];\n for (const result of results) {\n if (isErr(result)) return err(result.error);\n values.push(result.value);\n }\n return ok(values);\n};\n\n/** Tuple-preserving combine (neverthrow-style) */\nexport const combineTuple = <const R extends readonly Result<unknown, unknown>[]>(\n results: R,\n): CombineTuple<R> => {\n const values: unknown[] = [];\n for (const result of results) {\n if (isErr(result)) return err(result.error) as CombineTuple<R>;\n values.push(result.value);\n }\n return ok(values) as CombineTuple<R>;\n};\n\ntype _OkValue<R> = R extends { _tag: \"Ok\"; readonly value: infer T } ? T : never;\ntype _ErrValue<R> = R extends { _tag: \"Err\"; readonly error: infer E } ? E : never;\n\ntype CombineTuple<R extends readonly Result<unknown, unknown>[]> = Result<\n { [K in keyof R]: _OkValue<R[K]> },\n { [K in keyof R]: _ErrValue<R[K]> }[number]\n>;\n\n/**\n * Variadic value-first pipe — threads `value` through up to nine unary fns.\n *\n * ```ts\n * pipe(\n * parseConfig(raw),\n * map((cfg) => cfg.name),\n * flatMap((name) => name ? ok(name) : err({ kind: \"empty\" })),\n * tap(log),\n * );\n * ```\n */\nexport function pipe<A>(value: A): A;\nexport function pipe<A, B>(value: A, ab: (a: A) => B): B;\nexport function pipe<A, B, C>(value: A, ab: (a: A) => B, bc: (b: B) => C): C;\nexport function pipe<A, B, C, D>(value: A, ab: (a: A) => B, bc: (b: B) => C, cd: (c: C) => D): D;\nexport function pipe<A, B, C, D, E>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n): E;\nexport function pipe<A, B, C, D, E, F>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n): F;\nexport function pipe<A, B, C, D, E, F, G>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n): G;\nexport function pipe<A, B, C, D, E, F, G, H>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n gh: (g: G) => H,\n): H;\nexport function pipe<A, B, C, D, E, F, G, H, I>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n gh: (g: G) => H,\n hi: (h: H) => I,\n): I;\nexport function pipe(value: unknown, ...fns: ReadonlyArray<(x: unknown) => unknown>): unknown {\n let acc = value;\n for (const fn of fns) {\n acc = fn(acc);\n }\n return acc;\n}\n","import { err, isErr } from \"./result.js\";\nimport type { Result } from \"./types.js\";\n\n/** Internal control-flow signal — not part of public error types */\nclass ErrSignal<E> {\n readonly _tag = \"ErrSignal\" as const;\n constructor(readonly error: E) {}\n}\n\n/**\n * Unwrap a {@link Result} inside {@link tryGen} (Rust `?` for sync code).\n * Prefer {@link flatMap} / {@link fluent} for long pipelines.\n */\nexport const yieldResult = <T, E>(result: Result<T, E>): T => {\n if (isErr(result)) {\n throw new ErrSignal(result.error);\n }\n return result.value;\n};\n\nexport const $ = yieldResult;\n\n/**\n * Run a block that uses {@link yieldResult}; returns the final `Result` or the first Err.\n *\n * @example\n * ```ts\n * const out = tryGen(() => {\n * const a = $(parseA());\n * const b = $(parseB());\n * return ok(a + b);\n * });\n * ```\n */\nexport const tryGen = <T, E>(fn: () => Result<T, E>): Result<T, E> => {\n try {\n return fn();\n } catch (error) {\n if (error instanceof ErrSignal) {\n return err(error.error);\n }\n throw error;\n }\n};\n"]}
@@ -0,0 +1,21 @@
1
+ /** Tagged success/failure — no classes, tree-shake friendly. */
2
+ type Result<T, E> = {
3
+ readonly _tag: "Ok";
4
+ readonly value: T;
5
+ } | {
6
+ readonly _tag: "Err";
7
+ readonly error: E;
8
+ };
9
+ type Ok<T, E = never> = Extract<Result<T, E>, {
10
+ _tag: "Ok";
11
+ }>;
12
+ type Err<T, E> = Extract<Result<T, E>, {
13
+ _tag: "Err";
14
+ }>;
15
+ /** Thrown defect mapped when {@link fromAsync} has no `onDefect` */
16
+ declare class UnexpectedError extends Error {
17
+ readonly cause?: unknown | undefined;
18
+ constructor(message: string, cause?: unknown | undefined);
19
+ }
20
+
21
+ export { type Err as E, type Ok as O, type Result as R, UnexpectedError as U };
@@ -0,0 +1,21 @@
1
+ /** Tagged success/failure — no classes, tree-shake friendly. */
2
+ type Result<T, E> = {
3
+ readonly _tag: "Ok";
4
+ readonly value: T;
5
+ } | {
6
+ readonly _tag: "Err";
7
+ readonly error: E;
8
+ };
9
+ type Ok<T, E = never> = Extract<Result<T, E>, {
10
+ _tag: "Ok";
11
+ }>;
12
+ type Err<T, E> = Extract<Result<T, E>, {
13
+ _tag: "Err";
14
+ }>;
15
+ /** Thrown defect mapped when {@link fromAsync} has no `onDefect` */
16
+ declare class UnexpectedError extends Error {
17
+ readonly cause?: unknown | undefined;
18
+ constructor(message: string, cause?: unknown | undefined);
19
+ }
20
+
21
+ export { type Err as E, type Ok as O, type Result as R, UnexpectedError as U };
@@ -0,0 +1,70 @@
1
+ 'use strict';
2
+
3
+ // src/result.ts
4
+ var ok = (value) => ({
5
+ _tag: "Ok",
6
+ value
7
+ });
8
+ var err = (error) => ({
9
+ _tag: "Err",
10
+ error
11
+ });
12
+ var isErr = (result) => result._tag === "Err";
13
+
14
+ // src/validation.ts
15
+ var validateAll = (results, combineErrors) => {
16
+ const values = [];
17
+ let accumulated;
18
+ for (const result of results) {
19
+ if (isErr(result)) {
20
+ accumulated = accumulated === void 0 ? result.error : combineErrors(accumulated, result.error);
21
+ } else {
22
+ values.push(result.value);
23
+ }
24
+ }
25
+ return accumulated === void 0 ? ok(values) : err(accumulated);
26
+ };
27
+ var validateTuple = (results, combineErrors) => {
28
+ const values = [];
29
+ let accumulated;
30
+ for (const result of results) {
31
+ if (isErr(result)) {
32
+ const error = result.error;
33
+ accumulated = accumulated === void 0 ? error : combineErrors(accumulated, error);
34
+ } else {
35
+ values.push(result.value);
36
+ }
37
+ }
38
+ return accumulated === void 0 ? ok(values) : err(accumulated);
39
+ };
40
+ var validateAllArray = (results) => {
41
+ const values = [];
42
+ const errors = [];
43
+ for (const result of results) {
44
+ if (isErr(result)) {
45
+ errors.push(result.error);
46
+ } else {
47
+ values.push(result.value);
48
+ }
49
+ }
50
+ return errors.length === 0 ? ok(values) : err(errors);
51
+ };
52
+ var validateTupleArray = (results) => {
53
+ const values = [];
54
+ const errors = [];
55
+ for (const result of results) {
56
+ if (isErr(result)) {
57
+ errors.push(result.error);
58
+ } else {
59
+ values.push(result.value);
60
+ }
61
+ }
62
+ return errors.length === 0 ? ok(values) : err(errors);
63
+ };
64
+
65
+ exports.validateAll = validateAll;
66
+ exports.validateAllArray = validateAllArray;
67
+ exports.validateTuple = validateTuple;
68
+ exports.validateTupleArray = validateTupleArray;
69
+ //# sourceMappingURL=validation.cjs.map
70
+ //# sourceMappingURL=validation.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/result.ts","../src/validation.ts"],"names":[],"mappings":";;;AAaO,IAAM,EAAA,GAAK,CAAe,KAAA,MAA4B;AAAA,EAC3D,IAAA,EAAM,IAAA;AAAA,EACN;AACF,CAAA,CAAA;AAWO,IAAM,GAAA,GAAM,CAAyB,KAAA,MAA4B;AAAA,EACtE,IAAA,EAAM,KAAA;AAAA,EACN;AACF,CAAA,CAAA;AAwBO,IAAM,KAAA,GAAQ,CAAO,MAAA,KAA8C,MAAA,CAAO,IAAA,KAAS,KAAA;;;ACzBnF,IAAM,WAAA,GAAc,CACzB,OAAA,EACA,aAAA,KACmB;AACnB,EAAA,MAAM,SAAc,EAAC;AACrB,EAAA,IAAI,WAAA;AAEJ,EAAA,KAAA,MAAW,UAAU,OAAA,EAAS;AAC5B,IAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,MAAA,WAAA,GACE,gBAAgB,MAAA,GAAY,MAAA,CAAO,QAAQ,aAAA,CAAc,WAAA,EAAa,OAAO,KAAK,CAAA;AAAA,IACtF,CAAA,MAAO;AACL,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B;AAAA,EACF;AAEA,EAAA,OAAO,gBAAgB,MAAA,GAAY,EAAA,CAAG,MAAM,CAAA,GAAI,IAAI,WAAW,CAAA;AACjE;AAeO,IAAM,aAAA,GAAgB,CAC3B,OAAA,EACA,aAAA,KACwB;AACxB,EAAA,MAAM,SAAoB,EAAC;AAC3B,EAAA,IAAI,WAAA;AAEJ,EAAA,KAAA,MAAW,UAAU,OAAA,EAAS;AAC5B,IAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,MAAA,MAAM,QAAQ,MAAA,CAAO,KAAA;AACrB,MAAA,WAAA,GAAc,WAAA,KAAgB,MAAA,GAAY,KAAA,GAAQ,aAAA,CAAc,aAAa,KAAK,CAAA;AAAA,IACpF,CAAA,MAAO;AACL,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B;AAAA,EACF;AAEA,EAAA,OAAQ,gBAAgB,MAAA,GAAY,EAAA,CAAG,MAAM,CAAA,GAAI,IAAI,WAAW,CAAA;AAClE;AAaO,IAAM,gBAAA,GAAmB,CAC9B,OAAA,KAC8B;AAC9B,EAAA,MAAM,SAAc,EAAC;AACrB,EAAA,MAAM,SAAc,EAAC;AAErB,EAAA,KAAA,MAAW,UAAU,OAAA,EAAS;AAC5B,IAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B,CAAA,MAAO;AACL,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B;AAAA,EACF;AAEA,EAAA,OAAO,OAAO,MAAA,KAAW,CAAA,GAAI,GAAG,MAAM,CAAA,GAAI,IAAI,MAAM,CAAA;AACtD;AAaO,IAAM,kBAAA,GAAqB,CAChC,OAAA,KACgD;AAChD,EAAA,MAAM,SAAoB,EAAC;AAC3B,EAAA,MAAM,SAA2B,EAAC;AAElC,EAAA,KAAA,MAAW,UAAU,OAAA,EAAS;AAC5B,IAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAuB,CAAA;AAAA,IAC5C,CAAA,MAAO;AACL,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B;AAAA,EACF;AAEA,EAAA,OAAQ,OAAO,MAAA,KAAW,CAAA,GAAI,GAAG,MAAM,CAAA,GAAI,IAAI,MAAM,CAAA;AAIvD","file":"validation.cjs","sourcesContent":["import type { Err, Ok, Result } from \"./types.js\";\n\nexport type { Err, Ok, Result } from \"./types.js\";\n\n/**\n * Lifts a value into the success track.\n *\n * @example\n * ```ts\n * const r = ok(42); // Result<number, never>\n * const typed: Result<number, \"parse\"> = ok(1);\n * ```\n */\nexport const ok = <T, E = never>(value: T): Result<T, E> => ({\n _tag: \"Ok\",\n value,\n});\n\n/**\n * Lifts a value into the error track.\n *\n * @example\n * ```ts\n * const r = err({ kind: \"parse\", message: \"bad json\" });\n * // Result<never, { kind: \"parse\"; message: string }>\n * ```\n */\nexport const err = <T = never, E = unknown>(error: E): Result<T, E> => ({\n _tag: \"Err\",\n error,\n});\n\n/**\n * Type-narrowing predicate: returns `true` when the result is `Ok`.\n *\n * @example\n * ```ts\n * if (isOk(r)) {\n * console.log(r.value); // narrowed to Ok branch\n * }\n * ```\n */\nexport const isOk = <T, E>(result: Result<T, E>): result is Ok<T, E> => result._tag === \"Ok\";\n\n/**\n * Type-narrowing predicate: returns `true` when the result is `Err`.\n *\n * @example\n * ```ts\n * if (isErr(r)) {\n * metrics.inc(\"error\", { kind: r.error.kind });\n * }\n * ```\n */\nexport const isErr = <T, E>(result: Result<T, E>): result is Err<T, E> => result._tag === \"Err\";\n\n/** Fantasy Land `of` — alias of {@link ok}. */\nexport const of = ok;\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Dual-form helpers — each export accepts either shape:\n// data-first: `map(result, fn)`\n// curried: `map(fn)(result)`\n// Arity at the call site selects the overload.\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst mapImpl = <T, U, E>(result: Result<T, E>, fn: (value: T) => U): Result<U, E> =>\n isOk(result) ? ok(fn(result.value)) : err(result.error);\n\n/**\n * Transform the `Ok` value, passing `Err` through unchanged. Dual-form:\n * call data-first or curried (for use with {@link pipe}).\n *\n * @example\n * ```ts\n * map(ok(2), (n) => n * 3); // Ok 6 — data-first\n * pipe(ok(\"x\"), map((s) => s.length));// Ok 1 — curried\n * ```\n */\nexport function map<T, U, E>(result: Result<T, E>, fn: (value: T) => U): Result<U, E>;\nexport function map<T, U>(fn: (value: T) => U): <E>(result: Result<T, E>) => Result<U, E>;\nexport function map(\n ...args: [Result<unknown, unknown>, (value: unknown) => unknown] | [(value: unknown) => unknown]\n): unknown {\n if (args.length === 2) return mapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => mapImpl(result, fn);\n}\n\nconst mapErrImpl = <T, E, F>(result: Result<T, E>, fn: (error: E) => F): Result<T, F> =>\n isErr(result) ? err(fn(result.error)) : ok(result.value);\n\n/**\n * Transform the `Err` value, passing `Ok` through unchanged. Useful for\n * unifying heterogeneous failure types into one app-level union.\n *\n * @example\n * ```ts\n * type AppError = { kind: \"http\"; status: number } | { kind: \"parse\" };\n * pipe(\n * fetchSync(url), // Result<Body, { status: number }>\n * mapErr((e): AppError => ({ kind: \"http\", status: e.status })),\n * );\n * ```\n */\nexport function mapErr<T, E, F>(result: Result<T, E>, fn: (error: E) => F): Result<T, F>;\nexport function mapErr<E, F>(fn: (error: E) => F): <T>(result: Result<T, E>) => Result<T, F>;\nexport function mapErr(\n ...args: [Result<unknown, unknown>, (error: unknown) => unknown] | [(error: unknown) => unknown]\n): unknown {\n if (args.length === 2) return mapErrImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => mapErrImpl(result, fn);\n}\n\nconst bimapImpl = <T, U, E, F>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): Result<U, F> => (isOk(result) ? ok(onOk(result.value)) : err(onErr(result.error)));\n\n/**\n * Transform both tracks at once — `Ok` via `onOk`, `Err` via `onErr`.\n * Equivalent to `mapErr(onErr)(map(onOk)(result))` but in one pass.\n *\n * @example\n * ```ts\n * bimap(parsed, (cfg) => cfg.name, (e) => ({ kind: \"input\", cause: e }));\n * ```\n */\nexport function bimap<T, U, E, F>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): Result<U, F>;\nexport function bimap<T, U, E, F>(\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): (result: Result<T, E>) => Result<U, F>;\nexport function bimap(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => unknown, (error: unknown) => unknown]\n | [(value: unknown) => unknown, (error: unknown) => unknown]\n): unknown {\n if (args.length === 3) return bimapImpl(args[0], args[1], args[2]);\n const [onOk, onErr] = args;\n return (result: Result<unknown, unknown>) => bimapImpl(result, onOk, onErr);\n}\n\nconst flatMapImpl = <T, U, E, F>(\n result: Result<T, E>,\n fn: (value: T) => Result<U, F>,\n): Result<U, E | F> => (isOk(result) ? fn(result.value) : err(result.error));\n\n/**\n * Canonical bind (Fantasy Land `chain`). Chains a Result-returning step,\n * widening the error union to `E | F`. Short-circuits on `Err`.\n *\n * @example\n * ```ts\n * flatMap(parseInput(raw), (data) =>\n * data.id != null ? ok(data) : err({ kind: \"missing_id\" as const }),\n * );\n * // Result<Data, ParseError | { kind: \"missing_id\" }>\n * ```\n */\nexport function flatMap<T, U, E, F>(\n result: Result<T, E>,\n fn: (value: T) => Result<U, F>,\n): Result<U, E | F>;\nexport function flatMap<T, U, F>(\n fn: (value: T) => Result<U, F>,\n): <E>(result: Result<T, E>) => Result<U, E | F>;\nexport function flatMap(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => Result<unknown, unknown>]\n | [(value: unknown) => Result<unknown, unknown>]\n): unknown {\n if (args.length === 2) return flatMapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => flatMapImpl(result, fn);\n}\n\nconst recoverImpl = <T, E, F>(\n result: Result<T, E>,\n fn: (error: E) => Result<T, F>,\n): Result<T, F> => (isErr(result) ? fn(result.error) : ok(result.value));\n\n/**\n * Error-track bind — runs `fn` only when the result is `Err`, allowing\n * a failed workflow to recover to `Ok` or remap the failure. Mirror of\n * {@link flatMap} on the error channel.\n *\n * @example\n * ```ts\n * recover(networkResult, (e) =>\n * e.kind === \"rate_limit\" ? ok(cachedBody) : err(e),\n * );\n * ```\n */\nexport function recover<T, E, F>(\n result: Result<T, E>,\n fn: (error: E) => Result<T, F>,\n): Result<T, F>;\nexport function recover<T, E, F>(\n fn: (error: E) => Result<T, F>,\n): (result: Result<T, E>) => Result<T, F>;\nexport function recover(\n ...args:\n | [Result<unknown, unknown>, (error: unknown) => Result<unknown, unknown>]\n | [(error: unknown) => Result<unknown, unknown>]\n): unknown {\n if (args.length === 2) return recoverImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => recoverImpl(result, fn);\n}\n\nconst tapImpl = <T, E>(result: Result<T, E>, fn: (value: T) => void): Result<T, E> => {\n if (isOk(result)) fn(result.value);\n return result;\n};\n\n/**\n * Observe the `Ok` value for side effects (logging, metrics) without\n * changing the carried value. Passes `Err` through untouched.\n *\n * @example\n * ```ts\n * pipe(\n * parseConfig(raw),\n * tap((cfg) => log.info({ msg: \"parsed\", name: cfg.name })),\n * flatMap(validate),\n * );\n * ```\n */\nexport function tap<T, E>(result: Result<T, E>, fn: (value: T) => void): Result<T, E>;\nexport function tap<T>(fn: (value: T) => void): <E>(result: Result<T, E>) => Result<T, E>;\nexport function tap(\n ...args: [Result<unknown, unknown>, (value: unknown) => void] | [(value: unknown) => void]\n): unknown {\n if (args.length === 2) return tapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => tapImpl(result, fn);\n}\n\nconst tapErrImpl = <T, E>(result: Result<T, E>, fn: (error: E) => void): Result<T, E> => {\n if (isErr(result)) fn(result.error);\n return result;\n};\n\n/**\n * Observe the `Err` value for side effects (logging, metrics) without\n * changing the carried error. Passes `Ok` through untouched.\n *\n * @example\n * ```ts\n * pipe(\n * loadUser(id),\n * tapErr((e) => metrics.inc(\"user.load.fail\", { kind: e.kind })),\n * );\n * ```\n */\nexport function tapErr<T, E>(result: Result<T, E>, fn: (error: E) => void): Result<T, E>;\nexport function tapErr<E>(fn: (error: E) => void): <T>(result: Result<T, E>) => Result<T, E>;\nexport function tapErr(\n ...args: [Result<unknown, unknown>, (error: unknown) => void] | [(error: unknown) => void]\n): unknown {\n if (args.length === 2) return tapErrImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => tapErrImpl(result, fn);\n}\n\nconst matchImpl = <T, E, U>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): U => (isOk(result) ? onOk(result.value) : onErr(result.error));\n\n/**\n * Terminal collapse — fold both tracks into a single value. Dual-form:\n * 3-args data-first, 2-args curried for {@link pipe}. Returns whatever\n * the handlers return.\n *\n * For files that also import `match` from `ts-pattern`, the\n * collision-free alias {@link matchResult} is identical.\n *\n * @example\n * ```ts\n * const html = match(parsed, (cfg) => render(cfg), (e) => renderError(e));\n * ```\n */\nexport function match<T, E, U>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): U;\nexport function match<T, E, U>(\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): (result: Result<T, E>) => U;\nexport function match(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => unknown, (error: unknown) => unknown]\n | [(value: unknown) => unknown, (error: unknown) => unknown]\n): unknown {\n if (args.length === 3) return matchImpl(args[0], args[1], args[2]);\n const [onOk, onErr] = args;\n return (result: Result<unknown, unknown>) => matchImpl(result, onOk, onErr);\n}\n\n/** Collision-free alias for files that also import `match` from ts-pattern. */\nexport const matchResult = match;\n\n/**\n * Curried collapse with named slots — escape valve when positional `match`\n * order is unclear at the call site (e.g. when both handlers return the same\n * type and a transposed `match(r, onErr, onOk)` would silently compile).\n */\nexport const fold =\n <T, E, U>(handlers: { readonly ok: (value: T) => U; readonly err: (error: E) => U }) =>\n (result: Result<T, E>): U =>\n matchImpl(result, handlers.ok, handlers.err);\n\n/**\n * Returns the `Ok` value, or `defaultValue` when the result is `Err`.\n *\n * @example\n * ```ts\n * unwrapOr(parsedSetting, \"default-value\");\n * ```\n */\nexport const unwrapOr = <T, E>(result: Result<T, E>, defaultValue: T): T =>\n isOk(result) ? result.value : defaultValue;\n\n/** Test/assert helper — throws the original Err value when called on Err. */\nexport function unwrapOk<T, E>(result: Result<T, E>): T {\n if (isErr(result)) throw result.error;\n return result.value;\n}\n\n/** Test/assert helper — throws TypeError when called on Ok. */\nexport function unwrapErr<T, E>(result: Result<T, E>): E {\n if (isOk(result)) throw new TypeError(\"unwrapErr called on Ok\");\n return result.error;\n}\n\n/** Wrap a throwing sync function — neverthrow `Result.fromThrowable` */\nexport function trySync<A extends readonly unknown[], T, E>(\n fn: (...args: A) => T,\n onThrow: (error: unknown) => E,\n): (...args: A) => Result<T, E>;\nexport function trySync<F extends (...args: never) => unknown, E>(\n fn: F,\n onThrow: (error: unknown) => E,\n): (...args: Parameters<F>) => Result<ReturnType<F>, E>;\nexport function trySync(\n fn: (...args: never) => unknown,\n onThrow: (error: unknown) => unknown,\n): (...args: never) => Result<unknown, unknown> {\n return (...args: never) => {\n try {\n return ok(fn(...args));\n } catch (error) {\n return err(onThrow(error));\n }\n };\n}\n\n/** First failure wins; otherwise collects values in order */\nexport const combine = <T, E>(results: readonly Result<T, E>[]): Result<T[], E> => {\n const values: T[] = [];\n for (const result of results) {\n if (isErr(result)) return err(result.error);\n values.push(result.value);\n }\n return ok(values);\n};\n\n/** Tuple-preserving combine (neverthrow-style) */\nexport const combineTuple = <const R extends readonly Result<unknown, unknown>[]>(\n results: R,\n): CombineTuple<R> => {\n const values: unknown[] = [];\n for (const result of results) {\n if (isErr(result)) return err(result.error) as CombineTuple<R>;\n values.push(result.value);\n }\n return ok(values) as CombineTuple<R>;\n};\n\ntype _OkValue<R> = R extends { _tag: \"Ok\"; readonly value: infer T } ? T : never;\ntype _ErrValue<R> = R extends { _tag: \"Err\"; readonly error: infer E } ? E : never;\n\ntype CombineTuple<R extends readonly Result<unknown, unknown>[]> = Result<\n { [K in keyof R]: _OkValue<R[K]> },\n { [K in keyof R]: _ErrValue<R[K]> }[number]\n>;\n\n/**\n * Variadic value-first pipe — threads `value` through up to nine unary fns.\n *\n * ```ts\n * pipe(\n * parseConfig(raw),\n * map((cfg) => cfg.name),\n * flatMap((name) => name ? ok(name) : err({ kind: \"empty\" })),\n * tap(log),\n * );\n * ```\n */\nexport function pipe<A>(value: A): A;\nexport function pipe<A, B>(value: A, ab: (a: A) => B): B;\nexport function pipe<A, B, C>(value: A, ab: (a: A) => B, bc: (b: B) => C): C;\nexport function pipe<A, B, C, D>(value: A, ab: (a: A) => B, bc: (b: B) => C, cd: (c: C) => D): D;\nexport function pipe<A, B, C, D, E>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n): E;\nexport function pipe<A, B, C, D, E, F>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n): F;\nexport function pipe<A, B, C, D, E, F, G>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n): G;\nexport function pipe<A, B, C, D, E, F, G, H>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n gh: (g: G) => H,\n): H;\nexport function pipe<A, B, C, D, E, F, G, H, I>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n gh: (g: G) => H,\n hi: (h: H) => I,\n): I;\nexport function pipe(value: unknown, ...fns: ReadonlyArray<(x: unknown) => unknown>): unknown {\n let acc = value;\n for (const fn of fns) {\n acc = fn(acc);\n }\n return acc;\n}\n","import { err, isErr, ok } from \"./result.js\";\nimport type { Result } from \"./types.js\";\n\ntype OkValue<R> = R extends { readonly _tag: \"Ok\"; readonly value: infer T } ? T : never;\ntype ErrValue<R> = R extends { readonly _tag: \"Err\"; readonly error: infer E } ? E : never;\n\ntype ValidateTuple<R extends readonly Result<unknown, unknown>[], E> = Result<\n { [K in keyof R]: OkValue<R[K]> },\n E\n>;\ntype TupleErrors<R extends readonly Result<unknown, unknown>[]> = ErrValue<R[number]>;\n\n/**\n * Accumulate independent validation failures into a single combined error\n * via `combineErrors`. Returns `Ok<T[]>` only when every input is `Ok`;\n * otherwise returns `Err<E>` where `E` is the fold of all failures.\n *\n * Unlike `combine`, this does **not** short-circuit on first failure — use\n * for independent checks where you want to report all problems at once.\n *\n * @example\n * ```ts\n * validateAll(\n * [validateName(input), validateAge(input), validateEmail(input)],\n * (left, right) => [...left, ...right],\n * );\n * // Result<string[], FieldError[]>\n * ```\n */\nexport const validateAll = <T, E>(\n results: readonly Result<T, E>[],\n combineErrors: (left: E, right: E) => E,\n): Result<T[], E> => {\n const values: T[] = [];\n let accumulated: E | undefined;\n\n for (const result of results) {\n if (isErr(result)) {\n accumulated =\n accumulated === undefined ? result.error : combineErrors(accumulated, result.error);\n } else {\n values.push(result.value);\n }\n }\n\n return accumulated === undefined ? ok(values) : err(accumulated);\n};\n\n/**\n * Tuple-preserving variant of {@link validateAll}. All inputs must share\n * the same error type `E`. Output Ok preserves tuple positions.\n *\n * @example\n * ```ts\n * validateTuple(\n * [validateName(s), validateAge(s)] as const,\n * (l, r) => [...l, ...r],\n * );\n * // Result<readonly [string, number], FieldError[]>\n * ```\n */\nexport const validateTuple = <E, const R extends readonly Result<unknown, E>[]>(\n results: R,\n combineErrors: (left: E, right: E) => E,\n): ValidateTuple<R, E> => {\n const values: unknown[] = [];\n let accumulated: E | undefined;\n\n for (const result of results) {\n if (isErr(result)) {\n const error = result.error;\n accumulated = accumulated === undefined ? error : combineErrors(accumulated, error);\n } else {\n values.push(result.value);\n }\n }\n\n return (accumulated === undefined ? ok(values) : err(accumulated)) as ValidateTuple<R, E>;\n};\n\n/**\n * Like {@link validateAll}, but collects every failure into a readonly\n * array instead of folding via a combine function. The default choice\n * when you want \"all errors\" without an explicit join.\n *\n * @example\n * ```ts\n * validateAllArray([validateName(input), validateAge(input)]);\n * // Result<readonly Field[], readonly FieldError[]>\n * ```\n */\nexport const validateAllArray = <T, E>(\n results: readonly Result<T, E>[],\n): Result<T[], readonly E[]> => {\n const values: T[] = [];\n const errors: E[] = [];\n\n for (const result of results) {\n if (isErr(result)) {\n errors.push(result.error);\n } else {\n values.push(result.value);\n }\n }\n\n return errors.length === 0 ? ok(values) : err(errors);\n};\n\n/**\n * Tuple-preserving variant of {@link validateAllArray}. Heterogeneous\n * input tuple → preserved Ok tuple shape with a union of all input\n * error types collected into a readonly array.\n *\n * @example\n * ```ts\n * validateTupleArray([validateName(s), validateAge(s)] as const);\n * // Result<readonly [string, number], readonly FieldError[]>\n * ```\n */\nexport const validateTupleArray = <const R extends readonly Result<unknown, unknown>[]>(\n results: R,\n): ValidateTuple<R, readonly TupleErrors<R>[]> => {\n const values: unknown[] = [];\n const errors: TupleErrors<R>[] = [];\n\n for (const result of results) {\n if (isErr(result)) {\n errors.push(result.error as TupleErrors<R>);\n } else {\n values.push(result.value);\n }\n }\n\n return (errors.length === 0 ? ok(values) : err(errors)) as ValidateTuple<\n R,\n readonly TupleErrors<R>[]\n >;\n};\n"]}
@@ -0,0 +1,72 @@
1
+ import { R as Result } from './types-C2Dp1d5J.cjs';
2
+
3
+ type OkValue<R> = R extends {
4
+ readonly _tag: "Ok";
5
+ readonly value: infer T;
6
+ } ? T : never;
7
+ type ErrValue<R> = R extends {
8
+ readonly _tag: "Err";
9
+ readonly error: infer E;
10
+ } ? E : never;
11
+ type ValidateTuple<R extends readonly Result<unknown, unknown>[], E> = Result<{
12
+ [K in keyof R]: OkValue<R[K]>;
13
+ }, E>;
14
+ type TupleErrors<R extends readonly Result<unknown, unknown>[]> = ErrValue<R[number]>;
15
+ /**
16
+ * Accumulate independent validation failures into a single combined error
17
+ * via `combineErrors`. Returns `Ok<T[]>` only when every input is `Ok`;
18
+ * otherwise returns `Err<E>` where `E` is the fold of all failures.
19
+ *
20
+ * Unlike `combine`, this does **not** short-circuit on first failure — use
21
+ * for independent checks where you want to report all problems at once.
22
+ *
23
+ * @example
24
+ * ```ts
25
+ * validateAll(
26
+ * [validateName(input), validateAge(input), validateEmail(input)],
27
+ * (left, right) => [...left, ...right],
28
+ * );
29
+ * // Result<string[], FieldError[]>
30
+ * ```
31
+ */
32
+ declare const validateAll: <T, E>(results: readonly Result<T, E>[], combineErrors: (left: E, right: E) => E) => Result<T[], E>;
33
+ /**
34
+ * Tuple-preserving variant of {@link validateAll}. All inputs must share
35
+ * the same error type `E`. Output Ok preserves tuple positions.
36
+ *
37
+ * @example
38
+ * ```ts
39
+ * validateTuple(
40
+ * [validateName(s), validateAge(s)] as const,
41
+ * (l, r) => [...l, ...r],
42
+ * );
43
+ * // Result<readonly [string, number], FieldError[]>
44
+ * ```
45
+ */
46
+ declare const validateTuple: <E, const R extends readonly Result<unknown, E>[]>(results: R, combineErrors: (left: E, right: E) => E) => ValidateTuple<R, E>;
47
+ /**
48
+ * Like {@link validateAll}, but collects every failure into a readonly
49
+ * array instead of folding via a combine function. The default choice
50
+ * when you want "all errors" without an explicit join.
51
+ *
52
+ * @example
53
+ * ```ts
54
+ * validateAllArray([validateName(input), validateAge(input)]);
55
+ * // Result<readonly Field[], readonly FieldError[]>
56
+ * ```
57
+ */
58
+ declare const validateAllArray: <T, E>(results: readonly Result<T, E>[]) => Result<T[], readonly E[]>;
59
+ /**
60
+ * Tuple-preserving variant of {@link validateAllArray}. Heterogeneous
61
+ * input tuple → preserved Ok tuple shape with a union of all input
62
+ * error types collected into a readonly array.
63
+ *
64
+ * @example
65
+ * ```ts
66
+ * validateTupleArray([validateName(s), validateAge(s)] as const);
67
+ * // Result<readonly [string, number], readonly FieldError[]>
68
+ * ```
69
+ */
70
+ declare const validateTupleArray: <const R extends readonly Result<unknown, unknown>[]>(results: R) => ValidateTuple<R, readonly TupleErrors<R>[]>;
71
+
72
+ export { validateAll, validateAllArray, validateTuple, validateTupleArray };
@@ -0,0 +1,72 @@
1
+ import { R as Result } from './types-C2Dp1d5J.js';
2
+
3
+ type OkValue<R> = R extends {
4
+ readonly _tag: "Ok";
5
+ readonly value: infer T;
6
+ } ? T : never;
7
+ type ErrValue<R> = R extends {
8
+ readonly _tag: "Err";
9
+ readonly error: infer E;
10
+ } ? E : never;
11
+ type ValidateTuple<R extends readonly Result<unknown, unknown>[], E> = Result<{
12
+ [K in keyof R]: OkValue<R[K]>;
13
+ }, E>;
14
+ type TupleErrors<R extends readonly Result<unknown, unknown>[]> = ErrValue<R[number]>;
15
+ /**
16
+ * Accumulate independent validation failures into a single combined error
17
+ * via `combineErrors`. Returns `Ok<T[]>` only when every input is `Ok`;
18
+ * otherwise returns `Err<E>` where `E` is the fold of all failures.
19
+ *
20
+ * Unlike `combine`, this does **not** short-circuit on first failure — use
21
+ * for independent checks where you want to report all problems at once.
22
+ *
23
+ * @example
24
+ * ```ts
25
+ * validateAll(
26
+ * [validateName(input), validateAge(input), validateEmail(input)],
27
+ * (left, right) => [...left, ...right],
28
+ * );
29
+ * // Result<string[], FieldError[]>
30
+ * ```
31
+ */
32
+ declare const validateAll: <T, E>(results: readonly Result<T, E>[], combineErrors: (left: E, right: E) => E) => Result<T[], E>;
33
+ /**
34
+ * Tuple-preserving variant of {@link validateAll}. All inputs must share
35
+ * the same error type `E`. Output Ok preserves tuple positions.
36
+ *
37
+ * @example
38
+ * ```ts
39
+ * validateTuple(
40
+ * [validateName(s), validateAge(s)] as const,
41
+ * (l, r) => [...l, ...r],
42
+ * );
43
+ * // Result<readonly [string, number], FieldError[]>
44
+ * ```
45
+ */
46
+ declare const validateTuple: <E, const R extends readonly Result<unknown, E>[]>(results: R, combineErrors: (left: E, right: E) => E) => ValidateTuple<R, E>;
47
+ /**
48
+ * Like {@link validateAll}, but collects every failure into a readonly
49
+ * array instead of folding via a combine function. The default choice
50
+ * when you want "all errors" without an explicit join.
51
+ *
52
+ * @example
53
+ * ```ts
54
+ * validateAllArray([validateName(input), validateAge(input)]);
55
+ * // Result<readonly Field[], readonly FieldError[]>
56
+ * ```
57
+ */
58
+ declare const validateAllArray: <T, E>(results: readonly Result<T, E>[]) => Result<T[], readonly E[]>;
59
+ /**
60
+ * Tuple-preserving variant of {@link validateAllArray}. Heterogeneous
61
+ * input tuple → preserved Ok tuple shape with a union of all input
62
+ * error types collected into a readonly array.
63
+ *
64
+ * @example
65
+ * ```ts
66
+ * validateTupleArray([validateName(s), validateAge(s)] as const);
67
+ * // Result<readonly [string, number], readonly FieldError[]>
68
+ * ```
69
+ */
70
+ declare const validateTupleArray: <const R extends readonly Result<unknown, unknown>[]>(results: R) => ValidateTuple<R, readonly TupleErrors<R>[]>;
71
+
72
+ export { validateAll, validateAllArray, validateTuple, validateTupleArray };
@@ -0,0 +1,65 @@
1
+ // src/result.ts
2
+ var ok = (value) => ({
3
+ _tag: "Ok",
4
+ value
5
+ });
6
+ var err = (error) => ({
7
+ _tag: "Err",
8
+ error
9
+ });
10
+ var isErr = (result) => result._tag === "Err";
11
+
12
+ // src/validation.ts
13
+ var validateAll = (results, combineErrors) => {
14
+ const values = [];
15
+ let accumulated;
16
+ for (const result of results) {
17
+ if (isErr(result)) {
18
+ accumulated = accumulated === void 0 ? result.error : combineErrors(accumulated, result.error);
19
+ } else {
20
+ values.push(result.value);
21
+ }
22
+ }
23
+ return accumulated === void 0 ? ok(values) : err(accumulated);
24
+ };
25
+ var validateTuple = (results, combineErrors) => {
26
+ const values = [];
27
+ let accumulated;
28
+ for (const result of results) {
29
+ if (isErr(result)) {
30
+ const error = result.error;
31
+ accumulated = accumulated === void 0 ? error : combineErrors(accumulated, error);
32
+ } else {
33
+ values.push(result.value);
34
+ }
35
+ }
36
+ return accumulated === void 0 ? ok(values) : err(accumulated);
37
+ };
38
+ var validateAllArray = (results) => {
39
+ const values = [];
40
+ const errors = [];
41
+ for (const result of results) {
42
+ if (isErr(result)) {
43
+ errors.push(result.error);
44
+ } else {
45
+ values.push(result.value);
46
+ }
47
+ }
48
+ return errors.length === 0 ? ok(values) : err(errors);
49
+ };
50
+ var validateTupleArray = (results) => {
51
+ const values = [];
52
+ const errors = [];
53
+ for (const result of results) {
54
+ if (isErr(result)) {
55
+ errors.push(result.error);
56
+ } else {
57
+ values.push(result.value);
58
+ }
59
+ }
60
+ return errors.length === 0 ? ok(values) : err(errors);
61
+ };
62
+
63
+ export { validateAll, validateAllArray, validateTuple, validateTupleArray };
64
+ //# sourceMappingURL=validation.js.map
65
+ //# sourceMappingURL=validation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/result.ts","../src/validation.ts"],"names":[],"mappings":";AAaO,IAAM,EAAA,GAAK,CAAe,KAAA,MAA4B;AAAA,EAC3D,IAAA,EAAM,IAAA;AAAA,EACN;AACF,CAAA,CAAA;AAWO,IAAM,GAAA,GAAM,CAAyB,KAAA,MAA4B;AAAA,EACtE,IAAA,EAAM,KAAA;AAAA,EACN;AACF,CAAA,CAAA;AAwBO,IAAM,KAAA,GAAQ,CAAO,MAAA,KAA8C,MAAA,CAAO,IAAA,KAAS,KAAA;;;ACzBnF,IAAM,WAAA,GAAc,CACzB,OAAA,EACA,aAAA,KACmB;AACnB,EAAA,MAAM,SAAc,EAAC;AACrB,EAAA,IAAI,WAAA;AAEJ,EAAA,KAAA,MAAW,UAAU,OAAA,EAAS;AAC5B,IAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,MAAA,WAAA,GACE,gBAAgB,MAAA,GAAY,MAAA,CAAO,QAAQ,aAAA,CAAc,WAAA,EAAa,OAAO,KAAK,CAAA;AAAA,IACtF,CAAA,MAAO;AACL,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B;AAAA,EACF;AAEA,EAAA,OAAO,gBAAgB,MAAA,GAAY,EAAA,CAAG,MAAM,CAAA,GAAI,IAAI,WAAW,CAAA;AACjE;AAeO,IAAM,aAAA,GAAgB,CAC3B,OAAA,EACA,aAAA,KACwB;AACxB,EAAA,MAAM,SAAoB,EAAC;AAC3B,EAAA,IAAI,WAAA;AAEJ,EAAA,KAAA,MAAW,UAAU,OAAA,EAAS;AAC5B,IAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,MAAA,MAAM,QAAQ,MAAA,CAAO,KAAA;AACrB,MAAA,WAAA,GAAc,WAAA,KAAgB,MAAA,GAAY,KAAA,GAAQ,aAAA,CAAc,aAAa,KAAK,CAAA;AAAA,IACpF,CAAA,MAAO;AACL,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B;AAAA,EACF;AAEA,EAAA,OAAQ,gBAAgB,MAAA,GAAY,EAAA,CAAG,MAAM,CAAA,GAAI,IAAI,WAAW,CAAA;AAClE;AAaO,IAAM,gBAAA,GAAmB,CAC9B,OAAA,KAC8B;AAC9B,EAAA,MAAM,SAAc,EAAC;AACrB,EAAA,MAAM,SAAc,EAAC;AAErB,EAAA,KAAA,MAAW,UAAU,OAAA,EAAS;AAC5B,IAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B,CAAA,MAAO;AACL,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B;AAAA,EACF;AAEA,EAAA,OAAO,OAAO,MAAA,KAAW,CAAA,GAAI,GAAG,MAAM,CAAA,GAAI,IAAI,MAAM,CAAA;AACtD;AAaO,IAAM,kBAAA,GAAqB,CAChC,OAAA,KACgD;AAChD,EAAA,MAAM,SAAoB,EAAC;AAC3B,EAAA,MAAM,SAA2B,EAAC;AAElC,EAAA,KAAA,MAAW,UAAU,OAAA,EAAS;AAC5B,IAAA,IAAI,KAAA,CAAM,MAAM,CAAA,EAAG;AACjB,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAuB,CAAA;AAAA,IAC5C,CAAA,MAAO;AACL,MAAA,MAAA,CAAO,IAAA,CAAK,OAAO,KAAK,CAAA;AAAA,IAC1B;AAAA,EACF;AAEA,EAAA,OAAQ,OAAO,MAAA,KAAW,CAAA,GAAI,GAAG,MAAM,CAAA,GAAI,IAAI,MAAM,CAAA;AAIvD","file":"validation.js","sourcesContent":["import type { Err, Ok, Result } from \"./types.js\";\n\nexport type { Err, Ok, Result } from \"./types.js\";\n\n/**\n * Lifts a value into the success track.\n *\n * @example\n * ```ts\n * const r = ok(42); // Result<number, never>\n * const typed: Result<number, \"parse\"> = ok(1);\n * ```\n */\nexport const ok = <T, E = never>(value: T): Result<T, E> => ({\n _tag: \"Ok\",\n value,\n});\n\n/**\n * Lifts a value into the error track.\n *\n * @example\n * ```ts\n * const r = err({ kind: \"parse\", message: \"bad json\" });\n * // Result<never, { kind: \"parse\"; message: string }>\n * ```\n */\nexport const err = <T = never, E = unknown>(error: E): Result<T, E> => ({\n _tag: \"Err\",\n error,\n});\n\n/**\n * Type-narrowing predicate: returns `true` when the result is `Ok`.\n *\n * @example\n * ```ts\n * if (isOk(r)) {\n * console.log(r.value); // narrowed to Ok branch\n * }\n * ```\n */\nexport const isOk = <T, E>(result: Result<T, E>): result is Ok<T, E> => result._tag === \"Ok\";\n\n/**\n * Type-narrowing predicate: returns `true` when the result is `Err`.\n *\n * @example\n * ```ts\n * if (isErr(r)) {\n * metrics.inc(\"error\", { kind: r.error.kind });\n * }\n * ```\n */\nexport const isErr = <T, E>(result: Result<T, E>): result is Err<T, E> => result._tag === \"Err\";\n\n/** Fantasy Land `of` — alias of {@link ok}. */\nexport const of = ok;\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Dual-form helpers — each export accepts either shape:\n// data-first: `map(result, fn)`\n// curried: `map(fn)(result)`\n// Arity at the call site selects the overload.\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst mapImpl = <T, U, E>(result: Result<T, E>, fn: (value: T) => U): Result<U, E> =>\n isOk(result) ? ok(fn(result.value)) : err(result.error);\n\n/**\n * Transform the `Ok` value, passing `Err` through unchanged. Dual-form:\n * call data-first or curried (for use with {@link pipe}).\n *\n * @example\n * ```ts\n * map(ok(2), (n) => n * 3); // Ok 6 — data-first\n * pipe(ok(\"x\"), map((s) => s.length));// Ok 1 — curried\n * ```\n */\nexport function map<T, U, E>(result: Result<T, E>, fn: (value: T) => U): Result<U, E>;\nexport function map<T, U>(fn: (value: T) => U): <E>(result: Result<T, E>) => Result<U, E>;\nexport function map(\n ...args: [Result<unknown, unknown>, (value: unknown) => unknown] | [(value: unknown) => unknown]\n): unknown {\n if (args.length === 2) return mapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => mapImpl(result, fn);\n}\n\nconst mapErrImpl = <T, E, F>(result: Result<T, E>, fn: (error: E) => F): Result<T, F> =>\n isErr(result) ? err(fn(result.error)) : ok(result.value);\n\n/**\n * Transform the `Err` value, passing `Ok` through unchanged. Useful for\n * unifying heterogeneous failure types into one app-level union.\n *\n * @example\n * ```ts\n * type AppError = { kind: \"http\"; status: number } | { kind: \"parse\" };\n * pipe(\n * fetchSync(url), // Result<Body, { status: number }>\n * mapErr((e): AppError => ({ kind: \"http\", status: e.status })),\n * );\n * ```\n */\nexport function mapErr<T, E, F>(result: Result<T, E>, fn: (error: E) => F): Result<T, F>;\nexport function mapErr<E, F>(fn: (error: E) => F): <T>(result: Result<T, E>) => Result<T, F>;\nexport function mapErr(\n ...args: [Result<unknown, unknown>, (error: unknown) => unknown] | [(error: unknown) => unknown]\n): unknown {\n if (args.length === 2) return mapErrImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => mapErrImpl(result, fn);\n}\n\nconst bimapImpl = <T, U, E, F>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): Result<U, F> => (isOk(result) ? ok(onOk(result.value)) : err(onErr(result.error)));\n\n/**\n * Transform both tracks at once — `Ok` via `onOk`, `Err` via `onErr`.\n * Equivalent to `mapErr(onErr)(map(onOk)(result))` but in one pass.\n *\n * @example\n * ```ts\n * bimap(parsed, (cfg) => cfg.name, (e) => ({ kind: \"input\", cause: e }));\n * ```\n */\nexport function bimap<T, U, E, F>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): Result<U, F>;\nexport function bimap<T, U, E, F>(\n onOk: (value: T) => U,\n onErr: (error: E) => F,\n): (result: Result<T, E>) => Result<U, F>;\nexport function bimap(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => unknown, (error: unknown) => unknown]\n | [(value: unknown) => unknown, (error: unknown) => unknown]\n): unknown {\n if (args.length === 3) return bimapImpl(args[0], args[1], args[2]);\n const [onOk, onErr] = args;\n return (result: Result<unknown, unknown>) => bimapImpl(result, onOk, onErr);\n}\n\nconst flatMapImpl = <T, U, E, F>(\n result: Result<T, E>,\n fn: (value: T) => Result<U, F>,\n): Result<U, E | F> => (isOk(result) ? fn(result.value) : err(result.error));\n\n/**\n * Canonical bind (Fantasy Land `chain`). Chains a Result-returning step,\n * widening the error union to `E | F`. Short-circuits on `Err`.\n *\n * @example\n * ```ts\n * flatMap(parseInput(raw), (data) =>\n * data.id != null ? ok(data) : err({ kind: \"missing_id\" as const }),\n * );\n * // Result<Data, ParseError | { kind: \"missing_id\" }>\n * ```\n */\nexport function flatMap<T, U, E, F>(\n result: Result<T, E>,\n fn: (value: T) => Result<U, F>,\n): Result<U, E | F>;\nexport function flatMap<T, U, F>(\n fn: (value: T) => Result<U, F>,\n): <E>(result: Result<T, E>) => Result<U, E | F>;\nexport function flatMap(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => Result<unknown, unknown>]\n | [(value: unknown) => Result<unknown, unknown>]\n): unknown {\n if (args.length === 2) return flatMapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => flatMapImpl(result, fn);\n}\n\nconst recoverImpl = <T, E, F>(\n result: Result<T, E>,\n fn: (error: E) => Result<T, F>,\n): Result<T, F> => (isErr(result) ? fn(result.error) : ok(result.value));\n\n/**\n * Error-track bind — runs `fn` only when the result is `Err`, allowing\n * a failed workflow to recover to `Ok` or remap the failure. Mirror of\n * {@link flatMap} on the error channel.\n *\n * @example\n * ```ts\n * recover(networkResult, (e) =>\n * e.kind === \"rate_limit\" ? ok(cachedBody) : err(e),\n * );\n * ```\n */\nexport function recover<T, E, F>(\n result: Result<T, E>,\n fn: (error: E) => Result<T, F>,\n): Result<T, F>;\nexport function recover<T, E, F>(\n fn: (error: E) => Result<T, F>,\n): (result: Result<T, E>) => Result<T, F>;\nexport function recover(\n ...args:\n | [Result<unknown, unknown>, (error: unknown) => Result<unknown, unknown>]\n | [(error: unknown) => Result<unknown, unknown>]\n): unknown {\n if (args.length === 2) return recoverImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => recoverImpl(result, fn);\n}\n\nconst tapImpl = <T, E>(result: Result<T, E>, fn: (value: T) => void): Result<T, E> => {\n if (isOk(result)) fn(result.value);\n return result;\n};\n\n/**\n * Observe the `Ok` value for side effects (logging, metrics) without\n * changing the carried value. Passes `Err` through untouched.\n *\n * @example\n * ```ts\n * pipe(\n * parseConfig(raw),\n * tap((cfg) => log.info({ msg: \"parsed\", name: cfg.name })),\n * flatMap(validate),\n * );\n * ```\n */\nexport function tap<T, E>(result: Result<T, E>, fn: (value: T) => void): Result<T, E>;\nexport function tap<T>(fn: (value: T) => void): <E>(result: Result<T, E>) => Result<T, E>;\nexport function tap(\n ...args: [Result<unknown, unknown>, (value: unknown) => void] | [(value: unknown) => void]\n): unknown {\n if (args.length === 2) return tapImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => tapImpl(result, fn);\n}\n\nconst tapErrImpl = <T, E>(result: Result<T, E>, fn: (error: E) => void): Result<T, E> => {\n if (isErr(result)) fn(result.error);\n return result;\n};\n\n/**\n * Observe the `Err` value for side effects (logging, metrics) without\n * changing the carried error. Passes `Ok` through untouched.\n *\n * @example\n * ```ts\n * pipe(\n * loadUser(id),\n * tapErr((e) => metrics.inc(\"user.load.fail\", { kind: e.kind })),\n * );\n * ```\n */\nexport function tapErr<T, E>(result: Result<T, E>, fn: (error: E) => void): Result<T, E>;\nexport function tapErr<E>(fn: (error: E) => void): <T>(result: Result<T, E>) => Result<T, E>;\nexport function tapErr(\n ...args: [Result<unknown, unknown>, (error: unknown) => void] | [(error: unknown) => void]\n): unknown {\n if (args.length === 2) return tapErrImpl(args[0], args[1]);\n const fn = args[0];\n return (result: Result<unknown, unknown>) => tapErrImpl(result, fn);\n}\n\nconst matchImpl = <T, E, U>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): U => (isOk(result) ? onOk(result.value) : onErr(result.error));\n\n/**\n * Terminal collapse — fold both tracks into a single value. Dual-form:\n * 3-args data-first, 2-args curried for {@link pipe}. Returns whatever\n * the handlers return.\n *\n * For files that also import `match` from `ts-pattern`, the\n * collision-free alias {@link matchResult} is identical.\n *\n * @example\n * ```ts\n * const html = match(parsed, (cfg) => render(cfg), (e) => renderError(e));\n * ```\n */\nexport function match<T, E, U>(\n result: Result<T, E>,\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): U;\nexport function match<T, E, U>(\n onOk: (value: T) => U,\n onErr: (error: E) => U,\n): (result: Result<T, E>) => U;\nexport function match(\n ...args:\n | [Result<unknown, unknown>, (value: unknown) => unknown, (error: unknown) => unknown]\n | [(value: unknown) => unknown, (error: unknown) => unknown]\n): unknown {\n if (args.length === 3) return matchImpl(args[0], args[1], args[2]);\n const [onOk, onErr] = args;\n return (result: Result<unknown, unknown>) => matchImpl(result, onOk, onErr);\n}\n\n/** Collision-free alias for files that also import `match` from ts-pattern. */\nexport const matchResult = match;\n\n/**\n * Curried collapse with named slots — escape valve when positional `match`\n * order is unclear at the call site (e.g. when both handlers return the same\n * type and a transposed `match(r, onErr, onOk)` would silently compile).\n */\nexport const fold =\n <T, E, U>(handlers: { readonly ok: (value: T) => U; readonly err: (error: E) => U }) =>\n (result: Result<T, E>): U =>\n matchImpl(result, handlers.ok, handlers.err);\n\n/**\n * Returns the `Ok` value, or `defaultValue` when the result is `Err`.\n *\n * @example\n * ```ts\n * unwrapOr(parsedSetting, \"default-value\");\n * ```\n */\nexport const unwrapOr = <T, E>(result: Result<T, E>, defaultValue: T): T =>\n isOk(result) ? result.value : defaultValue;\n\n/** Test/assert helper — throws the original Err value when called on Err. */\nexport function unwrapOk<T, E>(result: Result<T, E>): T {\n if (isErr(result)) throw result.error;\n return result.value;\n}\n\n/** Test/assert helper — throws TypeError when called on Ok. */\nexport function unwrapErr<T, E>(result: Result<T, E>): E {\n if (isOk(result)) throw new TypeError(\"unwrapErr called on Ok\");\n return result.error;\n}\n\n/** Wrap a throwing sync function — neverthrow `Result.fromThrowable` */\nexport function trySync<A extends readonly unknown[], T, E>(\n fn: (...args: A) => T,\n onThrow: (error: unknown) => E,\n): (...args: A) => Result<T, E>;\nexport function trySync<F extends (...args: never) => unknown, E>(\n fn: F,\n onThrow: (error: unknown) => E,\n): (...args: Parameters<F>) => Result<ReturnType<F>, E>;\nexport function trySync(\n fn: (...args: never) => unknown,\n onThrow: (error: unknown) => unknown,\n): (...args: never) => Result<unknown, unknown> {\n return (...args: never) => {\n try {\n return ok(fn(...args));\n } catch (error) {\n return err(onThrow(error));\n }\n };\n}\n\n/** First failure wins; otherwise collects values in order */\nexport const combine = <T, E>(results: readonly Result<T, E>[]): Result<T[], E> => {\n const values: T[] = [];\n for (const result of results) {\n if (isErr(result)) return err(result.error);\n values.push(result.value);\n }\n return ok(values);\n};\n\n/** Tuple-preserving combine (neverthrow-style) */\nexport const combineTuple = <const R extends readonly Result<unknown, unknown>[]>(\n results: R,\n): CombineTuple<R> => {\n const values: unknown[] = [];\n for (const result of results) {\n if (isErr(result)) return err(result.error) as CombineTuple<R>;\n values.push(result.value);\n }\n return ok(values) as CombineTuple<R>;\n};\n\ntype _OkValue<R> = R extends { _tag: \"Ok\"; readonly value: infer T } ? T : never;\ntype _ErrValue<R> = R extends { _tag: \"Err\"; readonly error: infer E } ? E : never;\n\ntype CombineTuple<R extends readonly Result<unknown, unknown>[]> = Result<\n { [K in keyof R]: _OkValue<R[K]> },\n { [K in keyof R]: _ErrValue<R[K]> }[number]\n>;\n\n/**\n * Variadic value-first pipe — threads `value` through up to nine unary fns.\n *\n * ```ts\n * pipe(\n * parseConfig(raw),\n * map((cfg) => cfg.name),\n * flatMap((name) => name ? ok(name) : err({ kind: \"empty\" })),\n * tap(log),\n * );\n * ```\n */\nexport function pipe<A>(value: A): A;\nexport function pipe<A, B>(value: A, ab: (a: A) => B): B;\nexport function pipe<A, B, C>(value: A, ab: (a: A) => B, bc: (b: B) => C): C;\nexport function pipe<A, B, C, D>(value: A, ab: (a: A) => B, bc: (b: B) => C, cd: (c: C) => D): D;\nexport function pipe<A, B, C, D, E>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n): E;\nexport function pipe<A, B, C, D, E, F>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n): F;\nexport function pipe<A, B, C, D, E, F, G>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n): G;\nexport function pipe<A, B, C, D, E, F, G, H>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n gh: (g: G) => H,\n): H;\nexport function pipe<A, B, C, D, E, F, G, H, I>(\n value: A,\n ab: (a: A) => B,\n bc: (b: B) => C,\n cd: (c: C) => D,\n de: (d: D) => E,\n ef: (e: E) => F,\n fg: (f: F) => G,\n gh: (g: G) => H,\n hi: (h: H) => I,\n): I;\nexport function pipe(value: unknown, ...fns: ReadonlyArray<(x: unknown) => unknown>): unknown {\n let acc = value;\n for (const fn of fns) {\n acc = fn(acc);\n }\n return acc;\n}\n","import { err, isErr, ok } from \"./result.js\";\nimport type { Result } from \"./types.js\";\n\ntype OkValue<R> = R extends { readonly _tag: \"Ok\"; readonly value: infer T } ? T : never;\ntype ErrValue<R> = R extends { readonly _tag: \"Err\"; readonly error: infer E } ? E : never;\n\ntype ValidateTuple<R extends readonly Result<unknown, unknown>[], E> = Result<\n { [K in keyof R]: OkValue<R[K]> },\n E\n>;\ntype TupleErrors<R extends readonly Result<unknown, unknown>[]> = ErrValue<R[number]>;\n\n/**\n * Accumulate independent validation failures into a single combined error\n * via `combineErrors`. Returns `Ok<T[]>` only when every input is `Ok`;\n * otherwise returns `Err<E>` where `E` is the fold of all failures.\n *\n * Unlike `combine`, this does **not** short-circuit on first failure — use\n * for independent checks where you want to report all problems at once.\n *\n * @example\n * ```ts\n * validateAll(\n * [validateName(input), validateAge(input), validateEmail(input)],\n * (left, right) => [...left, ...right],\n * );\n * // Result<string[], FieldError[]>\n * ```\n */\nexport const validateAll = <T, E>(\n results: readonly Result<T, E>[],\n combineErrors: (left: E, right: E) => E,\n): Result<T[], E> => {\n const values: T[] = [];\n let accumulated: E | undefined;\n\n for (const result of results) {\n if (isErr(result)) {\n accumulated =\n accumulated === undefined ? result.error : combineErrors(accumulated, result.error);\n } else {\n values.push(result.value);\n }\n }\n\n return accumulated === undefined ? ok(values) : err(accumulated);\n};\n\n/**\n * Tuple-preserving variant of {@link validateAll}. All inputs must share\n * the same error type `E`. Output Ok preserves tuple positions.\n *\n * @example\n * ```ts\n * validateTuple(\n * [validateName(s), validateAge(s)] as const,\n * (l, r) => [...l, ...r],\n * );\n * // Result<readonly [string, number], FieldError[]>\n * ```\n */\nexport const validateTuple = <E, const R extends readonly Result<unknown, E>[]>(\n results: R,\n combineErrors: (left: E, right: E) => E,\n): ValidateTuple<R, E> => {\n const values: unknown[] = [];\n let accumulated: E | undefined;\n\n for (const result of results) {\n if (isErr(result)) {\n const error = result.error;\n accumulated = accumulated === undefined ? error : combineErrors(accumulated, error);\n } else {\n values.push(result.value);\n }\n }\n\n return (accumulated === undefined ? ok(values) : err(accumulated)) as ValidateTuple<R, E>;\n};\n\n/**\n * Like {@link validateAll}, but collects every failure into a readonly\n * array instead of folding via a combine function. The default choice\n * when you want \"all errors\" without an explicit join.\n *\n * @example\n * ```ts\n * validateAllArray([validateName(input), validateAge(input)]);\n * // Result<readonly Field[], readonly FieldError[]>\n * ```\n */\nexport const validateAllArray = <T, E>(\n results: readonly Result<T, E>[],\n): Result<T[], readonly E[]> => {\n const values: T[] = [];\n const errors: E[] = [];\n\n for (const result of results) {\n if (isErr(result)) {\n errors.push(result.error);\n } else {\n values.push(result.value);\n }\n }\n\n return errors.length === 0 ? ok(values) : err(errors);\n};\n\n/**\n * Tuple-preserving variant of {@link validateAllArray}. Heterogeneous\n * input tuple → preserved Ok tuple shape with a union of all input\n * error types collected into a readonly array.\n *\n * @example\n * ```ts\n * validateTupleArray([validateName(s), validateAge(s)] as const);\n * // Result<readonly [string, number], readonly FieldError[]>\n * ```\n */\nexport const validateTupleArray = <const R extends readonly Result<unknown, unknown>[]>(\n results: R,\n): ValidateTuple<R, readonly TupleErrors<R>[]> => {\n const values: unknown[] = [];\n const errors: TupleErrors<R>[] = [];\n\n for (const result of results) {\n if (isErr(result)) {\n errors.push(result.error as TupleErrors<R>);\n } else {\n values.push(result.value);\n }\n }\n\n return (errors.length === 0 ? ok(values) : err(errors)) as ValidateTuple<\n R,\n readonly TupleErrors<R>[]\n >;\n};\n"]}