import { Interface, ToInterface, Cast, ToConstructor, HasTail, Head, Tail, ListSpec, ToType, DefaultFieldConstructor } from "./Types"; import { Doc, Field } from "./Doc"; import { ObjectField } from "./ObjectField"; import { RefField } from "./RefField"; import { SelfProxy } from "./FieldSymbols"; 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 interface InterfaceFunc { (docs: Doc[]): makeInterface[]; (): makeInterface; (doc: Doc): makeInterface; } export type makeInterface = AllToInterface & 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): InterfaceFunc { const 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) { const desc = prop === "proto" ? Doc : (schema as any)[prop]; // bcz: proto doesn't appear in schemas ... maybe it should? if (typeof desc === "object" && "defaultVal" in desc && "type" in desc) {//defaultSpec return Cast(field, desc.type, desc.defaultVal); } else if (typeof desc === "function" && !ObjectField.isPrototypeOf(desc) && !RefField.isPrototypeOf(desc)) { const doc = Cast(field, Doc); if (doc === undefined) { return undefined; } else if (doc instanceof Doc) { return desc(doc); } else { return doc.then(doc => doc && desc(doc)); } } else { return Cast(field, desc); } } return field; }, set(target: any, prop, value, receiver) { receiver.doc[prop] = value; return true; } }); const fn = (doc: Doc) => { doc = doc[SelfProxy]; // 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; }; return function (doc?: Doc | Doc[]) { doc = doc || new Doc; if (doc instanceof Doc) { return fn(doc); } else { return doc.map(fn); } }; } 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 as any).proto = Doc; return schema as any; } export function listSpec>(type: U): ListSpec> { return { List: type as any };//TODO Types } export function defaultSpec>(type: T, defaultVal: ToType): DefaultFieldConstructor> { return { type: type as any, defaultVal }; }