// This file was originally part of the project "LURE - Linux User REpository", created by Elara Musayelyan. // It has been modified as part of "ALR - Any Linux Repository" by Евгений Храмов. // // ALR - Any Linux Repository // Copyright (C) 2025 Евгений Храмов // // 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" "github.com/jmoiron/sqlx" "plemya-x.ru/alr/internal/config" "plemya-x.ru/alr/pkg/loggerctx" ) // CurrentVersion is the current version of the database. // The database is reset if its version doesn't match this. const CurrentVersion = 2 // 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"` } type Config interface { GetPaths(ctx context.Context) *config.Paths } type Database struct { conn *sqlx.DB config Config } func New(config Config) *Database { return &Database{ config: config, } } func (d *Database) Init(ctx context.Context) error { err := d.Connect(ctx) if err != nil { return err } return d.initDB(ctx) } func (d *Database) Connect(ctx context.Context) error { dsn := d.config.GetPaths(ctx).DBPath db, err := sqlx.Open("sqlite", dsn) if err != nil { return err } d.conn = db return nil } func (d *Database) GetConn() *sqlx.DB { return d.conn } func (d *Database) initDB(ctx context.Context) error { log := loggerctx.From(ctx) d.conn = d.conn.Unsafe() conn := d.conn _, 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 := d.GetVersion(ctx) if ok && ver != CurrentVersion { log.Warn("Database version mismatch; resetting").Int("version", ver).Int("expected", CurrentVersion).Send() d.reset(ctx) return d.initDB(ctx) } else if !ok { log.Warn("Database version does not exist. Run alr fix if something isn't working.").Send() return d.addVersion(ctx, CurrentVersion) } return nil } func (d *Database) GetVersion(ctx context.Context) (int, bool) { var ver version err := d.conn.GetContext(ctx, &ver, "SELECT * FROM alr_db_version LIMIT 1;") if err != nil { return 0, false } return ver.Version, true } func (d *Database) addVersion(ctx context.Context, ver int) error { _, err := d.conn.ExecContext(ctx, `INSERT INTO alr_db_version(version) VALUES (?);`, ver) return err } func (d *Database) reset(ctx context.Context) error { _, err := d.conn.ExecContext(ctx, "DROP TABLE IF EXISTS pkgs;") if err != nil { return err } _, err = d.conn.ExecContext(ctx, "DROP TABLE IF EXISTS alr_db_version;") return err } func (d *Database) GetPkgs(ctx context.Context, where string, args ...any) (*sqlx.Rows, error) { stream, err := d.conn.QueryxContext(ctx, "SELECT * FROM pkgs WHERE "+where, args...) if err != nil { return nil, err } return stream, nil } func (d *Database) GetPkg(ctx context.Context, where string, args ...any) (*Package, error) { out := &Package{} err := d.conn.GetContext(ctx, out, "SELECT * FROM pkgs WHERE "+where+" LIMIT 1", args...) return out, err } func (d *Database) DeletePkgs(ctx context.Context, where string, args ...any) error { _, err := d.conn.ExecContext(ctx, "DELETE FROM pkgs WHERE "+where, args...) return err } func (d *Database) IsEmpty(ctx context.Context) bool { var count int err := d.conn.GetContext(ctx, &count, "SELECT count(1) FROM pkgs;") if err != nil { return true } return count == 0 } func (d *Database) InsertPackage(ctx context.Context, pkg Package) error { _, err := d.conn.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 } func (d *Database) Close() error { if d.conn != nil { return d.conn.Close() } else { return nil } }