Skip to content

Commit 609b398

Browse files
committed
safeTry should not require .safeUnwrap()
1 parent ac52282 commit 609b398

7 files changed

+2875
-10232
lines changed

.changeset/witty-pets-attend.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'neverthrow': minor
3+
---
4+
5+
safeTry should not require .safeUnwrap()

package-lock.json

+2,698-10,209
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+5-4
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
"scripts": {
1212
"local-ci": "npm run typecheck && npm run lint && npm run test && npm run format && npm run build",
1313
"test": "jest && npm run test-types",
14+
"jest": "jest",
1415
"test-types": "tsc --noEmit -p ./tests/tsconfig.tests.json",
1516
"lint": "eslint ./src --ext .ts",
1617
"format": "prettier --write 'src/**/*.ts?(x)' && npm run lint -- --fix",
@@ -36,21 +37,21 @@
3637
"@babel/preset-typescript": "7.24.7",
3738
"@changesets/changelog-github": "^0.5.0",
3839
"@changesets/cli": "^2.27.7",
39-
"@types/jest": "27.4.1",
40+
"@types/jest": "29.5.12",
4041
"@types/node": "^18.19.39",
4142
"@typescript-eslint/eslint-plugin": "4.28.1",
4243
"@typescript-eslint/parser": "4.28.1",
43-
"babel-jest": "27.5.1",
44+
"babel-jest": "29.7.0",
4445
"eslint": "7.30.0",
4546
"eslint-config-prettier": "7.1.0",
4647
"eslint-plugin-prettier": "3.4.0",
47-
"jest": "27.5.1",
48+
"jest": "29.7.0",
4849
"prettier": "2.2.1",
4950
"rollup": "^4.18.0",
5051
"rollup-plugin-dts": "^6.1.1",
5152
"rollup-plugin-typescript2": "^0.32.1",
5253
"testdouble": "3.20.2",
53-
"ts-jest": "27.1.5",
54+
"ts-jest": "29.2.5",
5455
"ts-toolbelt": "9.6.0",
5556
"typescript": "4.7.2"
5657
},

src/result-async.ts

+12
Original file line numberDiff line numberDiff line change
@@ -205,6 +205,18 @@ export class ResultAsync<T, E> implements PromiseLike<Result<T, E>> {
205205
): PromiseLike<A | B> {
206206
return this._promise.then(successCallback, failureCallback)
207207
}
208+
209+
async *[Symbol.asyncIterator](): AsyncGenerator<Err<never, E>, T> {
210+
const result = await this._promise
211+
212+
if (result.isErr()) {
213+
// @ts-expect-error -- This is structurally equivalent and safe
214+
yield errAsync(result.error)
215+
}
216+
217+
// @ts-expect-error -- This is structurally equivalent and safe
218+
return result.value
219+
}
208220
}
209221

210222
export const okAsync = <T, E = never>(value: T): ResultAsync<T, E> =>

src/result.ts

+14
Original file line numberDiff line numberDiff line change
@@ -381,6 +381,11 @@ export class Ok<T, E> implements IResult<T, E> {
381381
_unsafeUnwrapErr(config?: ErrorConfig): E {
382382
throw createNeverThrowError('Called `_unsafeUnwrapErr` on an Ok', this, config)
383383
}
384+
385+
// eslint-disable-next-line @typescript-eslint/no-this-alias require-yield
386+
*[Symbol.iterator](): Generator<Err<never, E>, T> {
387+
return this.value
388+
}
384389
}
385390

