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 18, 2023
1 parent 81da91f commit 17caff0
Show file tree
Hide file tree
Showing 5 changed files with 122 additions and 1 deletion.
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
79 changes: 79 additions & 0 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 Down Expand Up @@ -738,6 +816,7 @@ func (s *S) TestRun(c *C) {
Version: setupArchive.Version,
Suites: setupArchive.Suites,
Components: setupArchive.Components,
Priority: setupArchive.Priority,
Arch: test.arch,
},
}
Expand Down

0 comments on commit 17caff0

Please sign in to comment.