Add the CLI, site, and sample content so the project can run locally. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
146 lines
3.2 KiB
Go
146 lines
3.2 KiB
Go
package cli
|
|
|
|
import (
|
|
"database/sql"
|
|
_ "embed"
|
|
"errors"
|
|
"flag"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
_ "modernc.org/sqlite"
|
|
)
|
|
|
|
//go:embed sqlite_schema.sql
|
|
var sqliteSchema string
|
|
|
|
func runDB(root string, args []string) error {
|
|
if len(args) == 0 {
|
|
return errors.New("missing db subcommand")
|
|
}
|
|
switch args[0] {
|
|
case "init":
|
|
return runDBInit(root, args[1:])
|
|
case "status":
|
|
return runDBStatus(root, args[1:])
|
|
default:
|
|
return fmt.Errorf("unknown db subcommand %q", args[0])
|
|
}
|
|
}
|
|
|
|
func runDBInit(root string, args []string) error {
|
|
fs := flag.NewFlagSet("db init", flag.ContinueOnError)
|
|
fs.SetOutput(os.Stderr)
|
|
path := fs.String("path", defaultSQLitePath, "SQLite database path")
|
|
if err := fs.Parse(args); err != nil {
|
|
return err
|
|
}
|
|
|
|
dbPath := resolveRootPath(root, *path)
|
|
if err := os.MkdirAll(filepath.Dir(dbPath), 0o755); err != nil {
|
|
return err
|
|
}
|
|
|
|
db, err := openSQLite(dbPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer db.Close()
|
|
|
|
if err := applySQLiteSchema(db); err != nil {
|
|
return err
|
|
}
|
|
|
|
fmt.Printf("initialized SQLite database: %s\n", mustRel(root, dbPath))
|
|
return nil
|
|
}
|
|
|
|
func runDBStatus(root string, args []string) error {
|
|
fs := flag.NewFlagSet("db status", flag.ContinueOnError)
|
|
fs.SetOutput(os.Stderr)
|
|
path := fs.String("path", defaultSQLitePath, "SQLite database path")
|
|
if err := fs.Parse(args); err != nil {
|
|
return err
|
|
}
|
|
|
|
dbPath := resolveRootPath(root, *path)
|
|
info, err := os.Stat(dbPath)
|
|
if err != nil {
|
|
if errors.Is(err, os.ErrNotExist) {
|
|
fmt.Printf("database: %s\n", mustRel(root, dbPath))
|
|
fmt.Println("exists: no")
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
db, err := openSQLite(dbPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer db.Close()
|
|
|
|
fmt.Printf("database: %s\n", mustRel(root, dbPath))
|
|
fmt.Println("exists: yes")
|
|
fmt.Printf("size: %d bytes\n", info.Size())
|
|
for _, table := range []string{"posts", "settings", "sync_state"} {
|
|
ok, err := sqliteTableExists(db, table)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fmt.Printf("table %-10s %s\n", table+":", yesNo(ok))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func openSQLite(path string) (*sql.DB, error) {
|
|
db, err := sql.Open("sqlite", path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err := db.Ping(); err != nil {
|
|
db.Close()
|
|
return nil, err
|
|
}
|
|
return db, nil
|
|
}
|
|
|
|
func openProjectSQLite(root string, path string) (*sql.DB, string, error) {
|
|
dbPath := resolveRootPath(root, path)
|
|
if _, err := os.Stat(dbPath); err != nil {
|
|
if errors.Is(err, os.ErrNotExist) {
|
|
return nil, "", fmt.Errorf("database does not exist: %s; run `osaetctl db init` first", mustRel(root, dbPath))
|
|
}
|
|
return nil, "", err
|
|
}
|
|
db, err := openSQLite(dbPath)
|
|
if err != nil {
|
|
return nil, "", err
|
|
}
|
|
if err := applySQLiteSchema(db); err != nil {
|
|
db.Close()
|
|
return nil, "", err
|
|
}
|
|
return db, dbPath, nil
|
|
}
|
|
|
|
func applySQLiteSchema(db *sql.DB) error {
|
|
for _, statement := range strings.Split(sqliteSchema, ";") {
|
|
statement = strings.TrimSpace(statement)
|
|
if statement == "" {
|
|
continue
|
|
}
|
|
if _, err := db.Exec(statement); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func sqliteTableExists(db *sql.DB, table string) (bool, error) {
|
|
var count int
|
|
err := db.QueryRow(`SELECT count(*) FROM sqlite_master WHERE type='table' AND name=?`, table).Scan(&count)
|
|
return count > 0, err
|
|
}
|