sqlite-replication
A Typescript module to replicate SQLite DB with server.
sqlite-replication
was designed for collaborative offline-first mobile app built with capacitor sqlite plugin.
- Master > Slaves design
- Conflicts resolution on server side
- Framework agnostic (VueJs, React...)
- Server protocol agnostic (Rest API, GraphQL, websocket...)
- Query builder agnostic (no limit for SQL queries)
- Also Works with jeep-sqlite
Inspired from
Installation
npm install sqlite-replication --save
Usage
const db = ...
const storage = new ReplicationSQLiteStorage(db);
const replicationService = new ReplicationService(storage,
{
collections: [
ReplicationHelpers.getDefaultCollectionOptions(db, 'users'),
{
name: 'todos',
batchSize: 99,
getDocumentOffset: async (updatedAt: number,id: string) => { ... },
upsertAll: (documents: any[]) => { ... },
deleteAll: (documents: any[]) => { ... },
findChanges: (state: ReplicationConfig) => { ... };
}
],
fetchPull: async (pullConfig: any) => (await api.post(`${URL_BASE}/replicationPull`, pullConfig)).data,
fetchPush: async () => (await api.post(`${URL_BASE}/replicationPush`, {})).data,
}
);
await replicationService.init();
await replicationService.replicate();
Client requirements
Ensure your tables have at least these columns :
id
as text, as primary key (UUID recommended)updateAt
as timestamp, the last document change datedeletedAt
as timestamp, the deletion date (if deleted)_forkParent
as text, used for conflict handling.- Client should be able to store deleted documents ! Ensure no deletion, just add a
deletedAt
timestamp instead
Server requirements
- Ensure documents properties include :
- note that
_forkParent
is not required in server storage and could be removed - API should return deleted documents ! Ensure no deletion, just add a
deletedAt
timestamp instead - API should return documents in a predicable order. If you use SQL like database use
ORDER BY "updatedAt", "id"
- You would probably have to drop foreign constraints, it's not relevant in a distributed context, specially once you have circular relations
- ensure
id
is filled in a compatible way with distributed context (prefer UUID or similar)
Server Example with node Express API
const mobilePullBodySchema = {
body: {
type: 'object',
properties: {
collections: {
type: 'object',
properties: {},
additionalProperties: {
type: 'object',
properties: {
cursor: { type: 'number' },
limit: { type: 'number', format: 'int64' },
offset: { type: 'number', format: 'int64' },
},
required: ['cursor', 'limit', 'offset'],
},
},
},
required: ['collections'],
},
};
app.post('/replicationPull', validate(mobilePullBodySchema), async (req, res) => {
const collections = {};
if ( req.body.collections.users ) {
const { cursor, offset, limit } = req.body.collections.users;
const users = await database.select(
'id',
'name',
database.knex.raw('EXTRACT(EPOCH FROM "updatedAt") as "updatedAt"'),
'deletedAt',
)
.from('users')
.orderBy(['updatedAt', 'id'])
.where(database.knex.raw(`date_trunc('milliseconds',"updatedAt")`), '>=', database.knex.raw(`to_timestamp(${cursor})`)`))
// get the page data +1 to know if there is one more page next in a single sql query
.limit(limit + 1)
// skip the N first document with asked updateAt
.offset(offset);
}
collections.users = {
// escape the last document if the page limit is reach (due to limit+1)
documents: users.length > limit ? users.slice(0, -1) : users,
// define if there is a next page
hasMoreChanges: users.length > limit,
cursor,
offset,
limit,
});
}
res.json({ collections });
});
router.post('/replicationPush', validate(replicationPushSchema), async (req, res) => {
const collections = req.body.collections;
if (collections.users) {
await Promise.all(collections.users.map(pushUser));
}
res.json('ok');
});
async function pushUser(user) {
const forkParent = user._forkParent;
delete user._forkParent;
// remotely created case (insert), note that `id` is filled by client
if (forkParent.updatedAt === null) {
// insert in DB, note that `updateAt` default value is now()
return usersService.create(user)
}
// remotely deleted case (delete)
else if (user.deletedAt) {
// no deletion just flag the document with `deletedAt`=now() and `updateAt`=now() too.
return usersService.update({...user,updateAt:Date.now()});
}
//remotely update case
else if (forkParent.updatedAt !== user.updatedAt) {
const serverDocument = await usersService.getById(user.id),
// deleted on server (ignore remote update)
if (!serverDocument || serverDocument.deletedAt) {
return;
}
// no conflict case
else if (serverDocument.updatedAt === forkParent.updatedAt) {
return usersService.update({...user,updateAt:Date.now()});
}
//Conflict case
else {
const remoteChangedKeys = Object.keys(user).filter(
(key) => JSON.stringify(user[key]) !== JSON.stringify(forkParent[key]),
);
const serverChangedKeys = Object.keys(user).filter(
(key) => JSON.stringify(serverDocument[key]) !== JSON.stringify(forkParent[key]),
);
// handle conflict according to business rules
...
}
}
}
Client Replication process
A replication starts by sending local changes to the server then getting changes from the server.
By the way, server could fix conflicts and include merged documents during the pull step.
- Push Data
- get a batch of documents where
updatedAt
>= last updateAt
pulled (+ offset), this including deleted documents. - call fetchPush() hook to push them to the server. Each document includes a
_forkParent
property with a stringify version of the last server version known, to allow to the server to know changes at the property level. - loop until all documents are sent.
- Pull Data
- call fetchPull() to get a batch of documents. For each collection, cursor+offset+limit are sent to paginated results.
- each pulled document is stringified and set as
_forkParent
property document = {...document, _forkParent: JSON.stringify(document)}
- deleted documents from the server are removed from the local DB (real deletion this time)
- local replication state (cursor+offset) is updated
- loop pull process until all
hasMoreDocument
are false.
replicationService.replicationCompleted
observable trigger a new value.
Hooks & Customs
ReplicationOptions
interface defines two required hooks : fetchPush
&fetchPull
to define your way to reach the serverReplicationHelpers.getDefaultCollectionOptions(tableName)
generaly works by feel free to provide your own ReplicationCollectionOptions
to customize access to SQLite tables
export interface ReplicationCollectionOptions {
name: string;
batchSize: number;
upsertAll: (documents: any[]) => Promise<void> | Promise<capSQLiteChanges>;
deleteAll: (documents: any[]) => Promise<void> | Promise<capSQLiteChanges>;
getDocumentOffset: (updatedAt: number, id: string) => Promise<number>;
findChanges: (config: ReplicationConfig) => Promise<number>;
}
Data model upgrades
capacitor-community/sqlite UpgradeDatabaseVersion works well and is provided by SQLite itself.