2019-10-28 16:12:29 +00:00
|
|
|
// Copyright © 2019 Ettore Di Giacinto <mudler@gentoo.org>
|
|
|
|
//
|
|
|
|
// 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 2 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 pkg
|
|
|
|
|
|
|
|
import (
|
2019-11-25 19:03:37 +00:00
|
|
|
"encoding/base64"
|
2019-10-30 16:52:57 +00:00
|
|
|
"os"
|
2019-10-28 16:12:29 +00:00
|
|
|
"strconv"
|
2019-11-02 16:56:43 +00:00
|
|
|
"sync"
|
|
|
|
"time"
|
2019-10-28 16:12:29 +00:00
|
|
|
|
2019-11-04 16:13:53 +00:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
2019-10-28 16:12:29 +00:00
|
|
|
storm "github.com/asdine/storm"
|
|
|
|
"github.com/asdine/storm/q"
|
2019-11-02 16:56:43 +00:00
|
|
|
"go.etcd.io/bbolt"
|
2019-10-28 16:12:29 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
//var BoltInstance PackageDatabase
|
|
|
|
|
|
|
|
type BoltDatabase struct {
|
2019-11-02 16:56:43 +00:00
|
|
|
sync.Mutex
|
2019-12-17 18:32:31 +00:00
|
|
|
Path string
|
|
|
|
ProvidesDatabase map[string]map[string]Package
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewBoltDatabase(path string) PackageDatabase {
|
|
|
|
// if BoltInstance == nil {
|
|
|
|
// BoltInstance = &BoltDatabase{Path: path}
|
|
|
|
// }
|
|
|
|
//return BoltInstance, nil
|
2019-12-17 18:32:31 +00:00
|
|
|
return &BoltDatabase{Path: path, ProvidesDatabase: map[string]map[string]Package{}}
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *BoltDatabase) Get(s string) (string, error) {
|
2019-11-25 19:03:37 +00:00
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
var str string
|
|
|
|
bolt.Get("solver", s, &str)
|
|
|
|
|
|
|
|
return str, errors.New("Not implemented")
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *BoltDatabase) Set(k, v string) error {
|
2019-11-25 19:03:37 +00:00
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
return bolt.Set("solver", k, v)
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
2019-12-03 22:38:25 +00:00
|
|
|
func (db *BoltDatabase) Create(id string, v []byte) (string, error) {
|
2019-11-25 19:03:37 +00:00
|
|
|
enc := base64.StdEncoding.EncodeToString(v)
|
|
|
|
|
2019-12-03 22:38:25 +00:00
|
|
|
return id, db.Set(id, enc)
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *BoltDatabase) Retrieve(ID string) ([]byte, error) {
|
2019-11-25 19:03:37 +00:00
|
|
|
pa, err := db.Get(ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
enc, err := base64.StdEncoding.DecodeString(pa)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return enc, nil
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
2019-10-30 16:52:57 +00:00
|
|
|
func (db *BoltDatabase) FindPackage(tofind Package) (Package, error) {
|
2019-12-17 18:32:31 +00:00
|
|
|
// Provides: Return the replaced package here
|
|
|
|
if provided, err := db.getProvide(tofind); err == nil {
|
|
|
|
return provided, nil
|
|
|
|
}
|
|
|
|
|
2019-10-28 16:12:29 +00:00
|
|
|
p := &DefaultPackage{}
|
2019-11-02 16:56:43 +00:00
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
2019-10-28 16:12:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
|
2019-10-30 16:52:57 +00:00
|
|
|
err = bolt.Select(q.Eq("Name", tofind.GetName()), q.Eq("Category", tofind.GetCategory()), q.Eq("Version", tofind.GetVersion())).Limit(1).First(p)
|
2019-10-28 16:12:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return p, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *BoltDatabase) UpdatePackage(p Package) error {
|
2019-11-25 19:03:37 +00:00
|
|
|
// TODO: Change, but by query we cannot update by ID
|
|
|
|
err := db.RemovePackage(p)
|
2019-10-28 16:12:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-11-25 19:03:37 +00:00
|
|
|
_, err = db.CreatePackage(p)
|
2019-10-28 16:12:29 +00:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-11-02 16:56:43 +00:00
|
|
|
return nil
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *BoltDatabase) GetPackage(ID string) (Package, error) {
|
|
|
|
p := &DefaultPackage{}
|
2019-11-02 16:56:43 +00:00
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
2019-10-28 16:12:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
iid, err := strconv.Atoi(ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-11-25 19:03:37 +00:00
|
|
|
err = bolt.Select(q.Eq("ID", iid)).Limit(1).First(p)
|
|
|
|
|
|
|
|
//err = bolt.One("id", iid, p)
|
2019-10-28 16:12:29 +00:00
|
|
|
return p, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *BoltDatabase) GetPackages() []string {
|
|
|
|
ids := []string{}
|
2019-11-02 16:56:43 +00:00
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
2019-10-28 16:12:29 +00:00
|
|
|
if err != nil {
|
|
|
|
return []string{}
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
// Fetching records one by one (useful when the bucket contains a lot of records)
|
|
|
|
query := bolt.Select()
|
|
|
|
|
|
|
|
query.Each(new(DefaultPackage), func(record interface{}) error {
|
|
|
|
u := record.(*DefaultPackage)
|
|
|
|
ids = append(ids, strconv.Itoa(u.ID))
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return ids
|
|
|
|
}
|
|
|
|
|
2019-11-01 15:26:03 +00:00
|
|
|
func (db *BoltDatabase) GetAllPackages(packages chan Package) error {
|
2019-11-02 16:56:43 +00:00
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
2019-11-01 15:26:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
// Fetching records one by one (useful when the bucket contains a lot of records)
|
2019-11-02 16:56:43 +00:00
|
|
|
//query := bolt.Select()
|
2019-11-01 15:26:03 +00:00
|
|
|
|
2019-11-02 16:56:43 +00:00
|
|
|
var packs []Package
|
|
|
|
err = bolt.All(&packs)
|
|
|
|
if err != nil {
|
2019-11-01 15:26:03 +00:00
|
|
|
return err
|
2019-11-02 16:56:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, r := range packs {
|
|
|
|
packages <- r
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
// return query.Each(new(DefaultPackage), func(record interface{}) error {
|
|
|
|
// u := record.(*DefaultPackage)
|
|
|
|
// packages <- u
|
|
|
|
// return err
|
|
|
|
// })
|
2019-11-01 15:26:03 +00:00
|
|
|
}
|
|
|
|
|
2019-10-28 16:12:29 +00:00
|
|
|
// Encode encodes the package to string.
|
|
|
|
// It returns an ID which can be used to retrieve the package later on.
|
|
|
|
func (db *BoltDatabase) CreatePackage(p Package) (string, error) {
|
2019-11-02 16:56:43 +00:00
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
2019-10-28 16:12:29 +00:00
|
|
|
if err != nil {
|
2019-11-04 16:13:53 +00:00
|
|
|
return "", errors.Wrap(err, "Error opening boltdb "+db.Path)
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
|
|
|
|
dp, ok := p.(*DefaultPackage)
|
|
|
|
if !ok {
|
|
|
|
return "", errors.New("Bolt DB support only DefaultPackage type for now")
|
|
|
|
}
|
2019-11-04 16:13:53 +00:00
|
|
|
|
2019-10-28 16:12:29 +00:00
|
|
|
err = bolt.Save(dp)
|
|
|
|
if err != nil {
|
2019-11-04 16:13:53 +00:00
|
|
|
return "", errors.Wrap(err, "Error saving package to "+db.Path)
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
2019-12-17 18:32:31 +00:00
|
|
|
// Create extra cache between package -> []versions
|
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
|
|
|
// TODO: Replace with a bolt implementation (and not in memory)
|
|
|
|
// Provides: Store package provides, we will reuse this when walking deps
|
|
|
|
for _, provide := range dp.Provides {
|
|
|
|
if _, ok := db.ProvidesDatabase[provide.GetPackageName()]; !ok {
|
|
|
|
db.ProvidesDatabase[provide.GetPackageName()] = make(map[string]Package)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
db.ProvidesDatabase[provide.GetPackageName()][provide.GetVersion()] = p
|
|
|
|
}
|
|
|
|
|
2019-10-28 16:12:29 +00:00
|
|
|
return strconv.Itoa(dp.ID), err
|
|
|
|
}
|
2019-10-30 16:52:57 +00:00
|
|
|
|
2019-12-17 18:32:31 +00:00
|
|
|
// Dup from memory implementation
|
|
|
|
func (db *BoltDatabase) getProvide(p Package) (Package, error) {
|
|
|
|
db.Lock()
|
|
|
|
pa, ok := db.ProvidesDatabase[p.GetPackageName()][p.GetVersion()]
|
|
|
|
if !ok {
|
|
|
|
versions, ok := db.ProvidesDatabase[p.GetPackageName()]
|
|
|
|
db.Unlock()
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("No versions found for package")
|
|
|
|
}
|
|
|
|
|
|
|
|
for ve, _ := range versions {
|
|
|
|
|
2019-12-27 12:02:58 +00:00
|
|
|
match, err := p.VersionMatchSelector(ve)
|
2019-12-17 18:32:31 +00:00
|
|
|
if err != nil {
|
2019-12-27 12:02:58 +00:00
|
|
|
return nil, errors.Wrap(err, "Error on match version")
|
2019-12-17 18:32:31 +00:00
|
|
|
}
|
2019-12-27 12:02:58 +00:00
|
|
|
if match {
|
2019-12-17 18:32:31 +00:00
|
|
|
pa, ok := db.ProvidesDatabase[p.GetPackageName()][ve]
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("No versions found for package")
|
|
|
|
}
|
|
|
|
return pa, nil //pick the first (we shouldn't have providers that are conflicting)
|
|
|
|
// TODO: A find dbcall here would recurse, but would give chance to have providers of providers
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, errors.New("No package provides this")
|
|
|
|
}
|
|
|
|
db.Unlock()
|
|
|
|
return db.FindPackage(pa)
|
|
|
|
}
|
|
|
|
|
2019-10-30 16:52:57 +00:00
|
|
|
func (db *BoltDatabase) Clean() error {
|
2019-11-02 16:56:43 +00:00
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
2019-10-30 16:52:57 +00:00
|
|
|
return os.RemoveAll(db.Path)
|
|
|
|
}
|
2019-11-23 21:41:51 +00:00
|
|
|
|
|
|
|
func (db *BoltDatabase) GetPackageFiles(p Package) ([]string, error) {
|
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
|
|
|
if err != nil {
|
|
|
|
return []string{}, errors.Wrap(err, "Error opening boltdb "+db.Path)
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
|
|
|
|
files := bolt.From("files")
|
|
|
|
var pf PackageFile
|
|
|
|
err = files.One("PackageFingerprint", p.GetFingerPrint(), &pf)
|
|
|
|
if err != nil {
|
|
|
|
return []string{}, errors.Wrap(err, "While finding files")
|
|
|
|
}
|
|
|
|
return pf.Files, nil
|
|
|
|
}
|
2019-11-25 19:03:37 +00:00
|
|
|
func (db *BoltDatabase) SetPackageFiles(p *PackageFile) error {
|
2019-11-23 21:41:51 +00:00
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Error opening boltdb "+db.Path)
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
|
|
|
|
files := bolt.From("files")
|
|
|
|
return files.Save(p)
|
|
|
|
}
|
|
|
|
func (db *BoltDatabase) RemovePackageFiles(p Package) error {
|
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Error opening boltdb "+db.Path)
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
|
|
|
|
|
|
|
files := bolt.From("files")
|
|
|
|
var pf PackageFile
|
|
|
|
err = files.One("PackageFingerprint", p.GetFingerPrint(), &pf)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "While finding files")
|
|
|
|
}
|
|
|
|
return files.DeleteStruct(&pf)
|
|
|
|
}
|
2019-11-23 23:16:12 +00:00
|
|
|
|
|
|
|
func (db *BoltDatabase) RemovePackage(p Package) error {
|
|
|
|
bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second}))
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Error opening boltdb "+db.Path)
|
|
|
|
}
|
|
|
|
defer bolt.Close()
|
2019-11-25 19:03:37 +00:00
|
|
|
var found DefaultPackage
|
|
|
|
err = bolt.Select(q.Eq("Name", p.GetName()), q.Eq("Category", p.GetCategory()), q.Eq("Version", p.GetVersion())).Limit(1).Delete(&found)
|
2019-11-23 23:16:12 +00:00
|
|
|
if err != nil {
|
2019-11-25 19:03:37 +00:00
|
|
|
return errors.Wrap(err, "No package found to delete")
|
2019-11-23 23:16:12 +00:00
|
|
|
}
|
2019-11-25 19:03:37 +00:00
|
|
|
return nil
|
2019-11-23 23:16:12 +00:00
|
|
|
}
|
2019-11-29 18:01:48 +00:00
|
|
|
|
|
|
|
func (db *BoltDatabase) World() []Package {
|
|
|
|
|
|
|
|
var all []Package
|
|
|
|
// FIXME: This should all be locked in the db - for now forbid the solver to be run in threads.
|
|
|
|
for _, k := range db.GetPackages() {
|
|
|
|
pack, err := db.GetPackage(k)
|
|
|
|
if err == nil {
|
|
|
|
all = append(all, pack)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return all
|
|
|
|
}
|
2019-12-01 18:11:19 +00:00
|
|
|
|
|
|
|
func (db *BoltDatabase) FindPackageCandidate(p Package) (Package, error) {
|
|
|
|
|
|
|
|
required, err := db.FindPackage(p)
|
|
|
|
if err != nil {
|
2019-12-06 15:28:42 +00:00
|
|
|
|
2019-12-01 18:11:19 +00:00
|
|
|
// return nil, errors.Wrap(err, "Couldn't find required package in db definition")
|
2019-12-06 15:28:42 +00:00
|
|
|
packages, err := p.Expand(db)
|
2019-12-01 18:11:19 +00:00
|
|
|
// Info("Expanded", packages, err)
|
|
|
|
if err != nil || len(packages) == 0 {
|
|
|
|
required = p
|
|
|
|
} else {
|
|
|
|
required = Best(packages)
|
|
|
|
|
|
|
|
}
|
|
|
|
return required, nil
|
|
|
|
//required = &DefaultPackage{Name: "test"}
|
|
|
|
}
|
|
|
|
|
|
|
|
return required, err
|
|
|
|
|
|
|
|
}
|
2019-12-06 15:29:15 +00:00
|
|
|
|
2019-12-13 16:18:26 +00:00
|
|
|
// FindPackages return the list of the packages beloging to cat/name (any versions in requested range)
|
2019-12-06 15:29:15 +00:00
|
|
|
// FIXME: Optimize, see inmemorydb
|
|
|
|
func (db *BoltDatabase) FindPackages(p Package) ([]Package, error) {
|
2019-12-17 18:32:31 +00:00
|
|
|
// Provides: Treat as the replaced package here
|
|
|
|
if provided, err := db.getProvide(p); err == nil {
|
|
|
|
p = provided
|
|
|
|
}
|
2019-12-06 15:29:15 +00:00
|
|
|
var versionsInWorld []Package
|
|
|
|
for _, w := range db.World() {
|
|
|
|
if w.GetName() != p.GetName() || w.GetCategory() != p.GetCategory() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2019-12-27 12:02:58 +00:00
|
|
|
match, err := p.SelectorMatchVersion(w.GetVersion())
|
2019-12-06 15:29:15 +00:00
|
|
|
if err != nil {
|
2019-12-27 12:02:58 +00:00
|
|
|
return nil, errors.Wrap(err, "Error on match selector")
|
2019-12-06 15:29:15 +00:00
|
|
|
}
|
2019-12-27 12:02:58 +00:00
|
|
|
if match {
|
2019-12-06 15:29:15 +00:00
|
|
|
versionsInWorld = append(versionsInWorld, w)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return versionsInWorld, nil
|
|
|
|
}
|
2019-12-13 16:18:26 +00:00
|
|
|
|
|
|
|
// FindPackageVersions return the list of the packages beloging to cat/name
|
|
|
|
func (db *BoltDatabase) FindPackageVersions(p Package) ([]Package, error) {
|
|
|
|
var versionsInWorld []Package
|
|
|
|
for _, w := range db.World() {
|
|
|
|
if w.GetName() != p.GetName() || w.GetCategory() != p.GetCategory() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
versionsInWorld = append(versionsInWorld, w)
|
|
|
|
}
|
|
|
|
return versionsInWorld, nil
|
|
|
|
}
|