io-ts-extra
Version:
Adds pattern matching, optional properties, and several other helpers and types, to io-ts.
68 lines (65 loc) • 2.09 kB
text/typescript
import {Type, TypeOf, RefinementC, RefinementType, success, failure, getFunctionName, Context, Any} from 'io-ts'
import {either} from 'fp-ts/lib/Either'
const chain = either.chain
/**
* Like io-ts's refinement type but:
* 1. Not deprecated (see https://github.com/gcanti/io-ts/issues/373)
* 2. Passes in `Context` to the predicate argument, so you can check parent key names etc.
* 3. Optionally allows returning another io-ts codec instead of a boolean for better error messages.
*
* @example
* const CloudResources = narrow(
* t.type({
* database: t.type({username: t.string, password: t.string}),
* service: t.type({dbConnectionString: t.string}),
* }),
* ({database}) => t.type({
* service: t.type({dbConnectionString: t.literal(`${database.username}:${database.password}`)}),
* })
* )
*
* const valid = CloudResources.decode({
* database: {username: 'user', password: 'pass'},
* service: {dbConnectionString: 'user:pass'},
* })
* // returns a `Right`
*
* const invalid = CloudResources.decode({
* database: {username: 'user', password: 'pass'},
* service: {dbConnectionString: 'user:wrongpassword'},
* })
* // returns a `Left` - service.dbConnectionString expected "user:pass", but got "user:wrongpassword"
*/
export const narrow = <C extends Any, D extends Any>(
codec: C,
predicate: (value: TypeOf<C>, context: Context) => D | boolean,
name = `(${codec.name} | ${getFunctionName(predicate)})`
): RefinementC<C> => {
return new RefinementType(
name,
(u): u is TypeOf<C> => {
if (!codec.is(u)) {
return false
}
const refined = predicate(u, [])
if (refined instanceof Type) {
return refined.is(u)
}
return refined
},
(i, c) =>
chain(codec.validate(i, c), a => {
const refined = predicate(a, c)
if (refined instanceof Type) {
return refined.validate(a, c)
}
if (refined) {
return success(a)
}
return failure(a, c)
}),
codec.encode,
codec,
predicate as any
)
}