Skip to content

Commit db1d292

Browse files
Merge pull request #4274 from microsoft/mandeepsplaha/cherry-pick-rpm-resolution-mechanism-fix
Updated toolkit's package resolution to accept installed packages. (#4211)
2 parents e9b5520 + 66f5df4 commit db1d292

File tree

6 files changed

+95
-42
lines changed

6 files changed

+95
-42
lines changed

toolkit/tools/internal/pkggraph/pkggraph.go

+2-4
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ import (
1919
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/file"
2020
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/logger"
2121
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/pkgjson"
22+
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/sliceutils"
2223
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/versioncompare"
2324

2425
"gonum.org/v1/gonum/graph"
@@ -1417,10 +1418,7 @@ func rpmsProvidedBySRPM(srpmPath string, pkgGraph *PkgGraph, graphMutex *sync.RW
14171418
rpmsMap[node.RpmPath] = true
14181419
}
14191420

1420-
rpmFiles = make([]string, 0, len(rpmsMap))
1421-
for rpm := range rpmsMap {
1422-
rpmFiles = append(rpmFiles, rpm)
1423-
}
1421+
rpmFiles = sliceutils.StringsSetToSlice(rpmsMap)
14241422

14251423
return
14261424
}

toolkit/tools/internal/rpm/rpm.go

+25-22
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import (
1212
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/file"
1313
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/logger"
1414
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/shell"
15+
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/sliceutils"
1516
)
1617

