Skip to content

Commit

Permalink
Support archive priorities
Browse files Browse the repository at this point in the history
If a package to be downloaded exists in archive A and archive B, and
archive A has higher priority than archive B, the package is downloaded
from archive A even if archive B has a newer version. If two or more
archives have the same priority, the most recent version of the package
from all of them is selected, if any. Otherwise, lower priority archives
are tried, if any.
  • Loading branch information
woky committed May 16, 2023
1 parent f559603 commit 7e03f29
Show file tree
Hide file tree
Showing 5 changed files with 145 additions and 16 deletions.
1 change: 1 addition & 0 deletions cmd/chisel/cmd_cut.go
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,7 @@ func (cmd *cmdCut) Execute(args []string) error {
Suites: archiveInfo.Suites,
Components: archiveInfo.Components,
CacheDir: cache.DefaultDir("chisel"),
Priority: archiveInfo.Priority,
})
if err != nil {
return err
Expand Down
1 change: 1 addition & 0 deletions internal/archive/archive.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ type Options struct {
Suites []string
Components []string
CacheDir string
Priority int
}

func Open(options *Options) (Archive, error) {
Expand Down
3 changes: 3 additions & 0 deletions internal/setup/setup.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ type Archive struct {
Version string
Suites []string
Components []string
Priority int
}

// Package holds a collection of slices that represent parts of themselves.
Expand Down Expand Up @@ -321,6 +322,7 @@ type yamlArchive struct {
Version string `yaml:"version"`
Suites []string `yaml:"suites"`
Components []string `yaml:"components"`
Priority int `yaml:"priority"`
}

type yamlPackage struct {
Expand Down Expand Up @@ -427,6 +429,7 @@ func parseRelease(baseDir, filePath string, data []byte) (*Release, error) {
Version: details.Version,
Suites: details.Suites,
Components: details.Components,
Priority: details.Priority,
}
}

Expand Down
39 changes: 38 additions & 1 deletion internal/slicer/slicer.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,10 @@ import (
"bytes"
"fmt"
"io"
"math"
"os"
"path/filepath"
"sort"
"strings"
"syscall"

Expand Down Expand Up @@ -69,13 +71,48 @@ func Run(options *RunOptions) error {
targetDirAbs = filepath.Join(dir, targetDir)
}

// Archives are ordered in descending order by priority. The
// default priority is 0 and the maximum allowed priority is
// the upper limit of int.
//
// If a package to be downloaded exists in archive A and
// archive B, and archive A has higher priority than archive
// B, the package is downloaded from archive A even if archive
// B has a newer version.
//
// If two or more archives have the same priority, the most
// recent version of the package from all of them is selected,
// if any. Otherwise, archives with lower priority are tried,
// if any.
//
// If two or more archives with the same priority contain the
// package with the same version, the first archive from the
// options.Archives iterator is used. In effect, the choice is
// random.
orderedArchives := make([]archive.Archive, 0, len(options.Archives))
for _, archive := range options.Archives {
orderedArchives = append(orderedArchives, archive)
}
sort.Slice(orderedArchives, func(a, b int) bool {
prioA := orderedArchives[a].Options().Priority
prioB := orderedArchives[b].Options().Priority
return prioA > prioB
})

// Build information to process the selection.
for _, slice := range options.Selection.Slices {
extractPackage := extract[slice.Package]
if extractPackage == nil {
var selectedVersion string
var selectedArchive archive.Archive
for _, archive := range options.Archives {
currentPrio := math.MaxInt
for _, archive := range orderedArchives {
if prio := archive.Options().Priority; prio < currentPrio {
if selectedVersion != "" {
break
}
currentPrio = prio
}
version := archive.Version(slice.Package)
if version == "" {
continue
Expand Down
117 changes: 102 additions & 15 deletions internal/slicer/slicer_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -669,6 +669,84 @@ var slicerTests = []slicerTest{{
"/speed/cheetah": "file 0644 e98b0879",
"/speed/ostrich": "file 0644 c8fa2806",
},
}, {
summary: "Pick package from archive with highest priority",
pkgs: []testPackageInfo{
{
name: "hello",
version: "2.0-beta",
archive: "edge",
content: []testutil.TarEntry{
{Header: tar.Header{Name: "./"}},
{Header: tar.Header{Name: "./hello"}, Content: []byte("Hello, The Edge\n")},
},
},
{
name: "hello",
version: "1.8",
archive: "candidate",
content: []testutil.TarEntry{
{Header: tar.Header{Name: "./"}},
{Header: tar.Header{Name: "./hello"}, Content: []byte("Hello, The Candidate\n")},
},
},
{
name: "hello",
version: "1.2",
archive: "stable",
content: []testutil.TarEntry{
{Header: tar.Header{Name: "./"}},
{Header: tar.Header{Name: "./hello"}, Content: []byte("Hello, The Stable\n")},
},
},
{
name: "hello",
version: "1.0",
archive: "obsolete",
content: []testutil.TarEntry{
{Header: tar.Header{Name: "./"}},
{Header: tar.Header{Name: "./hello"}, Content: []byte("Hello, The Obsolete\n")},
},
},
},
release: map[string]string{
"chisel.yaml": `
format: chisel-v1
archives:
edge:
version: 1
suites: [main]
components: [main]
candidate:
version: 1
suites: [main]
components: [main]
priority: 5
stable:
version: 1
suites: [main]
components: [main]
priority: 10
obsolete:
version: 1
suites: [main]
components: [main]
priority: 10
`,
"slices/mydir/hello.yaml": `
package: hello
slices:
all:
contents:
/hello:
`,
},
slices: []setup.SliceKey{
{"hello", "all"},
},
result: map[string]string{
"/hello": "file 0644 b5621b65",
},
}}

const defaultChiselYaml = `
Expand All @@ -685,12 +763,12 @@ type testPackage struct {
}

type testArchive struct {
arch string
pkgs map[string]testPackage
options archive.Options
pkgs map[string]testPackage
}

func (a *testArchive) Options() *archive.Options {
return &archive.Options{Arch: a.arch}
return &a.options
}

func (a *testArchive) Fetch(pkg string) (io.ReadCloser, error) {
Expand Down Expand Up @@ -731,17 +809,27 @@ func (s *S) TestRun(c *C) {
c.Assert(err, IsNil)

archives := map[string]archive.Archive{}
for name, setupArchive := range release.Archives {
archive := &testArchive{
options: archive.Options{
Label: setupArchive.Name,
Version: setupArchive.Version,
Suites: setupArchive.Suites,
Components: setupArchive.Components,
Priority: setupArchive.Priority,
Arch: test.arch,
},
}
if test.pkgs != nil {
archive.pkgs = map[string]testPackage{}
}
archives[name] = archive
}

if test.pkgs != nil {
for _, pkgInfo := range test.pkgs {
archive, ok := archives[pkgInfo.archive].(*testArchive)
if !ok {
archive = &testArchive{
arch: test.arch,
pkgs: map[string]testPackage{},
}
archives[pkgInfo.archive] = archive
}
archive := archives[pkgInfo.archive].(*testArchive)
c.Assert(archive, NotNil)
deb, err := testutil.MakeDeb(pkgInfo.content)
c.Assert(err, IsNil)
archive.pkgs[pkgInfo.name] = testPackage{pkgInfo.version, deb}
Expand All @@ -755,10 +843,9 @@ func (s *S) TestRun(c *C) {
c.Assert(err, IsNil)
pkgs[name] = testPackage{"1", deb}
}
archives["ubuntu"] = &testArchive{
arch: test.arch,
pkgs: pkgs,
}
archive := archives["ubuntu"].(*testArchive)
c.Assert(archive, NotNil)
archive.pkgs = pkgs
}

targetDir := c.MkDir()
Expand Down

0 comments on commit 7e03f29

Please sign in to comment.