I wanted a single source of truth, and I wanted it without having to create schemas. You don't need to define every square inch of your data in a library domain specific way (ie. Typescript), instead just flatten and expand your data, nothing more. If you have many to many relationships, this is impossible to do without some configuration, so you will have to define which models are many-to-many relationships. There are many other configurations like aliases, but nothing is required. It's zero-config by nature.
yarn add sans-schema
import { flatten, expandModel, removeModel } from 'sans-schema';
This library has been used in production, but error handling is dealt with outside the library. If you don't provide the required parameters in the correct format, the code well crash (fail fast).
If you have mission critical code, feel free to wrap this code in exceptional handling. The code doesn't throw any exceptions currently. If anyone wants to contribute and add the throws, please feel free in a PR :)
import { configureStore } from 'redux-toolkit';
import { mergeModelReducer, flatten, expandModel, removeModel, mergeNormalizedModels } from 'sans-schema';
const store = configureStore({
companies: mergeModelReducer('companies'),
users: mergeModelReducer('users'),
});
const data = { id: 1, name: 'Jason', company: { id: 1, city: 'Montreal' } };
const flatData = flatten('users')(data);
const expandedFlatData = expandModel('users', { id: 1 }, flatData);
const flatDataSansCompanyWithId1 = removeModel('companies', data.company, flatData);
// Using the data with Redux
store.dispatch(mergeNormalizeModels(flatData));
const expandedUser = expandModel('users', { id: 1 }, store.getState());
// removing a model reference from all other models. ie. user: { id: 123, company: { id: 1 } } becomes { id: 123, company: null }, and the same for all other models referencing company = { id: 1 }
store.dispatch(mergeNormalizedModels(flatDataSansCompanyWithId1));
// ...then your action to remove the model from it's slice
store.disaptch(RemoveCompany({ id: 1 }));
flatten and removeModel return flattened data that should be merged in your reducers. By using a single action for all data changes you end up with a single render.
This is an example result from flatten(modelName, config)(modelInstance);
Also see sampleData.js
for examples
{
modelNames: { // model names must always be plural
1: {
// note: id is the only required data for EVERY model
id: 1, //the key is the id
name: 'hello', //arbitrary primitives
secondModelNames: {
id: 3, //Only the id well be here. expandModel well expand this.
},
}
},
secondModelNames: { ... },
thirdModelNames: { ... },
// Note: modelNames doesn't reference thirdModelNames, yet it's related through this m2m object
modelNamesXthirdModelNames: { // This key can be anything you want.
modelNames: {
1: {
id: 1, //Every model always has an id, even in m2m
thirdModelNames: [ //m2m only includes reference instances with ids only
{ id: 2 },
{ id: 3 },
...
],
}
...
},
thirdModelNames: {
2: {
id: 2,
modelNames: [
{ id: 3 },
...
],
},
...
},
},
...
}
Sans Schema demo [OUTDATED, the library is easier to use now]
... Note: I well eventually make another demo with commit by commit tutorial on using Sans Schema...
Flattens hierarchical data to be placed in a data store representing a single source of truth.
modelName : string (plural form)
-required
- Specifies which model the object, or array of objects represents.config :
Config Object
-optional
- Passed to every function, that defines custom relationships between models.data : object
-required
- Hierarchical data to be flattened returnsobject
- flat single-source data.
Flattens hierarchical data to be placed in a data store representing a single source of truth.
modelName : string (plural form)
-required
- Specifies which model the object, or array of objects represents.model : object
-required
- The model to be expandedstate : object
-required
- Single source flattened data (ie. store.getState())deepness : integer
-optional
- The depth the model is expandedconfig :
Config Object
-optional
- Passed to every function, that defines custom relationships between models. returnsobject
- Hierarchical data
Searches the state for references of model and nullifies them. This returns the flattened data representing the updated state of models that were changed, to be merged with the original state.
modelName : string (plural form)
-required
- Specifies which model the object, or array of objects represents.model : object
-required
- The model to be expandedstate : object
-required
- Single source flattened data (ie. store.getState())config :
Config Object
-optional
- Passed to every function, that defines custom relationships between models. returnsobject
- flat single-source data
Also, for Redux mergeModelReducer(modelName)
, mergeNormalizedModels(Action<{ models }>)
Also, mergeData(oldObj, newObj)
, which is a generic merging function that uses immer. It probably exists somewhere, and if it doesn't... someone should publish it =P
All keys in the config object are optional
{
models: [pluralModelName:string],
oneToOne: {
subjectModel: [referenceModel:string],
}
keyToModel: {
subjectModel: {
referenceKey: referenceModelName:string
}
},
manyToMany: {
m2mModelName: [
{
modelName:string: secondModelNameAlias:string
secondModelName:string: modelNameAlias:string
}
]
},
};
Note: You can name your manyToMany relationship whatever you want. In our examples and the tests we combine the names of the two models and put a capital X inbetween them.
Please read CONTRIBUTING.md for details on helping the project. Basically, make sure the tests pass and try to write nice code. If you're not sure whether you should submit a pull request, just do it, why not? ;)
We use SemVer for versioning. For the versions available, see the tags on this repository.
- Jason McCarrell - consultant - github
- Your name here