Skip to content

Commit

Permalink
WIP typed entity schemas
Browse files Browse the repository at this point in the history
  • Loading branch information
allanhortle committed Oct 14, 2023
1 parent 28a10bb commit 2212ab6
Show file tree
Hide file tree
Showing 8 changed files with 128 additions and 167 deletions.
31 changes: 13 additions & 18 deletions packages/enty/src/ArraySchema.ts
Original file line number Diff line number Diff line change
@@ -1,44 +1,39 @@
import {NormalizeState} from './util/definitions';
import {DenormalizeState} from './util/definitions';
import {Create} from './util/definitions';
import {StructuralSchemaInterface} from './util/definitions';
import {Merge} from './util/definitions';
import {StructuralSchemaOptions} from './util/definitions';
import {Structure} from './util/definitions';
import {StructureOptions} from './util/definitions';
import {Schema} from './util/definitions';
import {Entities} from './util/definitions';

import REMOVED_ENTITY from './util/RemovedEntity';

export default class ArraySchema<A extends Schema> implements StructuralSchemaInterface<A> {
shape: A;
create: Create;
merge: Merge;
export default class ArraySchema<T extends Array<any>> implements Structure<T> {
shape: Schema<T[number]>;
create: (next: T) => T;
merge: (previous: T, next: T) => T;

constructor(shape: A, options: StructuralSchemaOptions = {}) {
constructor(shape: Schema<T[number]>, options: StructureOptions<T> = {}) {
this.shape = shape;
this.merge = options.merge || ((_, bb) => bb);
this.create = options.create || ((aa) => aa);
}

normalize(data: any, entities: Entities = {}): NormalizeState {
normalize(data: T, entities: Entities = {}): NormalizeState {
let schemas = {};

const result = data.map((item: any): any => {
const {result, schemas: childSchemas} = this.shape.normalize(item, entities);
Object.assign(schemas, childSchemas);
return result;
});

return {entities, schemas, result: this.create(result)};
return {entities, schemas, result: this.create(result as T)};
}

denormalize(denormalizeState: DenormalizeState, path: Array<any> = []): any {
denormalize(denormalizeState: DenormalizeState, path: Array<any> = []): T | null {
const {result, entities} = denormalizeState;

// Filter out any deleted keys
if (result == null) {
return result;
}
// Map denormalize to our result List.
if (result == null) return result;

return result
.map((item: any): any => {
return this.shape.denormalize({result: item, entities}, path);
Expand Down
59 changes: 26 additions & 33 deletions packages/enty/src/EntitySchema.ts
Original file line number Diff line number Diff line change
@@ -1,64 +1,57 @@
import {NormalizeState} from './util/definitions';
import {DenormalizeState} from './util/definitions';
import {EntitySchemaOptions} from './util/definitions';
import {EntitySchemaInterface} from './util/definitions';
import {StructuralSchemaInterface} from './util/definitions';
import {IdAttribute} from './util/definitions';
import {Merge} from './util/definitions';
import {Structure} from './util/definitions';
import {Entities} from './util/definitions';

import {UndefinedIdError} from './util/Error';
import REMOVED_ENTITY from './util/RemovedEntity';
import ObjectSchema from './ObjectSchema';

export default class EntitySchema<A extends StructuralSchemaInterface<any>>
implements EntitySchemaInterface<A> {
type Options<T> = {
shape?: Structure<T>;
id?: (entity: T) => string;
merge?: (previous: T, next: T) => T;
};

export default class EntitySchema<T> {
name: string;
shape: A | null;
id: IdAttribute;
merge: Merge | null | undefined;
shape: Structure<T> | null;
id: (value: T) => string;
merge?: (previous: T, next: T) => T;

constructor(name: string, options: EntitySchemaOptions<any> = {}) {
constructor(name: string, options: Options<T> = {}) {
this.name = name;
this.merge = options.merge;

if (options.shape === null) {
this.shape = null;
this.id = options.id || (data => '' + data);
} else {
this.shape = options.shape || new ObjectSchema({});
this.id = options.id || ((data: any) => data?.id);
}
this.shape = options.shape ?? null;
this.id = options.id || ((data: any) => data?.id);
}

normalize(data: unknown, entities: Entities = {}): NormalizeState {
normalize(data: T, entities: Entities = {}): NormalizeState {
const {shape, name} = this;

let id = this.id(data);
let previousEntity;
let previousEntity: T | null = null;
let schemas: Record<string, any> = {};
let result;
let result: T;

if (id == null) {
throw UndefinedIdError(name, id);
}
if (id == null) throw UndefinedIdError(name, id);
id = id.toString();

entities[name] = entities[name] || {};

// only normalize if we have a defined shape
if (shape == null) {
result = data;
} else {
// only recurse if we have a defined shape
if (shape) {
let _ = shape.normalize(data, entities);
result = _.result;
schemas = _.schemas;
previousEntity = entities[name][id];
previousEntity = entities[name][id] as T;
} else {
result = data;
}

// list this schema as one that has been used
schemas[name] = this;

// Store the entity
entities[name][id] =
previousEntity && shape ? (this.merge || shape.merge)(previousEntity, result) : result;

Expand All @@ -69,13 +62,13 @@ export default class EntitySchema<A extends StructuralSchemaInterface<any>>
};
}

denormalize(denormalizeState: DenormalizeState, path: Array<any> = []): any {
denormalize(denormalizeState: DenormalizeState, path: Array<any> = []): T | null {
const {result, entities} = denormalizeState;
const {shape, name} = this;
const entity = entities?.[name]?.[result];

if (entity == null || entity === REMOVED_ENTITY || shape == null) {
return entity;
return null;
}

return shape.denormalize({result: entity, entities}, path);
Expand Down
87 changes: 64 additions & 23 deletions packages/enty/src/ObjectSchema.ts
Original file line number Diff line number Diff line change
@@ -1,37 +1,36 @@
import {NormalizeState} from './util/definitions';
import {NormalizeState, Structure} from './util/definitions';
import {DenormalizeState} from './util/definitions';
import {StructuralSchemaInterface} from './util/definitions';
import {Create} from './util/definitions';
import {Merge} from './util/definitions';
import {Entities} from './util/definitions';
import {Schema} from './util/definitions';
import {StructuralSchemaOptions} from './util/definitions';

import {StructureOptions} from './util/definitions';
import REMOVED_ENTITY from './util/RemovedEntity';

export default class ObjectSchema<A extends Record<string, Schema>>
implements StructuralSchemaInterface<A> {
create: Create;
merge: Merge;
shape: A;
export default class ObjectSchema<T extends Record<string, unknown>> implements Structure<T> {
create: (value: T) => T;
merge: (previous: T, next: T) => T;
relations: Partial<Record<keyof T, Schema<any>>>;

constructor(shape: A, options: StructuralSchemaOptions = {}) {
this.shape = shape;
this.create = options.create || (item => ({...item}));
constructor(
relations: Partial<Record<keyof T, Schema<any>>>,
options: StructureOptions<T> = {}
) {
this.relations = relations;
this.create = options.create || ((item) => ({...item}));
this.merge = options.merge || ((previous, next) => ({...previous, ...next}));
}

/**
* ObjectSchema.normalize
*/
normalize(data: any, entities: Entities = {}): NormalizeState {
const {shape} = this;
normalize(data: T, entities: Entities = {}): NormalizeState {
const {relations} = this;
const dataMap = data;
let schemas = {};

const result = Object.keys(shape).reduce((result: Object, key: any): any => {
const result = Object.keys(relations).reduce((result: Object, key: any): any => {
const value = dataMap[key];
const schema = shape[key];
const schema = relations[key];
if (!schema) throw new Error(`${String(key)} was not defined in shape`);
if (value) {
const {result: childResult, schemas: childSchemas} = schema.normalize(
value,
Expand All @@ -50,9 +49,9 @@ export default class ObjectSchema<A extends Record<string, Schema>>
/**
* ObjectSchema.denormalize
*/
denormalize(denormalizeState: DenormalizeState, path: Array<any> = []): any {
denormalize(denormalizeState: DenormalizeState, path: Array<any> = []): T | null {
const {result, entities} = denormalizeState;
const {shape} = this;
const {relations} = this;

if (result == null || result === REMOVED_ENTITY) {
return result;
Expand All @@ -62,16 +61,17 @@ export default class ObjectSchema<A extends Record<string, Schema>>
// if they have a corresponding schema. Otherwise return the plain value.
// Then filter out deleted keys, keeping track of ones deleted
// Then Pump the filtered object through `denormalizeFilter`
let item = {...result};
let item: T = {...result};
if (path.indexOf(this) !== -1) {
return item;
}

const keys = Object.keys(shape);
const keys = Object.keys(relations) as Array<keyof T>;

for (let key of keys) {
const schema = shape[key];
const schema = relations[key];
const result = item[key];
if (!schema) throw new Error(`${String(key)} was not defined in shape`);
const value = schema.denormalize({result, entities}, [...path, this]);

if (value !== REMOVED_ENTITY && value !== undefined) {
Expand All @@ -84,3 +84,44 @@ export default class ObjectSchema<A extends Record<string, Schema>>
return item;
}
}

//export class ClassSchema<T, A extends {new (a: any): any}> extends ObjectSchema<T> {
//model: A;
//constructor(model: A, shape: Record<string, Schema>, options = {}) {
//super(shape, options);
//this.model = model;
//}
//denormalize(denormalizeState, path): A {
//const data = super.denormalize(denormalizeState, path);
//return data ? new this.model(data) : data;
//}
//}
//type Cat = {id: string};
//type Dog = {id: string; woof: boolean; blah: {id: string}};
//type Person = {id: string; name: string; cat: Cat; dogs: Dog[]};
//const cat = new EntitySchema<Cat>('cat', {
////shape: new ObjectSchema({})
//});

//cat.shape

//const dog = new EntitySchema<Dog>('dog', {
//id: (x) => String(x.woof),
//merge: (a, b) => ({...a, ...b, woof: false}),
//shape: new ObjectSchema({blah: new ObjectSchema({})})
//});

//const person = new EntitySchema('person', {
//shape: new ObjectSchema<Person>({
//cat,
//dogs: new ArraySchema(cat)
//})
//});
//const state = person.normalize({id: 'asda', name: 'ads', cat: {id: 'foo'}, dogs: [{id: 'foo', woof: true}]);
//const pp = person.denormalize(state);

//const foo = new ObjectSchema<{blah: {name: string}}>({
//blah: person
//});

//const rad = foo.denormalize();
19 changes: 0 additions & 19 deletions packages/enty/src/__tests__/ArraySchema-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -73,25 +73,6 @@ test('ArraySchema will not mutate input objects', () => {
expect(arrayTest).toEqual([{id: '1'}]);
});

test('ArraySchemas can construct custom objects', () => {
class Foo {
data: string[];
constructor(data: string[]) {
this.data = data;
}
map(fn: (x: string) => string) {
this.data = this.data.map(fn);
return this;
}
}
const schema = new ArraySchema(new ObjectSchema({}), {
create: (data) => new Foo(data)
});
const state = schema.normalize([{foo: 1}, {bar: 2}], {});
expect(state.result).toBeInstanceOf(Foo);
expect(state.result.data[0]).toEqual({foo: 1});
});

it('will default replace array on merge', () => {
const foo = new EntitySchema('foo');
const schema = new ArraySchema(foo);
Expand Down
13 changes: 2 additions & 11 deletions packages/enty/src/__tests__/EntitySchema-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -79,20 +79,11 @@ describe('EntitySchema.normalize', () => {

it('will treat null shapes like an Id schema', () => {
const NullSchemaEntity = new EntitySchema('foo', {
shape: null,
id: data => `${data}-foo`
id: (data) => `${data}-foo`
});
const state = NullSchemaEntity.normalize(2, {});
expect(state.entities.foo['2-foo']).toBe(2);
});

it('will default id function to stringify if shape is null', () => {
const NullSchemaEntity = new EntitySchema('foo', {
shape: null
});
const state = NullSchemaEntity.normalize({}, {});
expect(state.entities.foo['[object Object]']).toEqual({});
});
});

describe('EntitySchema.denormalize', () => {
Expand Down Expand Up @@ -141,7 +132,7 @@ describe('EntitySchema.denormalize', () => {
it('can denormalize null shapes', () => {
const NullSchemaEntity = new EntitySchema('foo', {
shape: null,
id: data => `${data}-foo`
id: (data) => `${data}-foo`
});
const state = NullSchemaEntity.normalize(2, {});
expect(NullSchemaEntity.denormalize(state)).toBe(2);
Expand Down
26 changes: 3 additions & 23 deletions packages/enty/src/__tests__/ObjectSchema-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -32,10 +32,10 @@ test('ObjectSchema.denormalize is the inverse of ObjectSchema.normalize', () =>

test('ObjectSchema can normalize empty objects', () => {
const schema = new ObjectSchema({foo});
let {entities, result} = schema.normalize({bar: {}});
let {entities, result} = schema.normalize({foo: {}});

expect(entities).toEqual({});
expect(result).toEqual({bar: {}});
expect(result).toEqual({foo: {}});
});

test('ObjectSchema can denormalize objects', () => {
Expand Down Expand Up @@ -113,28 +113,8 @@ test('ObjectSchema will not mutate input objects', () => {
expect(objectTest).toEqual({foo: {id: '1'}});
});

test('ObjectSchemas can create objects', () => {
class Foo {
first: string;
last: string;
constructor(data: {first: string; last: string}) {
this.first = data.first;
this.last = data.last;
}
}
const schema = new ObjectSchema(
{},
{
create: (data) => new Foo(data)
}
);
const state = schema.normalize({first: 'foo', last: 'bar'}, {});

expect(state.result).toBeInstanceOf(Foo);
});

it('will not create extra keys if value is undefined', () => {
const schema = new ObjectSchema({
const schema = new ObjectSchema<{foo: {id: string}; bar?: {id: string}}>({
foo: new EntitySchema('foo'),
bar: new EntitySchema('bar')
});
Expand Down
Loading

0 comments on commit 2212ab6

Please sign in to comment.