386391
export class Err<T, E> implements IResult<T, E> {
@@ -467,6 +472,15 @@ export class Err<T, E> implements IResult<T, E> {
467472
_unsafeUnwrapErr(_?: ErrorConfig): E {
468473
return this.error
469474
}
475+
476+
*[Symbol.iterator](): Generator<Err<never, E>, T> {
477+
// eslint-disable-next-line @typescript-eslint/no-this-alias
478+
const self = this
479+
// @ts-expect-error -- This is structurally equivalent and safe
480+
yield self
481+
// @ts-expect-error -- This is structurally equivalent and safe
482+
return self
483+
}
470484
}
471485

472486
export const fromThrowable = Result.fromThrowable

tests/safe-try.test.ts

+121
Original file line numberDiff line numberDiff line change
@@ -236,3 +236,124 @@ describe("Tests if README's examples work", () => {
236236
expect(result._unsafeUnwrap()).toBe(okValue + okValue)
237237
})
238238
})
239+
240+
describe("it yields and works without safeUnwrap", () => {
241+
test("With synchronous Ok", () => {
242+
const res: Result<string, string> = ok("ok");
243+
244+
const actual = safeTry(function* () {
245+
const x = yield* res;
246+
return ok(x);
247+
});
248+
249+
expect(actual).toBeInstanceOf(Ok);
250+
expect(actual._unsafeUnwrap()).toBe("ok");
251+
});
252+
253+
test("With synchronous Err", () => {
254+
const res: Result<number, string> = err("error");
255+
256+
const actual = safeTry(function* () {
257+
const x = yield* res;
258+
return ok(x);
259+
});
260+
261+
expect(actual).toBeInstanceOf(Err);
262+
expect(actual._unsafeUnwrapErr()).toBe("error");
263+
});
264+
265+
const okValue = 3;
266+
const errValue = "err!";
267+
268+
function good(): Result<number, string> {
269+
return ok(okValue);
270+
}
271+
function bad(): Result<number, string> {
272+
return err(errValue);
273+
}
274+
function promiseGood(): Promise<Result<number, string>> {
275+
return Promise.resolve(ok(okValue));
276+
}
277+
function promiseBad(): Promise<Result<number, string>> {
278+
return Promise.resolve(err(errValue));
279+
}
280+
function asyncGood(): ResultAsync<number, string> {
281+
return okAsync(okValue);
282+
}
283+
function asyncBad(): ResultAsync<number, string> {
284+
return errAsync(errValue);
285+
}
286+
287+
test("mayFail2 error", () => {
288+
function fn(): Result<number, string> {
289+
return safeTry<number, string>(function* () {
290+
const first = yield* good().mapErr((e) => `1st, ${e}`);
291+
const second = yield* bad().mapErr((e) => `2nd, ${e}`);
292+
293+
return ok(first + second);
294+
});
295+
}
296+
297+
const result = fn();
298+
expect(result.isErr()).toBe(true);
299+
expect(result._unsafeUnwrapErr()).toBe(`2nd, ${errValue}`);
300+
});
301+
302+
test("all ok", () => {
303+
function myFunc(): Result<number, string> {
304+
return safeTry<number, string>(function* () {
305+
const first = yield* good().mapErr((e) => `1st, ${e}`);
306+
const second = yield* good().mapErr((e) => `2nd, ${e}`);
307+
return ok(first + second);
308+
});
309+
}
310+
311+
const result = myFunc();
312+
expect(result.isOk()).toBe(true);
313+
expect(result._unsafeUnwrap()).toBe(okValue + okValue);
314+
});
315+
316+
test("async mayFail1 error", async () => {
317+
function myFunc(): ResultAsync<number, string> {
318+
return safeTry<number, string>(async function* () {
319+
const first = yield* (await promiseBad()).mapErr((e) => `1st, ${e}`);
320+
const second = yield* asyncGood().mapErr((e) => `2nd, ${e}`);
321+
return ok(first + second);
322+
});
323+
}
324+
325+
const result = await myFunc();
326+
expect(result.isErr()).toBe(true);
327+
expect(result._unsafeUnwrapErr()).toBe(`1st, ${errValue}`);
328+
});
329+
330+
test("async mayFail2 error", async () => {
331+
function myFunc(): ResultAsync<number, string> {
332+
return safeTry<number, string>(async function* () {
333+
const goodResult = await promiseGood();
334+
const value = yield* goodResult.mapErr((e) => `1st, ${e}`);
335+
const value2 = yield* asyncBad().mapErr((e) => `2nd, ${e}`);
336+
337+
return okAsync(value + value2);
338+
});
339+
}
340+
341+
const result = await myFunc();
342+
expect(result.isErr()).toBe(true);
343+
expect(result._unsafeUnwrapErr()).toBe(`2nd, ${errValue}`);
344+
});
345+
346+
test("promise async all ok", async () => {
347+
function myFunc(): ResultAsync<number, string> {
348+
return safeTry<number, string>(async function* () {
349+
const first = yield* (await promiseGood()).mapErr((e) => `1st, ${e}`);
350+
const second = yield* asyncGood().mapErr((e) => `2nd, ${e}`);
351+
return ok(first + second);
352+
});
353+
}
354+
355+
const result = await myFunc();
356+
expect(result.isOk()).toBe(true);
357+
expect(result._unsafeUnwrap()).toBe(okValue + okValue);
358+
});
359+
})

tests/tsconfig.tests.json

+20-19
Original file line numberDiff line numberDiff line change
@@ -1,26 +1,27 @@
11
{
22
"compilerOptions": {
3-
"target": "es2016",
4-
"module": "ES2015",
5-
"noImplicitAny": true,
6-
"sourceMap": false,
7-
"downlevelIteration": true,
8-
"noUnusedLocals": false,
9-
"noUnusedParameters": false,
10-
"strictNullChecks": true,
11-
"strictFunctionTypes": true,
12-
"declaration": true,
13-
"moduleResolution": "Node",
14-
"baseUrl": "./src",
15-
"lib": [
16-
"dom",
17-
"es2016",
18-
"es2017.object"
19-
],
20-
"outDir": "dist",
3+
"target": "es2016",
4+
"module": "ES2015",
5+
"noImplicitAny": true,
6+
"sourceMap": false,
7+
"downlevelIteration": true,
8+
"noUnusedLocals": false,
9+
"noUnusedParameters": false,
10+
"strictNullChecks": true,
11+
"strictFunctionTypes": true,
12+
"declaration": true,
13+
"moduleResolution": "Node",
14+
"baseUrl": "./src",
15+
"lib": [
16+
"dom",
17+
"es2016",
18+
"es2017.object"
19+
],
20+
"outDir": "dist",
21+
"skipLibCheck": true
2122
},
2223
"include": [
2324
"./index.test.ts",
2425
"./typecheck-tests.ts"
25-
],
26+
]
2627
}

0 commit comments

Comments
 (0)