import { Interface, ToInterface, Cast, ToConstructor, HasTail, Head, Tail, ListSpec, ToType } from "./Types"; import { Doc, Field } from "./Doc"; type AllToInterface = { 1: ToInterface> & AllToInterface>, 0: ToInterface> }[HasTail extends true ? 1 : 0]; export const emptySchema = createSchema({}); export const Document = makeInterface(emptySchema); export type Document = makeInterface<[typeof emptySchema]>; export type makeInterface = Partial> & Doc & { proto: Doc | undefined }; // export function makeInterface(schemas: T): (doc: U) => All; // export function makeInterface(schema: T): (doc: U) => makeInterface; export function makeInterface(...schemas: T): (doc?: Doc) => makeInterface { let schema: Interface = {}; for (const s of schemas) { for (const key in s) { schema[key] = s[key]; } } const proto = new Proxy({}, { get(target: any, prop, receiver) { const field = receiver.doc[prop]; if (prop in schema) { return Cast(field, (schema as any)[prop]); } return field; }, set(target: any, prop, value, receiver) { receiver.doc[prop] = value; return true; } }); return function (doc?: Doc) { doc = doc || new Doc; if (!(doc instanceof Doc)) { throw new Error("Currently wrapping a schema in another schema isn't supported"); } const obj = Object.create(proto, { doc: { value: doc, writable: false } }); return obj; }; } export type makeStrictInterface = Partial>; export function makeStrictInterface(schema: T): (doc: Doc) => makeStrictInterface { const proto = {}; for (const key in schema) { const type = schema[key]; Object.defineProperty(proto, key, { get() { return Cast(this.__doc[key], type as any); }, set(value) { value = Cast(value, type as any); if (value !== undefined) { this.__doc[key] = value; return; } throw new TypeError("Expected type " + type); } }); } return function (doc: any) { if (!(doc instanceof Doc)) { throw new Error("Currently wrapping a schema in another schema isn't supported"); } const obj = Object.create(proto); obj.__doc = doc; return obj; }; } export function createSchema(schema: T): T & { proto: ToConstructor } { schema.proto = Doc; return schema as any; } export function listSpec>(type: U): ListSpec> { return { List: type as any };//TODO Types }