union codecs, error messages

This commit is contained in:
Florian Dold 2019-12-14 17:55:31 +01:00
parent 749b96284a
commit 60d154c36b
No known key found for this signature in database
GPG Key ID: D2E4F00F29D02A4B
2 changed files with 113 additions and 6 deletions

View File

@ -19,14 +19,34 @@
*/
import test from "ava";
import { stringCodec, objectCodec } from "./codec";
import {
stringCodec,
objectCodec,
unionCodec,
Codec,
stringConstCodec,
} from "./codec";
interface MyObj {
foo: string;
}
test("basic codec", (t) => {
const myObjCodec = objectCodec<MyObj>().property("foo", stringCodec).build("MyObj");
interface AltOne {
type: "one";
foo: string;
}
interface AltTwo {
type: "two";
bar: string;
}
type MyUnion = AltOne | AltTwo;
test("basic codec", t => {
const myObjCodec = objectCodec<MyObj>()
.property("foo", stringCodec)
.build<MyObj>("MyObj");
const res = myObjCodec.decode({ foo: "hello" });
t.assert(res.foo === "hello");
@ -34,3 +54,24 @@ test("basic codec", (t) => {
const res2 = myObjCodec.decode({ foo: 123 });
});
});
test("union", t => {
const altOneCodec: Codec<AltOne> = objectCodec<AltOne>()
.property("type", stringConstCodec("one"))
.property("foo", stringCodec)
.build("AltOne");
const altTwoCodec: Codec<AltTwo> = objectCodec<AltTwo>()
.property("type", stringConstCodec("two"))
.property("bar", stringCodec)
.build("AltTwo");
const myUnionCodec: Codec<MyUnion> = unionCodec<MyUnion, "type">("type")
.alternative("one", altOneCodec)
.alternative("two", altTwoCodec)
.build<MyUnion>("MyUnion");
const res = myUnionCodec.decode({ type: "one", foo: "bla" });
t.is(res.type, "one");
if (res.type == "one") {
t.is(res.foo, "bla");
}
});

View File

@ -69,6 +69,11 @@ interface Prop {
codec: Codec<any>;
}
interface Alternative {
tagValue: any;
codec: Codec<any>;
}
class ObjectCodecBuilder<T, TC> {
private propList: Prop[] = [];
@ -89,10 +94,10 @@ class ObjectCodecBuilder<T, TC> {
* @param objectDisplayName name of the object that this codec operates on,
* used in error messages.
*/
build(objectDisplayName: string): Codec<TC> {
build<R extends (TC & T)>(objectDisplayName: string): Codec<R> {
const propList = this.propList;
return {
decode(x: any, c?: Context): TC {
decode(x: any, c?: Context): R {
if (!c) {
c = {
path: [`(${objectDisplayName})`],
@ -107,12 +112,53 @@ class ObjectCodecBuilder<T, TC> {
);
obj[prop.name] = propVal;
}
return obj as TC;
return obj as R;
},
};
}
}
class UnionCodecBuilder<T, D extends keyof T, TC> {
private alternatives = new Map<any, Alternative>();
constructor(private discriminator: D) {}
/**
* Define a property for the object.
*/
alternative<V>(
tagValue: T[D],
codec: Codec<V>,
): UnionCodecBuilder<T, D, TC | V> {
this.alternatives.set(tagValue, { codec, tagValue });
return this as any;
}
/**
* Return the built codec.
*
* @param objectDisplayName name of the object that this codec operates on,
* used in error messages.
*/
build<R extends TC>(objectDisplayName: string): Codec<R> {
const alternatives = this.alternatives;
const discriminator = this.discriminator;
return {
decode(x: any, c?: Context): R {
const d = x[discriminator];
if (d === undefined) {
throw new DecodingError(`expected tag for ${objectDisplayName} at ${renderContext(c)}.${discriminator}`);
}
const alt = alternatives.get(d);
if (!alt) {
throw new DecodingError(`unknown tag for ${objectDisplayName} ${d} at ${renderContext(c)}.${discriminator}`);
}
return alt.codec.decode(x);
}
};
}
}
/**
* Return a codec for a value that must be a string.
*/
@ -125,6 +171,20 @@ export const stringCodec: Codec<string> = {
},
};
/**
* Return a codec for a value that must be a string.
*/
export function stringConstCodec<V extends string>(s: V): Codec<V> {
return {
decode(x: any, c?: Context): V {
if (x === s) {
return x;
}
throw new DecodingError(`expected string constant "${s}" at ${renderContext(c)}`);
}
}
};
/**
* Return a codec for a value that must be a number.
*/
@ -179,3 +239,9 @@ export function mapCodec<T>(innerCodec: Codec<T>): Codec<{ [x: string]: T }> {
export function objectCodec<T>(): ObjectCodecBuilder<T, {}> {
return new ObjectCodecBuilder<T, {}>();
}
export function unionCodec<T, D extends keyof T>(
discriminator: D,
): UnionCodecBuilder<T, D, never> {
return new UnionCodecBuilder<T, D, never>(discriminator);
}