2018-09-21 21:29:50 +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 solver
|
|
|
|
|
|
|
|
import (
|
2019-11-11 13:58:12 +00:00
|
|
|
"crypto/sha256"
|
2019-10-31 11:34:28 +00:00
|
|
|
"fmt"
|
2019-11-15 17:11:26 +00:00
|
|
|
"sort"
|
|
|
|
"unicode"
|
|
|
|
|
2019-06-05 17:13:09 +00:00
|
|
|
pkg "github.com/mudler/luet/pkg/package"
|
2019-11-29 18:01:41 +00:00
|
|
|
toposort "github.com/philopon/go-toposort"
|
2019-11-12 16:27:05 +00:00
|
|
|
"github.com/stevenle/topsort"
|
2018-09-21 21:29:50 +00:00
|
|
|
)
|
|
|
|
|
2019-11-11 10:02:32 +00:00
|
|
|
type PackagesAssertions []PackageAssert
|
|
|
|
|
2019-11-15 23:38:07 +00:00
|
|
|
type PackageHash struct {
|
|
|
|
BuildHash string
|
|
|
|
PackageHash string
|
|
|
|
}
|
|
|
|
|
2019-06-11 16:03:50 +00:00
|
|
|
// PackageAssert represent a package assertion.
|
|
|
|
// It is composed of a Package and a Value which is indicating the absence or not
|
|
|
|
// of the associated package state.
|
2019-06-04 19:25:17 +00:00
|
|
|
type PackageAssert struct {
|
2019-11-22 23:29:24 +00:00
|
|
|
Package *pkg.DefaultPackage
|
2019-06-04 19:25:17 +00:00
|
|
|
Value bool
|
2019-11-15 23:38:07 +00:00
|
|
|
Hash PackageHash
|
2019-06-04 19:25:17 +00:00
|
|
|
}
|
|
|
|
|
2019-06-11 16:03:50 +00:00
|
|
|
// DecodeModel decodes a model from the SAT solver to package assertions (PackageAssert)
|
2019-11-15 23:38:07 +00:00
|
|
|
func DecodeModel(model map[string]bool, db pkg.PackageDatabase) (PackagesAssertions, error) {
|
2019-11-11 10:02:32 +00:00
|
|
|
ass := make(PackagesAssertions, 0)
|
2018-09-21 21:29:50 +00:00
|
|
|
for k, v := range model {
|
2019-11-15 23:38:07 +00:00
|
|
|
a, err := pkg.DecodePackage(k, db)
|
2019-06-04 19:57:13 +00:00
|
|
|
if err != nil {
|
2019-06-04 19:25:17 +00:00
|
|
|
return nil, err
|
2019-06-04 19:57:13 +00:00
|
|
|
|
2018-09-21 21:29:50 +00:00
|
|
|
}
|
2019-11-22 23:29:24 +00:00
|
|
|
ass = append(ass, PackageAssert{Package: a.(*pkg.DefaultPackage), Value: v})
|
2018-09-21 21:29:50 +00:00
|
|
|
}
|
2019-12-13 22:05:09 +00:00
|
|
|
return ass, nil
|
2018-09-21 21:29:50 +00:00
|
|
|
}
|
2019-10-31 11:34:28 +00:00
|
|
|
|
|
|
|
func (a *PackageAssert) Explain() {
|
|
|
|
fmt.Println(a.ToString())
|
|
|
|
a.Package.Explain()
|
|
|
|
}
|
|
|
|
|
2019-11-09 13:59:20 +00:00
|
|
|
func (a *PackageAssert) String() string {
|
|
|
|
return a.ToString()
|
|
|
|
}
|
|
|
|
|
2019-10-31 11:34:28 +00:00
|
|
|
func (a *PackageAssert) ToString() string {
|
|
|
|
var msg string
|
2019-11-29 18:01:47 +00:00
|
|
|
if a.Value {
|
2019-10-31 11:34:28 +00:00
|
|
|
msg = "installed"
|
|
|
|
} else {
|
|
|
|
msg = "not installed"
|
|
|
|
}
|
2019-11-29 18:01:47 +00:00
|
|
|
return fmt.Sprintf("%s/%s %s %s", a.Package.GetCategory(), a.Package.GetName(), a.Package.GetVersion(), msg)
|
2019-10-31 11:34:28 +00:00
|
|
|
}
|
2019-11-11 10:02:32 +00:00
|
|
|
|
2019-11-12 16:27:05 +00:00
|
|
|
func (assertions PackagesAssertions) EnsureOrder() PackagesAssertions {
|
2019-11-11 10:02:32 +00:00
|
|
|
|
|
|
|
orderedAssertions := PackagesAssertions{}
|
|
|
|
unorderedAssertions := PackagesAssertions{}
|
|
|
|
fingerprints := []string{}
|
|
|
|
|
|
|
|
tmpMap := map[string]PackageAssert{}
|
|
|
|
|
|
|
|
for _, a := range assertions {
|
2019-11-11 23:13:03 +00:00
|
|
|
tmpMap[a.Package.GetFingerPrint()] = a
|
2019-11-12 16:27:05 +00:00
|
|
|
fingerprints = append(fingerprints, a.Package.GetFingerPrint())
|
|
|
|
unorderedAssertions = append(unorderedAssertions, a) // Build a list of the ones that must be ordered
|
|
|
|
|
2019-11-29 18:01:41 +00:00
|
|
|
if a.Value {
|
2019-11-11 10:02:32 +00:00
|
|
|
unorderedAssertions = append(unorderedAssertions, a) // Build a list of the ones that must be ordered
|
|
|
|
} else {
|
|
|
|
orderedAssertions = append(orderedAssertions, a) // Keep last the ones which are not meant to be installed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-12 16:27:05 +00:00
|
|
|
sort.Sort(unorderedAssertions)
|
|
|
|
|
2019-11-11 10:02:32 +00:00
|
|
|
// Build a topological graph
|
|
|
|
graph := toposort.NewGraph(len(unorderedAssertions))
|
|
|
|
graph.AddNodes(fingerprints...)
|
|
|
|
for _, a := range unorderedAssertions {
|
|
|
|
for _, req := range a.Package.GetRequires() {
|
|
|
|
graph.AddEdge(a.Package.GetFingerPrint(), req.GetFingerPrint())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result, ok := graph.Toposort()
|
|
|
|
if !ok {
|
2019-11-12 16:27:05 +00:00
|
|
|
panic("Cycle found")
|
2019-11-11 10:02:32 +00:00
|
|
|
}
|
|
|
|
for _, res := range result {
|
|
|
|
a, ok := tmpMap[res]
|
|
|
|
if !ok {
|
2019-11-12 16:27:05 +00:00
|
|
|
panic("fail")
|
|
|
|
// continue
|
|
|
|
}
|
|
|
|
orderedAssertions = append(orderedAssertions, a)
|
|
|
|
// orderedAssertions = append(PackagesAssertions{a}, orderedAssertions...) // push upfront
|
|
|
|
}
|
|
|
|
//helpers.ReverseAny(orderedAssertions)
|
|
|
|
return orderedAssertions
|
|
|
|
}
|
|
|
|
|
2019-11-29 18:01:52 +00:00
|
|
|
func (assertions PackagesAssertions) Order(definitiondb pkg.PackageDatabase, fingerprint string) PackagesAssertions {
|
2019-11-12 16:27:05 +00:00
|
|
|
|
|
|
|
orderedAssertions := PackagesAssertions{}
|
|
|
|
unorderedAssertions := PackagesAssertions{}
|
|
|
|
fingerprints := []string{}
|
2019-12-13 16:19:09 +00:00
|
|
|
|
2019-12-13 22:05:09 +00:00
|
|
|
tmpMap := map[string]PackageAssert{}
|
2019-11-12 16:27:05 +00:00
|
|
|
graph := topsort.NewGraph()
|
|
|
|
|
|
|
|
for _, a := range assertions {
|
|
|
|
graph.AddNode(a.Package.GetFingerPrint())
|
|
|
|
tmpMap[a.Package.GetFingerPrint()] = a
|
|
|
|
fingerprints = append(fingerprints, a.Package.GetFingerPrint())
|
|
|
|
unorderedAssertions = append(unorderedAssertions, a) // Build a list of the ones that must be ordered
|
|
|
|
|
2019-11-29 18:01:41 +00:00
|
|
|
if a.Value {
|
2019-11-12 16:27:05 +00:00
|
|
|
unorderedAssertions = append(unorderedAssertions, a) // Build a list of the ones that must be ordered
|
|
|
|
} else {
|
|
|
|
orderedAssertions = append(orderedAssertions, a) // Keep last the ones which are not meant to be installed
|
2019-11-11 10:02:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-12 16:27:05 +00:00
|
|
|
sort.Sort(unorderedAssertions)
|
|
|
|
|
|
|
|
// Build a topological graph
|
|
|
|
//graph := toposort.NewGraph(len(unorderedAssertions))
|
|
|
|
// graph.AddNodes(fingerprints...)
|
|
|
|
for _, a := range unorderedAssertions {
|
2019-11-29 18:01:52 +00:00
|
|
|
for _, requiredDef := range a.Package.GetRequires() {
|
2019-12-13 22:05:09 +00:00
|
|
|
req, err := definitiondb.FindPackageCandidate(requiredDef)
|
|
|
|
if err != nil {
|
|
|
|
req = requiredDef
|
|
|
|
}
|
2019-11-29 18:01:52 +00:00
|
|
|
|
|
|
|
// Expand also here, as we need to order them (or instead the solver should give back the dep correctly?)
|
2019-12-13 22:05:09 +00:00
|
|
|
graph.AddEdge(a.Package.GetFingerPrint(), req.GetFingerPrint())
|
2019-11-12 16:27:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
result, err := graph.TopSort(fingerprint)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
for _, res := range result {
|
2019-12-13 22:05:09 +00:00
|
|
|
a, ok := tmpMap[res]
|
|
|
|
if !ok {
|
|
|
|
panic("fail")
|
|
|
|
// continue
|
2019-11-12 16:27:05 +00:00
|
|
|
}
|
2019-12-13 22:05:09 +00:00
|
|
|
orderedAssertions = append(orderedAssertions, a)
|
2019-11-12 16:27:05 +00:00
|
|
|
// orderedAssertions = append(PackagesAssertions{a}, orderedAssertions...) // push upfront
|
|
|
|
}
|
|
|
|
//helpers.ReverseAny(orderedAssertions)
|
2019-11-11 10:02:32 +00:00
|
|
|
return orderedAssertions
|
|
|
|
}
|
2019-11-11 13:58:12 +00:00
|
|
|
|
|
|
|
func (assertions PackagesAssertions) Explain() string {
|
|
|
|
var fingerprint string
|
2019-11-12 16:27:05 +00:00
|
|
|
for _, assertion := range assertions { // Always order them
|
2019-11-11 13:58:12 +00:00
|
|
|
fingerprint += assertion.ToString() + "\n"
|
|
|
|
}
|
|
|
|
return fingerprint
|
|
|
|
}
|
|
|
|
|
2019-11-12 16:27:05 +00:00
|
|
|
func (a PackagesAssertions) Len() int { return len(a) }
|
|
|
|
func (a PackagesAssertions) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
|
|
func (a PackagesAssertions) Less(i, j int) bool {
|
|
|
|
|
|
|
|
iRunes := []rune(a[i].Package.GetName())
|
|
|
|
jRunes := []rune(a[j].Package.GetName())
|
|
|
|
|
|
|
|
max := len(iRunes)
|
|
|
|
if max > len(jRunes) {
|
|
|
|
max = len(jRunes)
|
|
|
|
}
|
|
|
|
|
|
|
|
for idx := 0; idx < max; idx++ {
|
|
|
|
ir := iRunes[idx]
|
|
|
|
jr := jRunes[idx]
|
|
|
|
|
|
|
|
lir := unicode.ToLower(ir)
|
|
|
|
ljr := unicode.ToLower(jr)
|
|
|
|
|
|
|
|
if lir != ljr {
|
|
|
|
return lir < ljr
|
|
|
|
}
|
|
|
|
|
|
|
|
// the lowercase runes are the same, so compare the original
|
|
|
|
if ir != jr {
|
|
|
|
return ir < jr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2019-11-11 13:58:12 +00:00
|
|
|
func (assertions PackagesAssertions) AssertionHash() string {
|
|
|
|
var fingerprint string
|
2019-11-12 16:27:05 +00:00
|
|
|
for _, assertion := range assertions { // Note: Always order them first!
|
2019-11-29 18:01:41 +00:00
|
|
|
if assertion.Value { // Tke into account only dependencies installed (get fingerprint of subgraph)
|
2019-11-11 13:58:12 +00:00
|
|
|
fingerprint += assertion.ToString() + "\n"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
hash := sha256.Sum256([]byte(fingerprint))
|
|
|
|
return fmt.Sprintf("%x", hash)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (assertions PackagesAssertions) Drop(p pkg.Package) PackagesAssertions {
|
|
|
|
ass := PackagesAssertions{}
|
|
|
|
|
|
|
|
for _, a := range assertions {
|
2019-11-15 17:04:46 +00:00
|
|
|
if !a.Package.Matches(p) {
|
2019-11-11 13:58:12 +00:00
|
|
|
ass = append(ass, a)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ass
|
|
|
|
}
|