-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathpipe.spec.ts
104 lines (86 loc) · 2.57 KB
/
pipe.spec.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
import { Maybe, MaybeOpFn, MaybeOpFnAsync } from '@/src/maybe';
import { MaybeAsync, MaybeAsyncOpFn } from '@/src/maybeAsync';
import { ActionOfT, FunctionOfTtoK, isPromise, Some } from '@/src/utilities';
describe('Maybe', () => {
describe('pipe', () => {
test('executes all operator functions', () => {
const sut = Maybe.some(1);
const maybe = sut.pipe(
map((n) => n + 1),
map((n) => n * 2),
map((n) => `Calculation: ${n}`)
);
expect(sut).toHaveValue(1);
expect(maybe).toHaveValue('Calculation: 4');
});
test('handles side-effect operator functions', () => {
let callCount = 0;
const sut = Maybe.some(1);
const maybe = sut.pipe(
tap((n) => callCount++),
tap((n) => callCount++),
tap((n) => callCount++),
tap((n) => callCount++)
);
expect(maybe).toHaveValue(1);
expect(callCount).toBe(4);
});
test('handles transitioning to a MaybeAsync', async () => {
const sut = Maybe.some(1);
const maybe = await sut
.pipe(
map((n) => n + 1),
mapAsync((n) => Promise.resolve(n * 2))
)
.pipe(asyncMap((n) => n + 3))
.toPromise();
expect(maybe).toHaveValue(7);
});
});
});
function map<TValue, TNewValue>(
projection: FunctionOfTtoK<TValue, Some<TNewValue>>
): MaybeOpFn<TValue, TNewValue> {
return (maybe) => {
return maybe.hasValue
? Maybe.some(projection(maybe.getValueOrThrow()))
: Maybe.none<TNewValue>();
};
}
function tap<TValue>(action: ActionOfT<TValue>): MaybeOpFn<TValue, TValue> {
return (maybe) => {
if (maybe.hasValue) {
action(maybe.getValueOrThrow());
}
return maybe;
};
}
function mapAsync<TValue, TNewValue>(
projection: FunctionOfTtoK<TValue, Promise<Some<TNewValue>>>
): MaybeOpFnAsync<TValue, TNewValue> {
return (maybe) => {
return maybe.hasValue
? MaybeAsync.from(projection(maybe.getValueOrThrow()))
: MaybeAsync.none();
};
}
function asyncMap<TValue, TNewValue>(
projection:
| FunctionOfTtoK<TValue, Promise<Some<TNewValue>>>
| FunctionOfTtoK<TValue, Some<TNewValue>>
): MaybeAsyncOpFn<TValue, TNewValue> {
return (maybe) => {
return MaybeAsync.from<TNewValue>(
maybe.toPromise().then(async (m) => {
if (m.hasNoValue) {
return Maybe.none<TNewValue>();
}
const result = projection(m.getValueOrThrow());
if (isPromise(result)) {
return result.then((r) => Maybe.some<TNewValue>(r));
}
return Maybe.some(result);
})
);
};
}