2019-06-11 21:04:01 +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 (
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/json"
|
2020-04-30 18:44:34 +00:00
|
|
|
"fmt"
|
2020-03-20 22:47:35 +00:00
|
|
|
"regexp"
|
2019-11-02 16:56:43 +00:00
|
|
|
"sync"
|
2019-12-06 15:29:15 +00:00
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2019-06-11 21:04:01 +00:00
|
|
|
)
|
|
|
|
|
2019-11-15 23:38:07 +00:00
|
|
|
var DBInMemoryInstance = &InMemoryDatabase{
|
2019-12-17 18:32:31 +00:00
|
|
|
Mutex: &sync.Mutex{},
|
|
|
|
FileDatabase: map[string][]string{},
|
|
|
|
Database: map[string]string{},
|
|
|
|
CacheNoVersion: map[string]map[string]interface{}{},
|
|
|
|
ProvidesDatabase: map[string]map[string]Package{},
|
2020-11-20 16:23:21 +00:00
|
|
|
RevDepsDatabase: map[string]map[string]Package{},
|
2019-12-06 15:29:15 +00:00
|
|
|
}
|
2019-06-11 21:04:01 +00:00
|
|
|
|
|
|
|
type InMemoryDatabase struct {
|
2019-11-02 16:56:43 +00:00
|
|
|
*sync.Mutex
|
2019-12-17 18:32:31 +00:00
|
|
|
Database map[string]string
|
|
|
|
FileDatabase map[string][]string
|
|
|
|
CacheNoVersion map[string]map[string]interface{}
|
|
|
|
ProvidesDatabase map[string]map[string]Package
|
2020-11-20 16:23:21 +00:00
|
|
|
RevDepsDatabase map[string]map[string]Package
|
2019-06-11 21:04:01 +00:00
|
|
|
}
|
|
|
|
|
2019-11-02 16:56:43 +00:00
|
|
|
func NewInMemoryDatabase(singleton bool) PackageDatabase {
|
2019-06-11 21:04:01 +00:00
|
|
|
// In memoryDB is a singleton
|
2019-11-02 16:56:43 +00:00
|
|
|
if !singleton {
|
|
|
|
return &InMemoryDatabase{
|
2019-12-17 18:32:31 +00:00
|
|
|
Mutex: &sync.Mutex{},
|
|
|
|
FileDatabase: map[string][]string{},
|
|
|
|
Database: map[string]string{},
|
|
|
|
CacheNoVersion: map[string]map[string]interface{}{},
|
|
|
|
ProvidesDatabase: map[string]map[string]Package{},
|
2020-11-20 16:23:21 +00:00
|
|
|
RevDepsDatabase: map[string]map[string]Package{},
|
2019-12-06 15:29:15 +00:00
|
|
|
}
|
2019-06-11 21:04:01 +00:00
|
|
|
}
|
|
|
|
return DBInMemoryInstance
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *InMemoryDatabase) Get(s string) (string, error) {
|
2019-11-02 16:56:43 +00:00
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
2019-06-11 21:04:01 +00:00
|
|
|
pa, ok := db.Database[s]
|
|
|
|
if !ok {
|
2020-04-30 18:44:34 +00:00
|
|
|
return "", errors.New(fmt.Sprintf("No key found for: %s", s))
|
2019-06-11 21:04:01 +00:00
|
|
|
}
|
|
|
|
return pa, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *InMemoryDatabase) Set(k, v string) error {
|
2019-11-02 16:56:43 +00:00
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
2019-06-11 21:04:01 +00:00
|
|
|
db.Database[k] = v
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-12-03 22:38:25 +00:00
|
|
|
func (db *InMemoryDatabase) Create(id string, v []byte) (string, error) {
|
2019-06-11 21:04:01 +00:00
|
|
|
enc := base64.StdEncoding.EncodeToString(v)
|
|
|
|
|
2019-12-03 22:38:25 +00:00
|
|
|
return id, db.Set(id, enc)
|
2019-06-11 21:04:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *InMemoryDatabase) Retrieve(ID string) ([]byte, error) {
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *InMemoryDatabase) GetPackage(ID string) (Package, error) {
|
|
|
|
|
|
|
|
enc, err := db.Retrieve(ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
p := &DefaultPackage{}
|
|
|
|
|
2019-11-29 18:01:50 +00:00
|
|
|
rawIn := json.RawMessage(enc)
|
|
|
|
bytes, err := rawIn.MarshalJSON()
|
|
|
|
if err != nil {
|
|
|
|
return &DefaultPackage{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(bytes, &p); err != nil {
|
2019-06-11 21:04:01 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return p, nil
|
|
|
|
}
|
|
|
|
|
2019-11-01 15:26:03 +00:00
|
|
|
func (db *InMemoryDatabase) GetAllPackages(packages chan Package) error {
|
2019-11-02 16:56:43 +00:00
|
|
|
packs := db.GetPackages()
|
|
|
|
for _, p := range packs {
|
|
|
|
pack, err := db.GetPackage(p)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
packages <- pack
|
|
|
|
}
|
|
|
|
return nil
|
2019-11-01 15:26:03 +00:00
|
|
|
}
|
|
|
|
|
2020-11-20 16:23:21 +00:00
|
|
|
func (db *InMemoryDatabase) getRevdeps(p Package, visited map[string]interface{}) (Packages, error) {
|
|
|
|
var versionsInWorld Packages
|
|
|
|
if _, ok := visited[p.HumanReadableString()]; ok {
|
|
|
|
return versionsInWorld, nil
|
|
|
|
}
|
|
|
|
visited[p.HumanReadableString()] = true
|
|
|
|
|
|
|
|
var res Packages
|
|
|
|
packs, err := db.FindPackages(p)
|
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
|
|
|
for _, pp := range packs {
|
|
|
|
// db.Lock()
|
|
|
|
list := db.RevDepsDatabase[pp.GetFingerPrint()]
|
|
|
|
// db.Unlock()
|
|
|
|
for _, revdep := range list {
|
|
|
|
dep, err := db.FindPackage(revdep)
|
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
|
|
|
res = append(res, dep)
|
|
|
|
|
|
|
|
packs, err := db.getRevdeps(dep, visited)
|
|
|
|
if err != nil {
|
|
|
|
return res, err
|
|
|
|
}
|
|
|
|
res = append(res, packs...)
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res.Unique(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetRevdeps returns the package reverse dependencies,
|
|
|
|
// matching also selectors in versions (>, <, >=, <=)
|
|
|
|
// TODO: Code should use db explictly
|
|
|
|
func (db *InMemoryDatabase) GetRevdeps(p Package) (Packages, error) {
|
|
|
|
return db.getRevdeps(p, make(map[string]interface{}))
|
|
|
|
}
|
|
|
|
|
2019-06-11 21:04:01 +00:00
|
|
|
// Encode encodes the package to string.
|
|
|
|
// It returns an ID which can be used to retrieve the package later on.
|
|
|
|
func (db *InMemoryDatabase) CreatePackage(p Package) (string, error) {
|
|
|
|
pd, ok := p.(*DefaultPackage)
|
|
|
|
if !ok {
|
|
|
|
return "", errors.New("InMemoryDatabase suports only DefaultPackage")
|
|
|
|
}
|
|
|
|
|
2019-11-29 18:01:50 +00:00
|
|
|
res, err := pd.JSON()
|
2019-06-11 21:04:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2019-12-03 22:38:25 +00:00
|
|
|
ID, err := db.Create(pd.GetFingerPrint(), res)
|
2019-06-11 21:04:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2019-12-06 15:29:15 +00:00
|
|
|
|
2020-11-20 16:23:21 +00:00
|
|
|
db.populateCaches(pd)
|
|
|
|
|
|
|
|
return ID, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *InMemoryDatabase) populateCaches(p Package) {
|
|
|
|
pd, _ := p.(*DefaultPackage)
|
|
|
|
|
2019-12-06 15:29:15 +00:00
|
|
|
// Create extra cache between package -> []versions
|
|
|
|
db.Lock()
|
2019-12-17 18:32:31 +00:00
|
|
|
|
|
|
|
// Provides: Store package provides, we will reuse this when walking deps
|
|
|
|
for _, provide := range pd.Provides {
|
|
|
|
if _, ok := db.ProvidesDatabase[provide.GetPackageName()]; !ok {
|
|
|
|
db.ProvidesDatabase[provide.GetPackageName()] = make(map[string]Package)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
db.ProvidesDatabase[provide.GetPackageName()][provide.GetVersion()] = p
|
|
|
|
}
|
|
|
|
|
2020-11-20 16:23:21 +00:00
|
|
|
_, ok := db.CacheNoVersion[p.GetPackageName()]
|
2019-12-06 15:29:15 +00:00
|
|
|
if !ok {
|
|
|
|
db.CacheNoVersion[p.GetPackageName()] = make(map[string]interface{})
|
|
|
|
}
|
|
|
|
db.CacheNoVersion[p.GetPackageName()][p.GetVersion()] = nil
|
2020-12-30 00:12:35 +00:00
|
|
|
|
|
|
|
// Updating Revdeps
|
|
|
|
// Given that when we populate the cache we don't have the full db at hand
|
|
|
|
// We cycle over reverse dependency of a package to update their entry if they are matching
|
|
|
|
// the version selector
|
|
|
|
toUpdate, ok := db.RevDepsDatabase[pd.GetPackageName()]
|
|
|
|
if ok {
|
|
|
|
for _, pp := range toUpdate {
|
|
|
|
for _, re := range pp.GetRequires() {
|
|
|
|
if match, _ := pd.VersionMatchSelector(re.GetVersion(), nil); match {
|
|
|
|
_, ok = db.RevDepsDatabase[pd.GetFingerPrint()]
|
|
|
|
if !ok {
|
|
|
|
db.RevDepsDatabase[pd.GetFingerPrint()] = make(map[string]Package)
|
|
|
|
}
|
|
|
|
db.RevDepsDatabase[pd.GetFingerPrint()][pp.GetFingerPrint()] = pp
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-11-20 16:23:21 +00:00
|
|
|
db.Unlock()
|
2019-12-06 15:29:15 +00:00
|
|
|
|
2020-11-20 16:23:21 +00:00
|
|
|
for _, re := range pd.GetRequires() {
|
|
|
|
packages, _ := db.FindPackages(re)
|
|
|
|
db.Lock()
|
|
|
|
|
|
|
|
for _, pa := range packages {
|
|
|
|
_, ok := db.RevDepsDatabase[pa.GetFingerPrint()]
|
|
|
|
if !ok {
|
|
|
|
db.RevDepsDatabase[pa.GetFingerPrint()] = make(map[string]Package)
|
|
|
|
}
|
|
|
|
db.RevDepsDatabase[pa.GetFingerPrint()][pd.GetFingerPrint()] = pd
|
2020-12-30 00:12:35 +00:00
|
|
|
_, ok = db.RevDepsDatabase[pa.GetPackageName()]
|
|
|
|
if !ok {
|
|
|
|
db.RevDepsDatabase[pa.GetPackageName()] = make(map[string]Package)
|
|
|
|
}
|
|
|
|
db.RevDepsDatabase[pa.GetPackageName()][pd.GetPackageName()] = pd
|
2020-11-20 16:23:21 +00:00
|
|
|
}
|
|
|
|
_, ok := db.RevDepsDatabase[re.GetFingerPrint()]
|
|
|
|
if !ok {
|
|
|
|
db.RevDepsDatabase[re.GetFingerPrint()] = make(map[string]Package)
|
|
|
|
}
|
|
|
|
db.RevDepsDatabase[re.GetFingerPrint()][pd.GetFingerPrint()] = pd
|
2020-12-30 00:12:35 +00:00
|
|
|
_, ok = db.RevDepsDatabase[re.GetPackageName()]
|
|
|
|
if !ok {
|
|
|
|
db.RevDepsDatabase[re.GetPackageName()] = make(map[string]Package)
|
|
|
|
}
|
|
|
|
db.RevDepsDatabase[re.GetPackageName()][pd.GetPackageName()] = pd
|
|
|
|
|
2020-11-20 16:23:21 +00:00
|
|
|
db.Unlock()
|
|
|
|
}
|
2019-06-11 21:04:01 +00:00
|
|
|
}
|
2019-10-28 16:12:29 +00:00
|
|
|
|
2019-12-17 18:32:31 +00:00
|
|
|
func (db *InMemoryDatabase) getProvide(p Package) (Package, error) {
|
2020-11-20 16:23:21 +00:00
|
|
|
|
2019-12-17 18:32:31 +00:00
|
|
|
db.Lock()
|
2020-11-20 16:23:21 +00:00
|
|
|
|
2019-12-17 18:32:31 +00:00
|
|
|
pa, ok := db.ProvidesDatabase[p.GetPackageName()][p.GetVersion()]
|
|
|
|
if !ok {
|
|
|
|
versions, ok := db.ProvidesDatabase[p.GetPackageName()]
|
2020-11-20 16:23:21 +00:00
|
|
|
defer db.Unlock()
|
2019-12-17 18:32:31 +00:00
|
|
|
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("No versions found for package")
|
|
|
|
}
|
|
|
|
|
|
|
|
for ve, _ := range versions {
|
|
|
|
|
2020-04-05 13:09:51 +00:00
|
|
|
match, err := p.VersionMatchSelector(ve, nil)
|
2019-12-17 18:32:31 +00:00
|
|
|
if err != nil {
|
2019-12-27 10:18:52 +00:00
|
|
|
return nil, errors.Wrap(err, "Error on match version")
|
2019-12-17 18:32:31 +00:00
|
|
|
}
|
2019-12-27 10:18:52 +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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, errors.New("No package provides this")
|
|
|
|
}
|
|
|
|
db.Unlock()
|
2020-11-20 16:23:21 +00:00
|
|
|
|
2019-12-17 18:32:31 +00:00
|
|
|
return db.FindPackage(pa)
|
|
|
|
}
|
|
|
|
|
2020-12-19 16:45:50 +00:00
|
|
|
func (db *InMemoryDatabase) Clone(to PackageDatabase) error {
|
|
|
|
return clone(db, to)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (db *InMemoryDatabase) Copy() (PackageDatabase, error) {
|
|
|
|
return copy(db)
|
|
|
|
}
|
|
|
|
|
2019-11-02 16:56:43 +00:00
|
|
|
func (db *InMemoryDatabase) encodePackage(p Package) (string, string, error) {
|
|
|
|
pd, ok := p.(*DefaultPackage)
|
|
|
|
if !ok {
|
|
|
|
return "", "", errors.New("InMemoryDatabase suports only DefaultPackage")
|
|
|
|
}
|
|
|
|
|
2019-11-29 18:01:50 +00:00
|
|
|
res, err := pd.JSON()
|
2019-11-02 16:56:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
enc := base64.StdEncoding.EncodeToString(res)
|
|
|
|
|
2019-12-03 22:38:25 +00:00
|
|
|
return p.GetFingerPrint(), enc, nil
|
2019-11-02 16:56:43 +00:00
|
|
|
}
|
|
|
|
|
2019-10-30 16:52:57 +00:00
|
|
|
func (db *InMemoryDatabase) FindPackage(p Package) (Package, error) {
|
2019-12-17 18:32:31 +00:00
|
|
|
|
|
|
|
// Provides: Return the replaced package here
|
|
|
|
if provided, err := db.getProvide(p); err == nil {
|
|
|
|
return provided, nil
|
|
|
|
}
|
|
|
|
|
2019-12-03 22:38:25 +00:00
|
|
|
return db.GetPackage(p.GetFingerPrint())
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
2019-12-13 16:18:26 +00:00
|
|
|
// FindPackages return the list of the packages beloging to cat/name
|
2020-04-04 12:29:08 +00:00
|
|
|
func (db *InMemoryDatabase) FindPackageVersions(p Package) (Packages, error) {
|
2020-11-05 19:52:02 +00:00
|
|
|
// Provides: Treat as the replaced package here
|
|
|
|
if provided, err := db.getProvide(p); err == nil {
|
|
|
|
p = provided
|
|
|
|
}
|
2020-11-20 16:23:21 +00:00
|
|
|
db.Lock()
|
2019-12-13 16:18:26 +00:00
|
|
|
versions, ok := db.CacheNoVersion[p.GetPackageName()]
|
2020-11-20 16:23:21 +00:00
|
|
|
db.Unlock()
|
2019-12-13 16:18:26 +00:00
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("No versions found for package")
|
|
|
|
}
|
|
|
|
var versionsInWorld []Package
|
|
|
|
for ve, _ := range versions {
|
|
|
|
w, err := db.FindPackage(&DefaultPackage{Name: p.GetName(), Category: p.GetCategory(), Version: ve})
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "Cache mismatch - this shouldn't happen")
|
|
|
|
}
|
|
|
|
versionsInWorld = append(versionsInWorld, w)
|
|
|
|
}
|
2020-04-04 12:29:08 +00:00
|
|
|
return Packages(versionsInWorld), nil
|
2019-12-13 16:18:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// FindPackages return the list of the packages beloging to cat/name (any versions in requested range)
|
2020-04-04 12:29:08 +00:00
|
|
|
func (db *InMemoryDatabase) FindPackages(p Package) (Packages, error) {
|
2020-11-20 16:23:21 +00:00
|
|
|
if !p.IsSelector() {
|
|
|
|
pack, err := db.FindPackage(p)
|
|
|
|
if err != nil {
|
|
|
|
return []Package{}, err
|
|
|
|
}
|
|
|
|
return []Package{pack}, nil
|
|
|
|
}
|
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
|
2020-12-16 21:17:13 +00:00
|
|
|
if !provided.IsSelector() {
|
|
|
|
return Packages{provided}, nil
|
|
|
|
}
|
2019-12-17 18:32:31 +00:00
|
|
|
}
|
2020-11-20 16:23:21 +00:00
|
|
|
|
|
|
|
db.Lock()
|
|
|
|
var matches []*DefaultPackage
|
2019-12-06 15:29:15 +00:00
|
|
|
versions, ok := db.CacheNoVersion[p.GetPackageName()]
|
2020-11-20 16:23:21 +00:00
|
|
|
for ve := range versions {
|
|
|
|
match, _ := p.SelectorMatchVersion(ve, nil)
|
|
|
|
if match {
|
|
|
|
matches = append(matches, &DefaultPackage{Name: p.GetName(), Category: p.GetCategory(), Version: ve})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
db.Unlock()
|
2019-12-06 15:29:15 +00:00
|
|
|
if !ok {
|
2020-04-30 18:44:34 +00:00
|
|
|
return nil, errors.New(fmt.Sprintf("No versions found for: %s", p.HumanReadableString()))
|
2019-12-06 15:29:15 +00:00
|
|
|
}
|
|
|
|
var versionsInWorld []Package
|
2020-11-20 16:23:21 +00:00
|
|
|
for _, p := range matches {
|
|
|
|
w, err := db.FindPackage(p)
|
2019-12-06 15:29:15 +00:00
|
|
|
if err != nil {
|
2020-11-20 16:23:21 +00:00
|
|
|
return nil, errors.Wrap(err, "Cache mismatch - this shouldn't happen")
|
2019-12-06 15:29:15 +00:00
|
|
|
}
|
2020-11-20 16:23:21 +00:00
|
|
|
versionsInWorld = append(versionsInWorld, w)
|
2019-12-06 15:29:15 +00:00
|
|
|
}
|
2020-04-04 12:29:08 +00:00
|
|
|
return Packages(versionsInWorld), nil
|
2019-12-06 15:29:15 +00:00
|
|
|
}
|
|
|
|
|
2019-10-28 16:12:29 +00:00
|
|
|
func (db *InMemoryDatabase) UpdatePackage(p Package) error {
|
2019-12-03 22:38:25 +00:00
|
|
|
|
|
|
|
_, enc, err := db.encodePackage(p)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-11-02 16:56:43 +00:00
|
|
|
}
|
|
|
|
|
2019-12-03 22:38:25 +00:00
|
|
|
return db.Set(p.GetFingerPrint(), enc)
|
2019-11-02 16:56:43 +00:00
|
|
|
|
2020-04-30 18:44:34 +00:00
|
|
|
return errors.New(fmt.Sprintf("Package not found: %s", p.HumanReadableString()))
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (db *InMemoryDatabase) GetPackages() []string {
|
2019-11-02 16:56:43 +00:00
|
|
|
keys := []string{}
|
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
|
|
|
for k, _ := range db.Database {
|
|
|
|
keys = append(keys, k)
|
|
|
|
}
|
|
|
|
return keys
|
2019-10-28 16:12:29 +00:00
|
|
|
}
|
2019-10-30 16:52:57 +00:00
|
|
|
|
|
|
|
func (db *InMemoryDatabase) Clean() error {
|
|
|
|
db.Database = map[string]string{}
|
|
|
|
return nil
|
|
|
|
}
|
2019-11-23 21:41:51 +00:00
|
|
|
|
|
|
|
func (db *InMemoryDatabase) GetPackageFiles(p Package) ([]string, error) {
|
|
|
|
|
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
|
|
|
|
|
|
|
pa, ok := db.FileDatabase[p.GetFingerPrint()]
|
|
|
|
if !ok {
|
2020-04-30 18:44:34 +00:00
|
|
|
return pa, errors.New(fmt.Sprintf("No key found for: %s", p.HumanReadableString()))
|
2019-11-23 21:41:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return pa, nil
|
|
|
|
}
|
2019-11-25 19:03:37 +00:00
|
|
|
func (db *InMemoryDatabase) SetPackageFiles(p *PackageFile) error {
|
2019-11-23 21:41:51 +00:00
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
|
|
|
db.FileDatabase[p.PackageFingerprint] = p.Files
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
func (db *InMemoryDatabase) RemovePackageFiles(p Package) error {
|
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
|
|
|
delete(db.FileDatabase, p.GetFingerPrint())
|
|
|
|
return nil
|
|
|
|
}
|
2019-11-23 23:16:12 +00:00
|
|
|
|
|
|
|
func (db *InMemoryDatabase) RemovePackage(p Package) error {
|
2019-12-03 22:38:25 +00:00
|
|
|
db.Lock()
|
|
|
|
defer db.Unlock()
|
|
|
|
|
|
|
|
delete(db.Database, p.GetFingerPrint())
|
|
|
|
return nil
|
2019-11-23 23:16:12 +00:00
|
|
|
}
|
2020-04-04 12:29:08 +00:00
|
|
|
func (db *InMemoryDatabase) World() Packages {
|
2019-11-29 18:01:48 +00:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
2020-04-04 12:29:08 +00:00
|
|
|
return Packages(all)
|
2019-11-29 18:01:48 +00:00
|
|
|
}
|
2019-12-01 18:11:19 +00:00
|
|
|
|
|
|
|
func (db *InMemoryDatabase) FindPackageCandidate(p Package) (Package, error) {
|
|
|
|
|
|
|
|
required, err := db.FindPackage(p)
|
|
|
|
if err != nil {
|
2020-12-19 13:57:42 +00:00
|
|
|
err = nil
|
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
|
2020-12-19 13:57:42 +00:00
|
|
|
err = errors.Wrap(err, "Candidate not found")
|
2019-12-01 18:11:19 +00:00
|
|
|
} else {
|
2020-04-04 13:33:14 +00:00
|
|
|
required = packages.Best(nil)
|
2019-12-01 18:11:19 +00:00
|
|
|
}
|
2020-12-19 13:57:42 +00:00
|
|
|
return required, err
|
2019-12-01 18:11:19 +00:00
|
|
|
//required = &DefaultPackage{Name: "test"}
|
|
|
|
}
|
|
|
|
|
|
|
|
return required, err
|
|
|
|
|
|
|
|
}
|
2020-03-20 22:47:35 +00:00
|
|
|
|
2020-04-04 12:29:08 +00:00
|
|
|
func (db *InMemoryDatabase) FindPackageLabel(labelKey string) (Packages, error) {
|
2020-03-20 22:47:35 +00:00
|
|
|
var ans []Package
|
|
|
|
|
|
|
|
for _, k := range db.GetPackages() {
|
|
|
|
pack, err := db.GetPackage(k)
|
|
|
|
if err != nil {
|
|
|
|
return ans, err
|
|
|
|
}
|
|
|
|
if pack.HasLabel(labelKey) {
|
|
|
|
ans = append(ans, pack)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-04 12:29:08 +00:00
|
|
|
return Packages(ans), nil
|
2020-03-20 22:47:35 +00:00
|
|
|
}
|
|
|
|
|
2020-04-04 12:29:08 +00:00
|
|
|
func (db *InMemoryDatabase) FindPackageLabelMatch(pattern string) (Packages, error) {
|
2020-03-20 22:47:35 +00:00
|
|
|
var ans []Package
|
|
|
|
|
|
|
|
re := regexp.MustCompile(pattern)
|
|
|
|
if re == nil {
|
|
|
|
return nil, errors.New("Invalid regex " + pattern + "!")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, k := range db.GetPackages() {
|
|
|
|
pack, err := db.GetPackage(k)
|
|
|
|
if err != nil {
|
|
|
|
return ans, err
|
|
|
|
}
|
|
|
|
if pack.MatchLabel(re) {
|
|
|
|
ans = append(ans, pack)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-04 12:29:08 +00:00
|
|
|
return Packages(ans), nil
|
2020-03-20 22:47:35 +00:00
|
|
|
}
|
|
|
|
|
2020-04-04 12:29:08 +00:00
|
|
|
func (db *InMemoryDatabase) FindPackageMatch(pattern string) (Packages, error) {
|
2020-03-20 22:47:35 +00:00
|
|
|
var ans []Package
|
|
|
|
|
|
|
|
re := regexp.MustCompile(pattern)
|
|
|
|
if re == nil {
|
|
|
|
return nil, errors.New("Invalid regex " + pattern + "!")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, k := range db.GetPackages() {
|
|
|
|
pack, err := db.GetPackage(k)
|
|
|
|
if err != nil {
|
|
|
|
return ans, err
|
|
|
|
}
|
|
|
|
|
2020-04-19 08:47:55 +00:00
|
|
|
if re.MatchString(pack.HumanReadableString()) {
|
2020-03-20 22:47:35 +00:00
|
|
|
ans = append(ans, pack)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-04 12:29:08 +00:00
|
|
|
return Packages(ans), nil
|
2020-03-20 22:47:35 +00:00
|
|
|
}
|