forked from arduino/arduino-cli
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathresolve_deps.go
103 lines (93 loc) · 3.54 KB
/
resolve_deps.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
// This file is part of arduino-cli.
//
// Copyright 2020 ARDUINO SA (http://www.arduino.cc/)
//
// This software is released under the GNU General Public License version 3,
// which covers the main part of arduino-cli.
// The terms of this license can be found at:
// https://www.gnu.org/licenses/gpl-3.0.en.html
//
// You can be released from the requirements of the above licenses by purchasing
// a commercial license. Buying such a license is mandatory if you want to
// modify or otherwise use the software for commercial activities involving the
// Arduino software without disclosing the source code of your own applications.
// To purchase a commercial license, send an email to [email protected].
package lib
import (
"context"
"errors"
"sort"
"github.com/arduino/arduino-cli/commands/cmderrors"
"github.com/arduino/arduino-cli/commands/internal/instances"
"github.com/arduino/arduino-cli/internal/arduino/libraries"
"github.com/arduino/arduino-cli/internal/arduino/libraries/librariesindex"
rpc "github.com/arduino/arduino-cli/rpc/cc/arduino/cli/commands/v1"
semver "go.bug.st/relaxed-semver"
)
// LibraryResolveDependencies FIXMEDOC
func LibraryResolveDependencies(ctx context.Context, req *rpc.LibraryResolveDependenciesRequest) (*rpc.LibraryResolveDependenciesResponse, error) {
lm, err := instances.GetLibraryManager(req.GetInstance())
if err != nil {
return nil, err
}
// Search the requested lib
reqLibRelease, err := findLibraryIndexRelease(lm.Index, req)
if err != nil {
return nil, err
}
// Extract all installed libraries
installedLibs := map[string]*libraries.Library{}
for _, lib := range listLibraries(lm, false, false) {
installedLibs[lib.Library.Name] = lib.Library
}
// Resolve all dependencies...
var overrides []*librariesindex.Release
if req.GetDoNotUpdateInstalledLibraries() {
libs := lm.FindAllInstalled()
libs = libs.FilterByVersionAndInstallLocation(nil, libraries.User)
for _, lib := range libs {
release := lm.Index.FindRelease(&librariesindex.Reference{
Name: lib.Name,
Version: lib.Version,
})
if release != nil {
overrides = append(overrides, release)
}
}
}
deps := lm.Index.ResolveDependencies(reqLibRelease, overrides)
// If no solution has been found
if len(deps) == 0 {
// Check if there is a problem with the first level deps
for _, directDep := range reqLibRelease.GetDependencies() {
if _, ok := lm.Index.Libraries[directDep.GetName()]; !ok {
err := errors.New(tr("dependency '%s' is not available", directDep.GetName()))
return nil, &cmderrors.LibraryDependenciesResolutionFailedError{Cause: err}
}
}
// Otherwise there is no possible solution, the depends field has an invalid formula
return nil, &cmderrors.LibraryDependenciesResolutionFailedError{}
}
res := []*rpc.LibraryDependencyStatus{}
for _, dep := range deps {
// ...and add information on currently installed versions of the libraries
var installed *semver.Version
required := dep.GetVersion()
if installedLib, has := installedLibs[dep.GetName()]; has {
installed = installedLib.Version
if installed != nil && required != nil && installed.Equal(required) {
// avoid situations like installed=0.53 and required=0.53.0
required = installed
}
}
res = append(res, &rpc.LibraryDependencyStatus{
Name: dep.GetName(),
VersionRequired: required.String(),
VersionInstalled: installed.String(),
})
}
sort.Slice(res, func(i, j int) bool {
return res[i].GetName() < res[j].GetName()
})
return &rpc.LibraryResolveDependenciesResponse{Dependencies: res}, nil
}