perkeep/pkg/sorted/sqlite/dbschema.go

111 lines
2.9 KiB
Go

/*
Copyright 2012 The Camlistore Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package sqlite
import (
"bytes"
"database/sql"
"fmt"
"log"
"os"
"os/exec"
"path/filepath"
"runtime"
"strconv"
"strings"
"camlistore.org/pkg/sorted"
)
const requiredSchemaVersion = 1
func SchemaVersion() int {
return requiredSchemaVersion
}
func SQLCreateTables() []string {
// sqlite ignores n in VARCHAR(n), but setting it as such for consistency with
// other sqls.
return []string{
`CREATE TABLE rows (
k VARCHAR(` + strconv.Itoa(sorted.MaxKeySize) + `) NOT NULL PRIMARY KEY,
v VARCHAR(` + strconv.Itoa(sorted.MaxValueSize) + `))`,
`CREATE TABLE meta (
metakey VARCHAR(255) NOT NULL PRIMARY KEY,
value VARCHAR(255) NOT NULL)`,
}
}
// IsWALCapable checks if the installed sqlite3 library can
// use Write-Ahead Logging (i.e version >= 3.7.0)
func IsWALCapable() bool {
// TODO(mpl): alternative to make it work on windows
cmdPath, err := exec.LookPath("pkg-config")
if err != nil {
log.Printf("Could not find pkg-config to check sqlite3 lib version: %v", err)
return false
}
var stderr bytes.Buffer
cmd := exec.Command(cmdPath, "--modversion", "sqlite3")
cmd.Stderr = &stderr
if runtime.GOOS == "darwin" && os.Getenv("PKG_CONFIG_PATH") == "" {
matches, err := filepath.Glob("/usr/local/Cellar/sqlite/*/lib/pkgconfig/sqlite3.pc")
if err == nil && len(matches) > 0 {
cmd.Env = append(os.Environ(), "PKG_CONFIG_PATH="+filepath.Dir(matches[0]))
}
}
out, err := cmd.Output()
if err != nil {
log.Printf("Could not check sqlite3 version: %v\n", stderr.String())
return false
}
version := strings.TrimRight(string(out), "\n")
return version >= "3.7.0"
}
// EnableWAL returns the statement to enable Write-Ahead Logging,
// which improves SQLite concurrency.
// Requires SQLite >= 3.7.0
func EnableWAL() string {
return "PRAGMA journal_mode = WAL"
}
// initDB creates a new sqlite database based on the file at path.
func initDB(path string) error {
db, err := sql.Open("sqlite3", path)
if err != nil {
return err
}
defer db.Close()
for _, tableSql := range SQLCreateTables() {
if _, err := db.Exec(tableSql); err != nil {
return err
}
}
if IsWALCapable() {
if _, err := db.Exec(EnableWAL()); err != nil {
return err
}
} else {
log.Print("WARNING: An SQLite DB without Write Ahead Logging will most likely fail. See http://camlistore.org/issues/114")
}
_, err = db.Exec(fmt.Sprintf(`REPLACE INTO meta VALUES ('version', '%d')`, SchemaVersion()))
return err
}