Skip to content

Commit

Permalink
feat: rebuild SQLite when migrations occur
Browse files Browse the repository at this point in the history
See [#436].

[#436]: #436
  • Loading branch information
EvanHahn committed Oct 30, 2024
1 parent c0c8361 commit 6e20091
Show file tree
Hide file tree
Showing 5 changed files with 132 additions and 17 deletions.
3 changes: 3 additions & 0 deletions src/constants.js
Original file line number Diff line number Diff line change
Expand Up @@ -32,3 +32,6 @@ export const NAMESPACE_SCHEMAS = /** @type {const} */ ({
})

export const SUPPORTED_CONFIG_VERSION = 1

// WARNING: This value is persisted. Be careful when changing it.
export const DRIZZLE_MIGRATIONS_TABLE = '__drizzle_migrations'
48 changes: 48 additions & 0 deletions src/lib/drizzle-helpers.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import { sql } from 'drizzle-orm'
import { assert } from '../utils.js'
/** @import { BetterSQLite3Database } from 'drizzle-orm/better-sqlite3' */

/**
* @param {unknown} queryResult
* @returns {number}
*/
const getNumberResult = (queryResult) => {
assert(
queryResult &&
typeof queryResult === 'object' &&
'result' in queryResult &&
typeof queryResult.result === 'number',
'expected query to return proper result'
)
return queryResult.result
}

/**
* Get the number of rows in a table using `SELECT COUNT(*)`.
* Returns 0 if the table doesn't exist.
*
* @param {BetterSQLite3Database} db
* @param {string} tableName
* @returns {number}
*/
export const tableCountIfExists = (db, tableName) =>
db.transaction((tx) => {
const existsQuery = sql`
SELECT EXISTS (
SELECT 1
FROM sqlite_master
WHERE type IS 'table'
AND name IS ${tableName}
) AS result
`
const existsResult = tx.get(existsQuery)
const exists = getNumberResult(existsResult)
if (!exists) return 0

const countQuery = sql`
SELECT COUNT(*) AS result
FROM ${sql.identifier(tableName)}
`
const countResult = tx.get(countQuery)
return getNumberResult(countResult)
})
60 changes: 43 additions & 17 deletions src/mapeo-project.js
Original file line number Diff line number Diff line change
@@ -1,12 +1,17 @@
import path from 'path'
import * as fs from 'node:fs'
import Database from 'better-sqlite3'
import { decodeBlockPrefix, decode, parseVersionId } from '@comapeo/schema'
import { drizzle } from 'drizzle-orm/better-sqlite3'
import { migrate } from 'drizzle-orm/better-sqlite3/migrator'
import { discoveryKey } from 'hypercore-crypto'
import { TypedEmitter } from 'tiny-typed-emitter'

import { NAMESPACES, NAMESPACE_SCHEMAS } from './constants.js'
import {
NAMESPACES,
NAMESPACE_SCHEMAS,
DRIZZLE_MIGRATIONS_TABLE,
} from './constants.js'
import { CoreManager } from './core-manager/index.js'
import { DataStore } from './datastore/index.js'
import { DataType, kCreateWithDocId } from './datatype/index.js'
Expand Down Expand Up @@ -44,6 +49,7 @@ import {
projectKeyToPublicId,
valueOf,
} from './utils.js'
import { tableCountIfExists } from './lib/drizzle-helpers.js'
import { omit } from './lib/omit.js'
import { MemberApi } from './member-api.js'
import { SyncApi, kHandleDiscoveryKey } from './sync/sync-api.js'
Expand Down Expand Up @@ -139,11 +145,42 @@ export class MapeoProject extends TypedEmitter {
this.#isArchiveDevice = isArchiveDevice

///////// 1. Setup database

this.#sqlite = new Database(dbPath)
const db = drizzle(this.#sqlite)
migrate(db, { migrationsFolder: projectMigrationsFolder })
const migrationsBefore = tableCountIfExists(db, DRIZZLE_MIGRATIONS_TABLE)
migrate(db, {
migrationsFolder: projectMigrationsFolder,
migrationsTable: DRIZZLE_MIGRATIONS_TABLE,
})
const migrationsAfter = tableCountIfExists(db, DRIZZLE_MIGRATIONS_TABLE)
const didMigrateDatabase = migrationsAfter !== migrationsBefore

const indexedTables = [
observationTable,
trackTable,
presetTable,
fieldTable,
coreOwnershipTable,
roleTable,
deviceInfoTable,
iconTable,
translationTable,
remoteDetectionAlertTable,
]

///////// 2. Wipe data if we need to re-index

if (didMigrateDatabase) {
fs.rmSync(INDEXER_STORAGE_FOLDER_NAME, {
force: true,
recursive: true,
maxRetries: 10,
})
for (const table of indexedTables) db.delete(table).run()
}

///////// 2. Setup random-access-storage functions
///////// 3. Setup random-access-storage functions

/** @type {ConstructorParameters<typeof CoreManager>[0]['storage']} */
const coreManagerStorage = (name) =>
Expand All @@ -153,7 +190,7 @@ export class MapeoProject extends TypedEmitter {
const indexerStorage = (name) =>
coreStorage(path.join(INDEXER_STORAGE_FOLDER_NAME, name))

///////// 3. Create instances
///////// 4. Create instances

this.#coreManager = new CoreManager({
projectSecretKey,
Expand All @@ -166,18 +203,7 @@ export class MapeoProject extends TypedEmitter {
})

this.#indexWriter = new IndexWriter({
tables: [
observationTable,
trackTable,
presetTable,
fieldTable,
coreOwnershipTable,
roleTable,
deviceInfoTable,
iconTable,
translationTable,
remoteDetectionAlertTable,
],
tables: indexedTables,
sqlite: this.#sqlite,
getWinner,
mapDoc: (doc, version) => {
Expand Down Expand Up @@ -363,7 +389,7 @@ export class MapeoProject extends TypedEmitter {
dataType: this.#dataTypes.translation,
})

///////// 4. Replicate local peers automatically
///////// 5. Replicate local peers automatically

// Replicate already connected local peers
for (const peer of localPeers.peers) {
Expand Down
8 changes: 8 additions & 0 deletions test-e2e/migration.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,14 @@ const projectMigrationsFolder = new URL('../drizzle/project', import.meta.url)
const clientMigrationsFolder = new URL('../drizzle/client', import.meta.url)
.pathname

test('migrations pick up values that were previously not understood', async () => {
// TODO(evanhahn) Write this test
// Receive an observation with a new field, `foo`
// Get the bytes, add it to the core, see that it's in SQLite without a `foo` column
// Do a migration where `foo` is added
// Reload the project and see that `foo` is now there
})

test('migration of localDeviceInfo table', async (t) => {
const dbFolder = temporaryDirectory()
const rootKey = KeyManager.generateRootKey()
Expand Down
30 changes: 30 additions & 0 deletions test/lib/drizzle-helpers.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
import Database from 'better-sqlite3'
import { drizzle } from 'drizzle-orm/better-sqlite3'
import assert from 'node:assert/strict'
import test, { describe } from 'node:test'
import { tableCountIfExists } from '../../src/lib/drizzle-helpers.js'

describe('table count if exists', () => {
const db = new Database(':memory:')

db.exec('CREATE TABLE empty (ignored)')

db.exec('CREATE TABLE filled (n INT)')
db.exec('INSERT INTO filled (n) VALUES (9)')
db.exec('INSERT INTO filled (n) VALUES (8)')
db.exec('INSERT INTO filled (n) VALUES (7)')

const driz = drizzle(db)

test("when table doesn't exist", () => {
assert.equal(tableCountIfExists(driz, 'doesnt_exist'), 0)
})

test('when table is empty', () => {
assert.equal(tableCountIfExists(driz, 'empty'), 0)
})

test('when table has rows', () => {
assert.equal(tableCountIfExists(driz, 'filled'), 3)
})
})

0 comments on commit 6e20091

Please sign in to comment.