zk/adapter/sqlite/db.go

182 lines
5.4 KiB
Go
Raw Normal View History

2021-01-02 11:29:21 +00:00
package sqlite
import (
"database/sql"
_ "github.com/mattn/go-sqlite3"
"github.com/mickael-menu/zk/core/note"
2021-01-02 11:29:21 +00:00
"github.com/mickael-menu/zk/util/errors"
)
// DB holds the connections to a SQLite database.
type DB struct {
2021-01-03 16:39:04 +00:00
db *sql.DB
2021-01-02 11:29:21 +00:00
}
// Open creates a new DB instance for the SQLite database at the given path.
func Open(path string) (*DB, error) {
2021-01-04 19:03:32 +00:00
return open("file:" + path)
}
// OpenInMemory creates a new in-memory DB instance.
func OpenInMemory() (*DB, error) {
return open(":memory:")
}
func open(uri string) (*DB, error) {
2021-01-25 20:44:44 +00:00
wrap := errors.Wrapper("failed to open the database")
2021-01-04 19:03:32 +00:00
db, err := sql.Open("sqlite3", uri)
2021-01-02 11:29:21 +00:00
if err != nil {
2021-01-25 20:44:44 +00:00
return nil, wrap(err)
2021-01-02 11:29:21 +00:00
}
2021-01-25 20:44:44 +00:00
// Make sure that CASCADE statements are properly applied by enabling
// foreign keys.
_, err = db.Exec("PRAGMA foreign_keys = ON")
if err != nil {
return nil, wrap(err)
}
2021-01-02 11:29:21 +00:00
return &DB{db}, nil
}
// Close terminates the connections to the SQLite database.
func (db *DB) Close() error {
2021-01-03 16:39:04 +00:00
err := db.db.Close()
2021-01-02 11:29:21 +00:00
return errors.Wrap(err, "failed to close the database")
}
// Migrate upgrades the SQL schema of the database.
func (db *DB) Migrate() (needsReindexing bool, err error) {
err = db.WithTransaction(func(tx Transaction) error {
2021-01-03 16:39:04 +00:00
var version int
err := tx.QueryRow("PRAGMA user_version").Scan(&version)
if err != nil {
return err
}
2021-01-02 11:29:21 +00:00
if version <= 0 {
2021-01-03 16:39:04 +00:00
err = tx.ExecStmts([]string{
2021-01-25 20:44:44 +00:00
// Notes
`CREATE TABLE IF NOT EXISTS notes (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
path TEXT NOT NULL,
sortable_path TEXT NOT NULL,
title TEXT DEFAULT('') NOT NULL,
lead TEXT DEFAULT('') NOT NULL,
body TEXT DEFAULT('') NOT NULL,
raw_content TEXT DEFAULT('') NOT NULL,
word_count INTEGER DEFAULT(0) NOT NULL,
checksum TEXT NOT NULL,
created DATETIME DEFAULT(CURRENT_TIMESTAMP) NOT NULL,
modified DATETIME DEFAULT(CURRENT_TIMESTAMP) NOT NULL,
UNIQUE(path)
)`,
2021-01-07 18:29:04 +00:00
`CREATE INDEX IF NOT EXISTS index_notes_checksum ON notes (checksum)`,
`CREATE INDEX IF NOT EXISTS index_notes_path ON notes (path)`,
2021-01-25 20:44:44 +00:00
// Links
`CREATE TABLE IF NOT EXISTS links (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
source_id INTEGER NOT NULL REFERENCES notes(id)
ON DELETE CASCADE,
target_id INTEGER REFERENCES notes(id)
ON DELETE SET NULL,
title TEXT DEFAULT('') NOT NULL,
href TEXT NOT NULL,
external INT DEFAULT(0) NOT NULL,
2021-01-30 15:32:58 +00:00
rels TEXT DEFAULT('') NOT NULL,
snippet TEXT DEFAULT('') NOT NULL
2021-01-25 20:44:44 +00:00
)`,
`CREATE INDEX IF NOT EXISTS index_links_source_id_target_id ON links (source_id, target_id)`,
// FTS index
`CREATE VIRTUAL TABLE IF NOT EXISTS notes_fts USING fts5(
path, title, body,
content = notes,
content_rowid = id,
2021-01-11 19:19:39 +00:00
tokenize = "porter unicode61 remove_diacritics 1 tokenchars '''&/'"
)`,
2021-01-03 16:39:04 +00:00
// Triggers to keep the FTS index up to date.
`CREATE TRIGGER IF NOT EXISTS trigger_notes_ai AFTER INSERT ON notes BEGIN
INSERT INTO notes_fts(rowid, path, title, body) VALUES (new.id, new.path, new.title, new.body);
END`,
`CREATE TRIGGER IF NOT EXISTS trigger_notes_ad AFTER DELETE ON notes BEGIN
INSERT INTO notes_fts(notes_fts, rowid, path, title, body) VALUES('delete', old.id, old.path, old.title, old.body);
END`,
`CREATE TRIGGER IF NOT EXISTS trigger_notes_au AFTER UPDATE ON notes BEGIN
INSERT INTO notes_fts(notes_fts, rowid, path, title, body) VALUES('delete', old.id, old.path, old.title, old.body);
INSERT INTO notes_fts(rowid, path, title, body) VALUES (new.id, new.path, new.title, new.body);
END`,
2021-01-03 16:39:04 +00:00
`PRAGMA user_version = 1`,
})
2021-01-02 11:29:21 +00:00
2021-01-03 16:39:04 +00:00
if err != nil {
return err
}
}
2021-01-02 11:29:21 +00:00
if version <= 1 {
err = tx.ExecStmts([]string{
// Collections
`CREATE TABLE IF NOT EXISTS collections (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
kind TEXT NO NULL,
name TEXT NOT NULL,
UNIQUE(kind, name)
)`,
`CREATE INDEX IF NOT EXISTS index_collections ON collections (kind, name)`,
// Note-Collection association
`CREATE TABLE IF NOT EXISTS notes_collections (
id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,
note_id INTEGER NOT NULL REFERENCES notes(id)
ON DELETE CASCADE,
collection_id INTEGER NOT NULL REFERENCES collections(id)
ON DELETE CASCADE
)`,
`CREATE INDEX IF NOT EXISTS index_notes_collections ON notes_collections (note_id, collection_id)`,
// View of notes with their associated metadata (e.g. tags), for simpler queries.
`CREATE VIEW notes_with_metadata AS
SELECT n.*, GROUP_CONCAT(c.name, '` + "\x01" + `') AS tags
FROM notes n
LEFT JOIN notes_collections nc ON nc.note_id = n.id
LEFT JOIN collections c ON nc.collection_id = c.id AND c.kind = '` + string(note.CollectionKindTag) + `'
GROUP BY n.id`,
`PRAGMA user_version = 2`,
})
if err != nil {
return err
}
}
if version <= 2 {
err = tx.ExecStmts([]string{
// Add a `metadata` column to `notes`
`ALTER TABLE notes ADD COLUMN metadata TEXT DEFAULT('{}') NOT NULL`,
// Add snippet's start and end offsets to `links`
`ALTER TABLE links ADD COLUMN snippet_start INTEGER DEFAULT(0) NOT NULL`,
`ALTER TABLE links ADD COLUMN snippet_end INTEGER DEFAULT(0) NOT NULL`,
`PRAGMA user_version = 3`,
})
if err != nil {
return err
}
needsReindexing = true
}
2021-01-03 16:39:04 +00:00
return nil
})
2021-01-02 11:29:21 +00:00
err = errors.Wrap(err, "database migration failed")
return
2021-01-02 11:29:21 +00:00
}