/* * ALR - Any Linux Repository * Copyright (C) 2024 Евгений Храмов * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package db import ( "context" "database/sql" "database/sql/driver" "encoding/json" "errors" "fmt" "sync" "github.com/jmoiron/sqlx" "plemya-x.ru/alr/internal/config" "plemya-x.ru/alr/pkg/loggerctx" "golang.org/x/exp/slices" "modernc.org/sqlite" ) // CurrentVersion is the current version of the database. // The database is reset if its version doesn't match this. const CurrentVersion = 2 func init() { sqlite.MustRegisterScalarFunction("json_array_contains", 2, jsonArrayContains) } // Package is a ALR package's database representation type Package struct { Name string `sh:"name,required" db:"name"` Version string `sh:"version,required" db:"version"` Release int `sh:"release,required" db:"release"` Epoch uint `sh:"epoch" db:"epoch"` Description JSON[map[string]string] `db:"description"` Homepage JSON[map[string]string] `db:"homepage"` Maintainer JSON[map[string]string] `db:"maintainer"` Architectures JSON[[]string] `sh:"architectures" db:"architectures"` Licenses JSON[[]string] `sh:"license" db:"licenses"` Provides JSON[[]string] `sh:"provides" db:"provides"` Conflicts JSON[[]string] `sh:"conflicts" db:"conflicts"` Replaces JSON[[]string] `sh:"replaces" db:"replaces"` Depends JSON[map[string][]string] `db:"depends"` BuildDepends JSON[map[string][]string] `db:"builddepends"` OptDepends JSON[map[string][]string] `db:"optdepends"` Repository string `db:"repository"` } type version struct { Version int `db:"version"` } var ( mu sync.Mutex conn *sqlx.DB closed = true ) // DB returns the ALR database. // The first time it's called, it opens the SQLite database file. // Subsequent calls return the same connection. func DB(ctx context.Context) *sqlx.DB { log := loggerctx.From(ctx) if conn != nil && !closed { return getConn() } _, err := open(ctx, config.GetPaths(ctx).DBPath) if err != nil { log.Fatal("Error opening database").Err(err).Send() } return getConn() } func getConn() *sqlx.DB { mu.Lock() defer mu.Unlock() return conn } func open(ctx context.Context, dsn string) (*sqlx.DB, error) { db, err := sqlx.Open("sqlite", dsn) if err != nil { return nil, err } mu.Lock() conn = db closed = false mu.Unlock() err = initDB(ctx, dsn) if err != nil { return nil, err } return db, nil } // Close closes the database func Close() error { closed = true if conn != nil { return conn.Close() } else { return nil } } // initDB initializes the database func initDB(ctx context.Context, dsn string) error { log := loggerctx.From(ctx) conn = conn.Unsafe() _, err := conn.ExecContext(ctx, ` CREATE TABLE IF NOT EXISTS pkgs ( name TEXT NOT NULL, repository TEXT NOT NULL, version TEXT NOT NULL, release INT NOT NULL, epoch INT, description TEXT CHECK(description = 'null' OR (JSON_VALID(description) AND JSON_TYPE(description) = 'object')), homepage TEXT CHECK(homepage = 'null' OR (JSON_VALID(homepage) AND JSON_TYPE(homepage) = 'object')), maintainer TEXT CHECK(maintainer = 'null' OR (JSON_VALID(maintainer) AND JSON_TYPE(maintainer) = 'object')), architectures TEXT CHECK(architectures = 'null' OR (JSON_VALID(architectures) AND JSON_TYPE(architectures) = 'array')), licenses TEXT CHECK(licenses = 'null' OR (JSON_VALID(licenses) AND JSON_TYPE(licenses) = 'array')), provides TEXT CHECK(provides = 'null' OR (JSON_VALID(provides) AND JSON_TYPE(provides) = 'array')), conflicts TEXT CHECK(conflicts = 'null' OR (JSON_VALID(conflicts) AND JSON_TYPE(conflicts) = 'array')), replaces TEXT CHECK(replaces = 'null' OR (JSON_VALID(replaces) AND JSON_TYPE(replaces) = 'array')), depends TEXT CHECK(depends = 'null' OR (JSON_VALID(depends) AND JSON_TYPE(depends) = 'object')), builddepends TEXT CHECK(builddepends = 'null' OR (JSON_VALID(builddepends) AND JSON_TYPE(builddepends) = 'object')), optdepends TEXT CHECK(optdepends = 'null' OR (JSON_VALID(optdepends) AND JSON_TYPE(optdepends) = 'object')), UNIQUE(name, repository) ); CREATE TABLE IF NOT EXISTS alr_db_version ( version INT NOT NULL ); `) if err != nil { return err } ver, ok := GetVersion(ctx) if ok && ver != CurrentVersion { log.Warn("Database version mismatch; resetting").Int("version", ver).Int("expected", CurrentVersion).Send() reset(ctx) return initDB(ctx, dsn) } else if !ok { log.Warn("Database version does not exist. Run alr fix if something isn't working.").Send() return addVersion(ctx, CurrentVersion) } return nil } // reset drops all the database tables func reset(ctx context.Context) error { _, err := DB(ctx).ExecContext(ctx, "DROP TABLE IF EXISTS pkgs;") if err != nil { return err } _, err = DB(ctx).ExecContext(ctx, "DROP TABLE IF EXISTS alr_db_version;") return err } // IsEmpty returns true if the database has no packages in it, otherwise it returns false. func IsEmpty(ctx context.Context) bool { var count int err := DB(ctx).GetContext(ctx, &count, "SELECT count(1) FROM pkgs;") if err != nil { return true } return count == 0 } // GetVersion returns the database version and a boolean indicating // whether the database contained a version number func GetVersion(ctx context.Context) (int, bool) { var ver version err := DB(ctx).GetContext(ctx, &ver, "SELECT * FROM alr_db_version LIMIT 1;") if err != nil { return 0, false } return ver.Version, true } func addVersion(ctx context.Context, ver int) error { _, err := DB(ctx).ExecContext(ctx, `INSERT INTO alr_db_version(version) VALUES (?);`, ver) return err } // InsertPackage adds a package to the database func InsertPackage(ctx context.Context, pkg Package) error { _, err := DB(ctx).NamedExecContext(ctx, ` INSERT OR REPLACE INTO pkgs ( name, repository, version, release, epoch, description, homepage, maintainer, architectures, licenses, provides, conflicts, replaces, depends, builddepends, optdepends ) VALUES ( :name, :repository, :version, :release, :epoch, :description, :homepage, :maintainer, :architectures, :licenses, :provides, :conflicts, :replaces, :depends, :builddepends, :optdepends ); `, pkg) return err } // GetPkgs returns a result containing packages that match the where conditions func GetPkgs(ctx context.Context, where string, args ...any) (*sqlx.Rows, error) { stream, err := DB(ctx).QueryxContext(ctx, "SELECT * FROM pkgs WHERE "+where, args...) if err != nil { return nil, err } return stream, nil } // GetPkg returns a single package that matches the where conditions func GetPkg(ctx context.Context, where string, args ...any) (*Package, error) { out := &Package{} err := DB(ctx).GetContext(ctx, out, "SELECT * FROM pkgs WHERE "+where+" LIMIT 1", args...) return out, err } // DeletePkgs deletes all packages matching the where conditions func DeletePkgs(ctx context.Context, where string, args ...any) error { _, err := DB(ctx).ExecContext(ctx, "DELETE FROM pkgs WHERE "+where, args...) return err } // jsonArrayContains is an SQLite function that checks if a JSON array // in the database contains a given value func jsonArrayContains(ctx *sqlite.FunctionContext, args []driver.Value) (driver.Value, error) { value, ok := args[0].(string) if !ok { return nil, errors.New("both arguments to json_array_contains must be strings") } item, ok := args[1].(string) if !ok { return nil, errors.New("both arguments to json_array_contains must be strings") } var array []string err := json.Unmarshal([]byte(value), &array) if err != nil { return nil, err } return slices.Contains(array, item), nil } // JSON represents a JSON value in the database type JSON[T any] struct { Val T } // NewJSON creates a new database JSON value func NewJSON[T any](v T) JSON[T] { return JSON[T]{Val: v} } func (s *JSON[T]) Scan(val any) error { if val == nil { return nil } switch val := val.(type) { case string: err := json.Unmarshal([]byte(val), &s.Val) if err != nil { return err } case sql.NullString: if val.Valid { err := json.Unmarshal([]byte(val.String), &s.Val) if err != nil { return err } } default: return errors.New("sqlite json types must be strings") } return nil } func (s JSON[T]) Value() (driver.Value, error) { data, err := json.Marshal(s.Val) if err != nil { return nil, err } return string(data), nil } func (s JSON[T]) MarshalYAML() (any, error) { return s.Val, nil } func (s JSON[T]) String() string { return fmt.Sprint(s.Val) } func (s JSON[T]) GoString() string { return fmt.Sprintf("%#v", s.Val) }