luet/pkg/package/package.go

611 lines
15 KiB
Go
Raw Normal View History

2019-06-04 19:25:17 +00:00
// Copyright © 2019 Ettore Di Giacinto <mudler@gentoo.org>
2018-09-21 21:29:50 +00:00
//
// 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 (
"bytes"
2019-11-09 13:59:20 +00:00
"encoding/json"
"fmt"
2019-11-04 11:36:31 +00:00
"path/filepath"
"sort"
"strings"
// . "github.com/mudler/luet/pkg/logger"
2020-02-11 13:58:17 +00:00
"github.com/crillab/gophersat/bf"
version "github.com/hashicorp/go-version"
2018-09-21 21:29:50 +00:00
"github.com/jinzhu/copier"
"github.com/ghodss/yaml"
2018-09-21 21:29:50 +00:00
)
// Package is a package interface (TBD)
// FIXME: Currently some of the methods are returning DefaultPackages due to JSON serialization of the package
2018-09-21 21:29:50 +00:00
type Package interface {
Encode(PackageDatabase) (string, error)
BuildFormula(PackageDatabase, PackageDatabase) ([]bf.Formula, error)
2018-09-21 21:29:50 +00:00
IsFlagged(bool) Package
2019-06-04 19:25:17 +00:00
Flagged() bool
GetFingerPrint() string
GetPackageName() string
2019-06-04 19:25:17 +00:00
Requires([]*DefaultPackage) Package
Conflicts([]*DefaultPackage) Package
Revdeps(PackageDatabase) []Package
GetProvides() []*DefaultPackage
SetProvides([]*DefaultPackage) Package
GetRequires() []*DefaultPackage
GetConflicts() []*DefaultPackage
Expand(PackageDatabase) ([]Package, error)
SetCategory(string)
GetName() string
GetCategory() string
GetVersion() string
RequiresContains(PackageDatabase, Package) (bool, error)
2019-11-15 17:04:46 +00:00
Matches(m Package) bool
Bigger(m Package) bool
2019-06-12 19:32:15 +00:00
AddUse(use string)
RemoveUse(use string)
GetUses() []string
Yaml() ([]byte, error)
Explain()
SetPath(string)
GetPath() string
2019-11-04 11:36:31 +00:00
Rel(string) string
GetDescription() string
SetDescription(string)
AddURI(string)
GetURI() []string
SetLicense(string)
GetLicense() string
IsSelector() bool
2019-12-27 10:18:52 +00:00
VersionMatchSelector(string) (bool, error)
SelectorMatchVersion(string) (bool, error)
String() string
2018-09-21 21:29:50 +00:00
}
2019-07-27 09:29:39 +00:00
type Tree interface {
GetPackageSet() PackageDatabase
Prelude() string // A tree might have a prelude to be able to consume a tree
SetPackageSet(s PackageDatabase)
World() ([]Package, error)
FindPackage(Package) (Package, error)
}
// >> Unmarshallers
// DefaultPackageFromYaml decodes a package from yaml bytes
func DefaultPackageFromYaml(yml []byte) (DefaultPackage, error) {
var unescaped DefaultPackage
source, err := yaml.YAMLToJSON(yml)
if err != nil {
return DefaultPackage{}, err
}
rawIn := json.RawMessage(source)
bytes, err := rawIn.MarshalJSON()
if err != nil {
return DefaultPackage{}, err
}
err = json.Unmarshal(bytes, &unescaped)
if err != nil {
return DefaultPackage{}, err
}
return unescaped, nil
}
// Major and minor gets escaped when marshalling in JSON, making compiler fails recognizing selectors for expansion
func (t *DefaultPackage) JSON() ([]byte, error) {
buffer := &bytes.Buffer{}
encoder := json.NewEncoder(buffer)
encoder.SetEscapeHTML(false)
err := encoder.Encode(t)
return buffer.Bytes(), err
2019-07-27 09:29:39 +00:00
}
// DefaultPackage represent a standard package definition
2018-09-21 21:29:50 +00:00
type DefaultPackage struct {
ID int `storm:"id,increment" json:"id"` // primary key with auto increment
Name string `json:"name"` // Affects YAML field names too.
Version string `json:"version"` // Affects YAML field names too.
Category string `json:"category"` // Affects YAML field names too.
UseFlags []string `json:"use_flags"` // Affects YAML field names too.
2018-09-21 21:29:50 +00:00
State State
PackageRequires []*DefaultPackage `json:"requires"` // Affects YAML field names too.
PackageConflicts []*DefaultPackage `json:"conflicts"` // Affects YAML field names too.
IsSet bool `json:"set"` // Affects YAML field names too.
Provides []*DefaultPackage `json:"provides"` // Affects YAML field names too.
// TODO: Annotations?
// Path is set only internally when tree is loaded from disk
Path string `json:"path,omitempty"`
Description string `json:"description"`
Uri []string `json:"uri"`
License string `json:"license"`
2018-09-21 21:29:50 +00:00
}
// State represent the package state
2018-09-21 21:29:50 +00:00
type State string
// NewPackage returns a new package
2019-06-04 19:25:17 +00:00
func NewPackage(name, version string, requires []*DefaultPackage, conflicts []*DefaultPackage) *DefaultPackage {
2018-09-21 21:29:50 +00:00
return &DefaultPackage{Name: name, Version: version, PackageRequires: requires, PackageConflicts: conflicts}
}
2019-11-09 13:59:20 +00:00
func (p *DefaultPackage) String() string {
b, err := p.JSON()
2019-11-09 13:59:20 +00:00
if err != nil {
return fmt.Sprintf("{ id: \"%d\", name: \"%s\", version: \"%s\", category: \"%s\" }", p.ID, p.Name, p.Version, p.Category)
2019-11-09 13:59:20 +00:00
}
return fmt.Sprintf("%s", string(b))
}
// GetFingerPrint returns a UUID of the package.
// FIXME: this needs to be unique, now just name is generalized
func (p *DefaultPackage) GetFingerPrint() string {
return fmt.Sprintf("%s-%s-%s", p.Name, p.Category, p.Version)
}
2018-09-21 21:29:50 +00:00
func FromString(s string) Package {
var unescaped DefaultPackage
err := json.Unmarshal([]byte(s), &unescaped)
if err != nil {
return &unescaped
}
return &unescaped
}
func (p *DefaultPackage) GetPackageName() string {
return fmt.Sprintf("%s-%s", p.Name, p.Category)
}
// GetPath returns the path where the definition file was found
func (p *DefaultPackage) GetPath() string {
return p.Path
}
2019-11-04 11:36:31 +00:00
func (p *DefaultPackage) Rel(s string) string {
return filepath.Join(p.GetPath(), s)
}
func (p *DefaultPackage) SetPath(s string) {
p.Path = s
}
func (p *DefaultPackage) IsSelector() bool {
return strings.ContainsAny(p.GetVersion(), "<>=")
}
// AddUse adds a use to a package
2018-09-21 21:29:50 +00:00
func (p *DefaultPackage) AddUse(use string) {
for _, v := range p.UseFlags {
if v == use {
return
}
}
p.UseFlags = append(p.UseFlags, use)
}
// RemoveUse removes a use to a package
2018-09-21 21:29:50 +00:00
func (p *DefaultPackage) RemoveUse(use string) {
for i := len(p.UseFlags) - 1; i >= 0; i-- {
if p.UseFlags[i] == use {
p.UseFlags = append(p.UseFlags[:i], p.UseFlags[i+1:]...)
}
}
}
// Encode encodes the package to string.
// It returns an ID which can be used to retrieve the package later on.
func (p *DefaultPackage) Encode(db PackageDatabase) (string, error) {
return db.CreatePackage(p)
2018-09-21 21:29:50 +00:00
}
func (p *DefaultPackage) Yaml() ([]byte, error) {
j, err := p.JSON()
if err != nil {
return []byte{}, err
}
y, err := yaml.JSONToYAML(j)
if err != nil {
return []byte{}, err
}
return y, nil
}
2018-09-21 21:29:50 +00:00
func (p *DefaultPackage) IsFlagged(b bool) Package {
p.IsSet = b
return p
}
2019-06-04 19:25:17 +00:00
func (p *DefaultPackage) Flagged() bool {
return p.IsSet
}
func (p *DefaultPackage) GetName() string {
return p.Name
}
func (p *DefaultPackage) GetVersion() string {
return p.Version
}
func (p *DefaultPackage) GetDescription() string {
return p.Description
}
func (p *DefaultPackage) SetDescription(s string) {
p.Description = s
}
func (p *DefaultPackage) GetLicense() string {
return p.License
}
func (p *DefaultPackage) SetLicense(s string) {
p.License = s
}
func (p *DefaultPackage) AddURI(s string) {
p.Uri = append(p.Uri, s)
}
func (p *DefaultPackage) GetURI() []string {
return p.Uri
}
func (p *DefaultPackage) GetCategory() string {
return p.Category
}
func (p *DefaultPackage) SetCategory(s string) {
p.Category = s
}
2019-06-12 19:32:15 +00:00
func (p *DefaultPackage) GetUses() []string {
return p.UseFlags
}
func (p *DefaultPackage) GetProvides() []*DefaultPackage {
return p.Provides
}
func (p *DefaultPackage) SetProvides(req []*DefaultPackage) Package {
p.Provides = req
return p
}
func (p *DefaultPackage) GetRequires() []*DefaultPackage {
return p.PackageRequires
}
func (p *DefaultPackage) GetConflicts() []*DefaultPackage {
return p.PackageConflicts
}
2019-06-04 19:25:17 +00:00
func (p *DefaultPackage) Requires(req []*DefaultPackage) Package {
2018-09-21 21:29:50 +00:00
p.PackageRequires = req
return p
}
2019-06-04 19:25:17 +00:00
func (p *DefaultPackage) Conflicts(req []*DefaultPackage) Package {
2018-09-21 21:29:50 +00:00
p.PackageConflicts = req
return p
}
func (p *DefaultPackage) Clone() Package {
new := &DefaultPackage{}
copier.Copy(&new, &p)
return new
}
2019-11-15 17:04:46 +00:00
func (p *DefaultPackage) Matches(m Package) bool {
if p.GetFingerPrint() == m.GetFingerPrint() {
return true
}
return false
}
func (p *DefaultPackage) Bigger(m Package) bool {
low := Lower([]Package{p, m})
if low.Matches(m) {
return true
}
return false
}
2019-11-15 17:04:46 +00:00
func (p *DefaultPackage) Expand(definitiondb PackageDatabase) ([]Package, error) {
var versionsInWorld []Package
all, err := definitiondb.FindPackages(p)
if err != nil {
return nil, err
}
for _, w := range all {
2019-12-27 10:18:52 +00:00
match, err := p.SelectorMatchVersion(w.GetVersion())
2019-06-11 16:47:07 +00:00
if err != nil {
return nil, err
}
2019-12-27 10:18:52 +00:00
if match {
2019-06-11 16:47:07 +00:00
versionsInWorld = append(versionsInWorld, w)
}
}
return versionsInWorld, nil
}
func (p *DefaultPackage) Revdeps(definitiondb PackageDatabase) []Package {
2019-11-14 22:44:22 +00:00
var versionsInWorld []Package
for _, w := range definitiondb.World() {
2019-11-15 17:04:46 +00:00
if w.Matches(p) {
2019-11-14 22:44:22 +00:00
continue
}
2019-11-15 17:04:46 +00:00
for _, re := range w.GetRequires() {
if re.Matches(p) {
2019-11-14 22:44:22 +00:00
versionsInWorld = append(versionsInWorld, w)
versionsInWorld = append(versionsInWorld, w.Revdeps(definitiondb)...)
2019-11-14 22:44:22 +00:00
}
}
}
return versionsInWorld
}
func DecodePackage(ID string, db PackageDatabase) (Package, error) {
return db.GetPackage(ID)
2018-09-21 21:29:50 +00:00
}
func (pack *DefaultPackage) RequiresContains(definitiondb PackageDatabase, s Package) (bool, error) {
p, err := definitiondb.FindPackage(pack)
if err != nil {
p = pack //relax things
//return false, errors.Wrap(err, "Package not found in definition db")
2019-06-05 16:40:33 +00:00
}
for _, re := range p.GetRequires() {
2019-11-15 17:04:46 +00:00
if re.Matches(s) {
return true, nil
}
packages, _ := re.Expand(definitiondb)
for _, pa := range packages {
if pa.Matches(s) {
return true, nil
}
}
if contains, err := re.RequiresContains(definitiondb, s); err == nil && contains {
return true, nil
}
}
return false, nil
}
func Lower(set []Package) Package {
var versionsMap map[string]Package = make(map[string]Package)
if len(set) == 0 {
panic("Best needs a list with elements")
}
versionsRaw := []string{}
for _, p := range set {
versionsRaw = append(versionsRaw, p.GetVersion())
versionsMap[p.GetVersion()] = p
}
versions := make([]*version.Version, len(versionsRaw))
for i, raw := range versionsRaw {
v, _ := version.NewVersion(raw)
versions[i] = v
}
// After this, the versions are properly sorted
sort.Sort(version.Collection(versions))
return versionsMap[versions[0].Original()]
}
func Best(set []Package) Package {
var versionsMap map[string]Package = make(map[string]Package)
if len(set) == 0 {
panic("Best needs a list with elements")
}
versionsRaw := []string{}
for _, p := range set {
versionsRaw = append(versionsRaw, p.GetVersion())
versionsMap[p.GetVersion()] = p
}
versions := make([]*version.Version, len(versionsRaw))
for i, raw := range versionsRaw {
v, _ := version.NewVersion(raw)
versions[i] = v
}
// After this, the versions are properly sorted
sort.Sort(version.Collection(versions))
return versionsMap[versions[len(versions)-1].Original()]
}
func (pack *DefaultPackage) BuildFormula(definitiondb PackageDatabase, db PackageDatabase) ([]bf.Formula, error) {
p, err := definitiondb.FindPackage(pack)
if err != nil {
2019-11-29 18:01:52 +00:00
p = pack // Relax failures and trust the def
}
encodedA, err := p.Encode(db)
2018-09-21 21:29:50 +00:00
if err != nil {
return nil, err
}
A := bf.Var(encodedA)
2019-06-04 19:57:13 +00:00
var formulas []bf.Formula
for _, requiredDef := range p.GetRequires() {
required, err := definitiondb.FindPackage(requiredDef)
if err != nil || requiredDef.IsSelector() {
if err == nil {
requiredDef = required.(*DefaultPackage)
}
2019-12-13 23:07:46 +00:00
packages, err := definitiondb.FindPackages(requiredDef)
if err != nil || len(packages) == 0 {
required = requiredDef
} else {
if len(packages) == 1 {
required = packages[0]
} else {
var ALO, priorityConstraints, priorityALO []bf.Formula
// Try to prio best match
// Force the solver to consider first our candidate (if does exists).
// Then builds ALO and AMO for the requires.
c, candidateErr := definitiondb.FindPackageCandidate(requiredDef)
var C bf.Formula
if candidateErr == nil {
// We have a desired candidate, try to look a solution with that included first
for _, o := range packages {
encodedB, err := o.Encode(db)
if err != nil {
return nil, err
}
B := bf.Var(encodedB)
if !o.Matches(c) {
priorityConstraints = append(priorityConstraints, bf.Not(B))
priorityALO = append(priorityALO, B)
}
}
encodedC, err := c.Encode(db)
if err != nil {
return nil, err
}
C = bf.Var(encodedC)
// Or the Candidate is true, or all the others might be not true
// This forces the CDCL sat implementation to look first at a solution with C=true
formulas = append(formulas, bf.Or(bf.Or(C, bf.Or(priorityConstraints...)), bf.Or(bf.Not(C), bf.Or(priorityALO...))))
}
// AMO - At most one
for _, o := range packages {
encodedB, err := o.Encode(db)
if err != nil {
return nil, err
}
B := bf.Var(encodedB)
ALO = append(ALO, B)
for _, i := range packages {
encodedI, err := i.Encode(db)
if err != nil {
return nil, err
}
I := bf.Var(encodedI)
if !o.Matches(i) {
formulas = append(formulas, bf.Or(bf.Not(I), bf.Not(B)))
}
}
}
formulas = append(formulas, bf.Or(ALO...)) // ALO - At least one
continue
}
}
}
2018-09-21 21:29:50 +00:00
encodedB, err := required.Encode(db)
2018-09-21 21:29:50 +00:00
if err != nil {
return nil, err
}
B := bf.Var(encodedB)
2019-06-04 19:25:17 +00:00
formulas = append(formulas, bf.Or(bf.Not(A), B))
2018-09-21 21:29:50 +00:00
f, err := required.BuildFormula(definitiondb, db)
2019-06-03 21:23:02 +00:00
if err != nil {
return nil, err
}
formulas = append(formulas, f...)
2018-09-21 21:29:50 +00:00
}
for _, requiredDef := range p.GetConflicts() {
required, err := definitiondb.FindPackage(requiredDef)
if err != nil || requiredDef.IsSelector() {
if err == nil {
requiredDef = required.(*DefaultPackage)
}
packages, err := definitiondb.FindPackages(requiredDef)
if err != nil || len(packages) == 0 {
required = requiredDef
} else {
if len(packages) == 1 {
required = packages[0]
} else {
for _, p := range packages {
encodedB, err := p.Encode(db)
if err != nil {
return nil, err
}
B := bf.Var(encodedB)
formulas = append(formulas, bf.Or(bf.Not(A),
bf.Not(B)))
f, err := p.BuildFormula(definitiondb, db)
if err != nil {
return nil, err
}
formulas = append(formulas, f...)
}
continue
}
}
}
encodedB, err := required.Encode(db)
2018-09-21 21:29:50 +00:00
if err != nil {
return nil, err
}
B := bf.Var(encodedB)
2019-06-04 19:25:17 +00:00
formulas = append(formulas, bf.Or(bf.Not(A),
bf.Not(B)))
2019-06-03 21:23:02 +00:00
f, err := required.BuildFormula(definitiondb, db)
2019-06-03 21:23:02 +00:00
if err != nil {
return nil, err
}
formulas = append(formulas, f...)
2018-09-21 21:29:50 +00:00
}
2019-06-04 19:25:17 +00:00
return formulas, nil
2018-09-21 21:29:50 +00:00
}
func (p *DefaultPackage) Explain() {
fmt.Println("====================")
fmt.Println("Name: ", p.GetName())
fmt.Println("Category: ", p.GetCategory())
fmt.Println("Version: ", p.GetVersion())
fmt.Println("Installed: ", p.IsSet)
for _, req := range p.GetRequires() {
fmt.Println("\t-> ", req)
}
for _, con := range p.GetConflicts() {
fmt.Println("\t!! ", con)
}
fmt.Println("====================")
}