forked from Plemya-x/ALR
		
	Initial commit
This commit is contained in:
		
							
								
								
									
										346
									
								
								internal/db/db.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										346
									
								
								internal/db/db.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,346 @@
 | 
			
		||||
/*
 | 
			
		||||
 * LURE - Linux User REpository
 | 
			
		||||
 * Copyright (C) 2023 Elara Musayelyan
 | 
			
		||||
 *
 | 
			
		||||
 * 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 <http://www.gnu.org/licenses/>.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
package db
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"database/sql"
 | 
			
		||||
	"database/sql/driver"
 | 
			
		||||
	"encoding/json"
 | 
			
		||||
	"errors"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"sync"
 | 
			
		||||
 | 
			
		||||
	"github.com/jmoiron/sqlx"
 | 
			
		||||
	"lure.sh/lure/internal/config"
 | 
			
		||||
	"lure.sh/lure/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 LURE 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 LURE 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 lure_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 lure 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 lure_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 lure_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 lure_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)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										250
									
								
								internal/db/db_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										250
									
								
								internal/db/db_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,250 @@
 | 
			
		||||
/*
 | 
			
		||||
 * LURE - Linux User REpository
 | 
			
		||||
 * Copyright (C) 2023 Elara Musayelyan
 | 
			
		||||
 *
 | 
			
		||||
 * 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 <http://www.gnu.org/licenses/>.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
package db_test
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"reflect"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"testing"
 | 
			
		||||
 | 
			
		||||
	"github.com/jmoiron/sqlx"
 | 
			
		||||
	"lure.sh/lure/internal/db"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var testPkg = db.Package{
 | 
			
		||||
	Name:    "test",
 | 
			
		||||
	Version: "0.0.1",
 | 
			
		||||
	Release: 1,
 | 
			
		||||
	Epoch:   2,
 | 
			
		||||
	Description: db.NewJSON(map[string]string{
 | 
			
		||||
		"en": "Test package",
 | 
			
		||||
		"ru": "Проверочный пакет",
 | 
			
		||||
	}),
 | 
			
		||||
	Homepage: db.NewJSON(map[string]string{
 | 
			
		||||
		"en": "https://lure.sh/",
 | 
			
		||||
	}),
 | 
			
		||||
	Maintainer: db.NewJSON(map[string]string{
 | 
			
		||||
		"en": "Elara Musayelyan <elara@elara.ws>",
 | 
			
		||||
		"ru": "Элара Мусаелян <elara@elara.ws>",
 | 
			
		||||
	}),
 | 
			
		||||
	Architectures: db.NewJSON([]string{"arm64", "amd64"}),
 | 
			
		||||
	Licenses:      db.NewJSON([]string{"GPL-3.0-or-later"}),
 | 
			
		||||
	Provides:      db.NewJSON([]string{"test"}),
 | 
			
		||||
	Conflicts:     db.NewJSON([]string{"test"}),
 | 
			
		||||
	Replaces:      db.NewJSON([]string{"test-old"}),
 | 
			
		||||
	Depends: db.NewJSON(map[string][]string{
 | 
			
		||||
		"": {"sudo"},
 | 
			
		||||
	}),
 | 
			
		||||
	BuildDepends: db.NewJSON(map[string][]string{
 | 
			
		||||
		"":     {"golang"},
 | 
			
		||||
		"arch": {"go"},
 | 
			
		||||
	}),
 | 
			
		||||
	Repository: "default",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestInit(t *testing.T) {
 | 
			
		||||
	_, err := db.Open(":memory:")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
	defer db.Close()
 | 
			
		||||
 | 
			
		||||
	_, err = db.DB().Exec("SELECT * FROM pkgs")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ver, ok := db.GetVersion()
 | 
			
		||||
	if !ok {
 | 
			
		||||
		t.Errorf("Expected version to be present")
 | 
			
		||||
	} else if ver != db.CurrentVersion {
 | 
			
		||||
		t.Errorf("Expected version %d, got %d", db.CurrentVersion, ver)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestInsertPackage(t *testing.T) {
 | 
			
		||||
	_, err := db.Open(":memory:")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
	defer db.Close()
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(testPkg)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	dbPkg := db.Package{}
 | 
			
		||||
	err = sqlx.Get(db.DB(), &dbPkg, "SELECT * FROM pkgs WHERE name = 'test' AND repository = 'default'")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if !reflect.DeepEqual(testPkg, dbPkg) {
 | 
			
		||||
		t.Errorf("Expected test package to be the same as database package")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestGetPkgs(t *testing.T) {
 | 
			
		||||
	_, err := db.Open(":memory:")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
	defer db.Close()
 | 
			
		||||
 | 
			
		||||
	x1 := testPkg
 | 
			
		||||
	x1.Name = "x1"
 | 
			
		||||
	x2 := testPkg
 | 
			
		||||
	x2.Name = "x2"
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(x1)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(x2)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	result, err := db.GetPkgs("name LIKE 'x%'")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	for result.Next() {
 | 
			
		||||
		var dbPkg db.Package
 | 
			
		||||
		err = result.StructScan(&dbPkg)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if !strings.HasPrefix(dbPkg.Name, "x") {
 | 
			
		||||
			t.Errorf("Expected package name to start with 'x', got %s", dbPkg.Name)
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestGetPkg(t *testing.T) {
 | 
			
		||||
	_, err := db.Open(":memory:")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
	defer db.Close()
 | 
			
		||||
 | 
			
		||||
	x1 := testPkg
 | 
			
		||||
	x1.Name = "x1"
 | 
			
		||||
	x2 := testPkg
 | 
			
		||||
	x2.Name = "x2"
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(x1)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(x2)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	pkg, err := db.GetPkg("name LIKE 'x%' ORDER BY name")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if pkg.Name != "x1" {
 | 
			
		||||
		t.Errorf("Expected x1 package, got %s", pkg.Name)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if !reflect.DeepEqual(*pkg, x1) {
 | 
			
		||||
		t.Errorf("Expected x1 to be %v, got %v", x1, *pkg)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestDeletePkgs(t *testing.T) {
 | 
			
		||||
	_, err := db.Open(":memory:")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
	defer db.Close()
 | 
			
		||||
 | 
			
		||||
	x1 := testPkg
 | 
			
		||||
	x1.Name = "x1"
 | 
			
		||||
	x2 := testPkg
 | 
			
		||||
	x2.Name = "x2"
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(x1)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(x2)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err = db.DeletePkgs("name = 'x1'")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var dbPkg db.Package
 | 
			
		||||
	err = db.DB().Get(&dbPkg, "SELECT * FROM pkgs WHERE name LIKE 'x%' ORDER BY name LIMIT 1;")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if dbPkg.Name != "x2" {
 | 
			
		||||
		t.Errorf("Expected x2 package, got %s", dbPkg.Name)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestJsonArrayContains(t *testing.T) {
 | 
			
		||||
	_, err := db.Open(":memory:")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
	defer db.Close()
 | 
			
		||||
 | 
			
		||||
	x1 := testPkg
 | 
			
		||||
	x1.Name = "x1"
 | 
			
		||||
	x2 := testPkg
 | 
			
		||||
	x2.Name = "x2"
 | 
			
		||||
	x2.Provides.Val = append(x2.Provides.Val, "x")
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(x1)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err = db.InsertPackage(x2)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Errorf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var dbPkg db.Package
 | 
			
		||||
	err = db.DB().Get(&dbPkg, "SELECT * FROM pkgs WHERE json_array_contains(provides, 'x');")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Expected no error, got %s", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if dbPkg.Name != "x2" {
 | 
			
		||||
		t.Errorf("Expected x2 package, got %s", dbPkg.Name)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user