1718
const (
@@ -53,36 +54,38 @@ const (
5354
)
5455

5556
const (
57+
installedRPMRegexRPMIndex = 1
58+
5659
rpmProgram = "rpm"
5760
rpmSpecProgram = "rpmspec"
5861
rpmBuildProgram = "rpmbuild"
5962
)
6063

61-
var goArchToRpmArch = map[string]string{
62-
"amd64": "x86_64",
63-
"arm64": "aarch64",
64-
}
65-
66-
// GetRpmArch converts the GOARCH arch into an RPM arch
67-
func GetRpmArch(goArch string) (rpmArch string, err error) {
68-
rpmArch, ok := goArchToRpmArch[goArch]
69-
if !ok {
70-
err = fmt.Errorf("Unknown GOARCH detected (%s)", goArch)
64+
var (
65+
goArchToRpmArch = map[string]string{
66+
"amd64": "x86_64",
67+
"arm64": "aarch64",
7168
}
72-
return
73-
}
7469

75-
var (
7670
// Output from 'rpm' prints installed RPMs in a line with the following format:
7771
//
7872
// D: ========== +++ [name]-[version]-[release].[distribution] [architecture]-linux [hex_value]
7973
//
8074
// Example:
8175
//
8276
// D: ========== +++ systemd-devel-239-42.cm2 x86_64-linux 0x0
83-
installedRPMLineRegex = regexp.MustCompile(`^D: =+ \+{3} (\S+).*$`)
77+
installedRPMRegex = regexp.MustCompile(`^D: =+ \+{3} (\S+).*$`)
8478
)
8579

80+
// GetRpmArch converts the GOARCH arch into an RPM arch
81+
func GetRpmArch(goArch string) (rpmArch string, err error) {
82+
rpmArch, ok := goArchToRpmArch[goArch]
83+
if !ok {
84+
err = fmt.Errorf("Unknown GOARCH detected (%s)", goArch)
85+
}
86+
return
87+
}
88+
8689
// SetMacroDir adds RPM_CONFIGDIR=$(newMacroDir) into the shell's environment for the duration of a program.
8790
// To restore the environment the caller can use shell.SetEnvironment() with the returned origenv.
8891
// On an empty string argument return success immediately and do not modify the environment.
@@ -318,13 +321,13 @@ func QueryRPMProvides(rpmFile string) (provides []string, err error) {
318321
// end up being installed after resolving outdated, obsoleted, or conflicting packages.
319322
func ResolveCompetingPackages(rootDir string, rpmPaths ...string) (resolvedRPMs []string, err error) {
320323
const (
321-
queryFormat = ""
322-
installedRPMIndex = 1
323-
squashErrors = true
324+
queryFormat = ""
325+
squashErrors = true
324326
)
325327

326328
args := []string{
327329
"-Uvvh",
330+
"--replacepkgs",
328331
"--nodeps",
329332
"--root",
330333
rootDir,
@@ -340,15 +343,15 @@ func ResolveCompetingPackages(rootDir string, rpmPaths ...string) (resolvedRPMs
340343
}
341344

342345
splitStdout := strings.Split(stderr, "\n")
346+
uniqueResolvedRPMs := map[string]bool{}
343347
for _, line := range splitStdout {
344-
matches := installedRPMLineRegex.FindStringSubmatch(line)
345-
if len(matches) == 0 {
346-
continue
348+
matches := installedRPMRegex.FindStringSubmatch(line)
349+
if len(matches) != 0 {
350+
uniqueResolvedRPMs[matches[installedRPMRegexRPMIndex]] = true
347351
}
348-
349-
resolvedRPMs = append(resolvedRPMs, matches[installedRPMIndex])
350352
}
351353

354+
resolvedRPMs = sliceutils.StringsSetToSlice(uniqueResolvedRPMs)
352355
return
353356
}
354357

toolkit/tools/internal/sliceutils/sliceutils.go

+14
Original file line numberDiff line numberDiff line change
@@ -41,3 +41,17 @@ func FindMatches(slice []string, isMatch func(string) bool) []string {
4141
func StringMatch(expected, given interface{}) bool {
4242
return expected.(string) == given.(string)
4343
}
44+
45+
func StringsSetToSlice(inputSet map[string]bool) []string {
46+
index := 0
47+
outputSlice := make([]string, len(inputSet))
48+
49+
for element, elementInSet := range inputSet {
50+
if elementInSet {
51+
outputSlice[index] = element
52+
index++
53+
}
54+
}
55+
56+
return outputSlice[:index]
57+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
// Copyright (c) Microsoft Corporation.
2+
// Licensed under the MIT License.
3+
4+
package sliceutils
5+
6+
import (
7+
"os"
8+
"testing"
9+
10+
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/logger"
11+
"github.com/stretchr/testify/assert"
12+
)
13+
14+
func TestMain(m *testing.M) {
15+
logger.InitStderrLog()
16+
os.Exit(m.Run())
17+
}
18+
19+
func TestShouldCreateEmptySliceFromNil(t *testing.T) {
20+
outputSlice := StringsSetToSlice(nil)
21+
22+
assert.NotNil(t, outputSlice)
23+
assert.Empty(t, outputSlice)
24+
}
25+
26+
func TestShouldCreateEmptySliceFromEmptySet(t *testing.T) {
27+
outputSlice := StringsSetToSlice(map[string]bool{})
28+
29+
assert.NotNil(t, outputSlice)
30+
assert.Empty(t, outputSlice)
31+
}
32+
33+
func TestShouldReturnValuesForAllTrueElementsInSet(t *testing.T) {
34+
inputSet := map[string]bool{
35+
"A": true,
36+
"B": true,
37+
"X": false,
38+
"Y": false,
39+
}
40+
outputSlice := StringsSetToSlice(inputSet)
41+
42+
assert.NotNil(t, outputSlice)
43+
assert.Len(t, outputSlice, 2)
44+
assert.Contains(t, outputSlice, "A")
45+
assert.Contains(t, outputSlice, "B")
46+
assert.NotContains(t, outputSlice, "X")
47+
assert.NotContains(t, outputSlice, "Y")
48+
}

toolkit/tools/scheduler/schedulerutils/buildworker.go

+1-4
Original file line numberDiff line numberDiff line change
@@ -181,10 +181,7 @@ func getBuildDependencies(node *pkggraph.PkgNode, pkgGraph *pkggraph.PkgGraph, g
181181
return
182182
})
183183

184-
dependencies = make([]string, 0, len(dependencyLookup))
185-
for depName := range dependencyLookup {
186-
dependencies = append(dependencies, depName)
187-
}
184+
dependencies = sliceutils.StringsSetToSlice(dependencyLookup)
188185

189186
return
190187
}

toolkit/tools/scheduler/schedulerutils/graphbuildstate.go

+5-12
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import (
99

1010
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/logger"
1111
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/pkggraph"
12+
"github.com/microsoft/CBL-Mariner/toolkit/tools/internal/sliceutils"
1213
)
1314

1415
// nodeState represents the build state of a single node
@@ -90,26 +91,18 @@ func (g *GraphBuildState) BuildFailures() []*BuildResult {
9091
// ConflictingRPMs will return a list of *.rpm files which should not have been rebuilt.
9192
// This list is based on the manifest of pre-built toolchain rpms.
9293
func (g *GraphBuildState) ConflictingRPMs() (rpms []string) {
93-
rpms = make([]string, len(g.conflictingRPMs))
94-
i := 0
95-
for f := range g.conflictingRPMs {
96-
rpms[i] = f
97-
i++
98-
}
94+
rpms = sliceutils.StringsSetToSlice(g.conflictingRPMs)
9995
sort.Strings(rpms)
96+
10097
return rpms
10198
}
10299

103100
// ConflictingSRPMs will return a list of *.src.rpm files which created rpms that should not have been rebuilt.
104101
// This list is based on the manifest of pre-built toolchain rpms.
105102
func (g *GraphBuildState) ConflictingSRPMs() (srpms []string) {
106-
srpms = make([]string, len(g.conflictingSRPMs))
107-
i := 0
108-
for f := range g.conflictingSRPMs {
109-
srpms[i] = f
110-
i++
111-
}
103+
srpms = sliceutils.StringsSetToSlice(g.conflictingSRPMs)
112104
sort.Strings(srpms)
105+
113106
return srpms
114107
}
115108

0 commit comments

Comments
 (0)