relational data at its simplest
react-use-database gives you an opinionated interface, efficient data flow, and concise global state management. It forces you to think about your client-side data in the context of a queryable database. It gives you two global data stores: an entity store and a query store.
The entity store contains all of your model data. It’s just a big json blob that is the source of truth for any database entity that you have defined via Normalizr’s notion of schemas (eg UserSchema, TodoSchema, CommentSchema, etc.). Note, Normalizr is a peer dependency of react-use-database, you should familiarize yourself with it.
But, what good is a database without a way to pull data from it? That’s where the query store comes in. A query is comprised of a schema and a value. The query {schema: UserSchema, value: 32}
would return you a user whose id is 32. The query {schema: [UserSchema], value: [32, 33, 34]}
would return you an array of three users. By tracking and updating your entities and queries in their respective stores, every component can have access to the latest data with almost no additional code.
I used to implement relational state management with Redux/Normalizr. It worked well, but there was too much boilerplate and trying to onboard a new-hire to the idea was daunting. Redux and Normalizr are an incredible pairing, Dan Abramov was really onto something with his creation of each (He’s even done an Egghead tutorial series in which he describes almost the exact design pattern I’ve implemented in this library). The problem with Redux is that it lacks opinionation and it can be overwhelmingly verbose. It’s vulnerable to anti-patterns and there’s nothing inherent to its API to enforce its proper use. When used improperly, redux can become more of a burden than an asset.
For a more detailed introduction to react-use-database, see this medium article
npm install --save react-use-database
Live Demo: https://malerba118.github.io/react-use-database/
Demo Code: https://github.com/malerba118/react-use-database/tree/master/example/src
import React from "react";
import ReactDOM from "react-dom";
import { schema } from "normalizr";
import createDB from "react-use-database";
const TodoSchema = new schema.Entity("Todo");
let [DatabaseProvider, useDB] = createDB([TodoSchema], {
// Seed the database
defaultEntities: {
Todo: {
1: {
id: 1,
text: "Buy cheese"
}
}
}
});
const queries = {
getTodoById: id => {
return {
schema: TodoSchema,
value: id
};
}
};
const App = props => {
let db = useDB();
let queryToGetTodoWithIdOne = queries.getTodoById(1);
let todo = db.executeQuery(queryToGetTodoWithIdOne);
return <span>{todo.text}</span>;
};
ReactDOM.render(
<DatabaseProvider>
<App />
</DatabaseProvider>,
document.getElementById("root")
);
See the live example and example code for a complex implementation
Once we wrap our app in the DatabaseProvider we’re good to go. We can now use our database hook in any component to query the database.
Boom, now you can fetchTodos from anywhere and your TodosComponent will re-render with the latest list of todos.
This has to be my favorite one. You don’t even need to update any queries. You just take the updated Todo from the response body, normalize it, and deep merge it into the entity store and then your TodosComponent re-renders with the updated data. Virtually zero effort involved.
You might be seeing a pattern here. Updating the database is almost always as simple as normalizing data and passing the entities object to mergeEntities. This deep merges the entities patch onto the existing entities object. Once our Todo is created, we need to add its id to our ALL_TODOS query. Once this is done, our TodosComponent will re-render with the new todo.
To delete a todo, we can add a soft delete indicator to the todo in the database and we can update any relevant queries to omit the deleted todo id. Here’s a case where we probably don’t want to just normalize the data from the response and call mergeEntities on it.
We also can easily perform optimistic updates by normalizing and merging entities before the API call has finished. And then if the API call comes back with an error level status code, we can merge the original todo back into entities to revert the update.
Because a query is just a schema and value, we can create our own queries whose state is not tracked by the query store. For example, if we received a todo id as a url parameter, we could do something like the above.
Creates DatabaseProvider and useDB hook.
entitySchemas
: required Array or object whose values are normalizr Entity schemasoptions
: optional optionsstoredQueryDefinitions
: Object whose keys are query names and whose values have form{ schema, defaultValue }
defaultEntities
: An entities object to seed the database
let [ DatabaseProvider, useDB ] = createDB(
models,
{
storedQueryDefinitions: {
ALL_TODOS: {
schema: [models.TodoSchema],
defaultValue: []
},
ACTIVE_TODOS: {
schema: [models.TodoSchema],
defaultValue: []
},
COMPLETED_TODOS: {
schema: [models.TodoSchema],
defaultValue: []
},
},
defaultEntities: {
Todo: {
1: {
id: 1,
text: 'Buy cheese',
completed: false
}
}
}
}
);
React context provider that enables react-use-database to have global state.
ReactDOM.render(
<DatabaseProvider>
<App />
</DatabaseProvider>,
document.getElementById("root")
);
React database hook that allows you to query and update the database
const useNormalizedApi = () => {
let db = useDB();
return {
...
addTodo: async (text) => {
let todo = await api.addTodo(text);
let { result, entities } = normalize(
todo,
apiSchemas.addTodoResponseSchema
);
db.mergeEntities(entities); // Merge new todo data into database
db.updateStoredQuery('ALL_TODOS', (prevArray) => [...prevArray, todo.id]);
},
...
};
};
const TodosComponent = (props) => {
let db = useDB();
let allTodosQuery = db.getStoredQuery('ALL_TODOS');
let todos = db.executeQuery(allTodosQuery);
return (
<JSON data={todos} />
);
};
Immutably deep merges an entities patch on to the current entities object to produce next entities state. Triggers re-render for components consuming useDB.
entitiesPatch
: required function or partial entities object. If function, one argument will be passed, the current entities. Under the hood, lodash's mergeWith is called to merge the entitiesPatch onto the current entities to produce the next enitities object.options
: optional optionscustomizer
: overrides default customizer implementation passed to lodash's mergeWith
const TodosComponent = (props) => {
let db = useDB();
let todo = db.executeQuery({schema: TodoSchema, value: 1});
useEffect(() => {
db.mergeEntities({
Todo: {
1: {
id: 1,
text: 'Buy cheese',
completed: false
}
}
});
}, []);
return (
<JSON data={todo} />
);
};
Executes a query against the database (db.entities).
query
: required object with shape{schema: normalizr.schema, value: any}
const TodosComponent = (props) => {
let db = useDB();
let todos = db.executeQuery({schema: [TodoSchema], value: [1, 2, 3]});
return (
<JSON data={todos} />
);
};
Gets the current query state for the query name provided.
storedQueryName
: required query name from keys defined in storedQueryDefinitions
const models = [TodoSchema];
let [ DatabaseProvider, useDB ] = createDB(
models,
{
storedQueryDefinitions: {
ALL_TODOS: {
schema: [TodoSchema],
defaultValue: [1, 2, 3]
}
}
}
);
const TodosComponent = (props) => {
let db = useDB();
let allTodosQuery = db.getStoredQuery('ALL_TODOS');
console.log(allTodosQuery); // -> {schema: [TodoSchema], value: [1, 2, 3]}
let todos = db.executeQuery(allTodosQuery);
return (
<JSON data={todos} />
);
};
Updates the value of the query with name equal to storedQueryName and triggers re-render for components consuming useDB.
storedQueryName
: required query name from keys defined in storedQueryDefinitionsnextValue
: required The next value of the stored query or a function that takes in the current value of the stored query and returns the next value
const TodosComponent = (props) => {
let db = useDB();
let todo = db.executeStoredQuery('ALL_TODOS');
useEffect(() => {
db.updateStoredQuery('ALL_TODOS', (prevArray) => [...prevArray, 32]);
}, []);
return (
<JSON data={todo} />
);
};
An alias for db.executeQuery(db.getStoredQuery(storedQueryName))
.
storedQueryName
: required query name from keys defined in storedQueryDefinitions
const TodosComponent = (props) => {
let db = useDB();
let todos = db.executeStoredQuery('ALL_TODOS');
return (
<JSON data={todos} />
);
};
The root data object from the entity store. Useful to listen to state changes. Could be used to persist parts of state to local storage or to implement undo/redo features. Entities saved to local storage could be used to hydrate the store via createDB's defaultEntities option.
let [ DatabaseProvider, useDB ] = createDB(
models,
{
defaultEntities: LocalStorageClient.loadState('entities')
}
);
const useEntityListener = () => {
let db = useDB();
useEffect(() => {
LocalStorageClient.saveState('entities', db.entities)
}, [db.entities]);
};
The root data object from the query store. Useful to listen to state changes. Could be used to persist parts of state to local storage or to implement undo/redo features.
const useStoredQueryListener = () => {
let db = useDB();
useEffect(() => {
//do something
}, [db.storedQueries]);
};
MIT © malerba118