Skip to content

Commit

Permalink
imagefilter: add support to filter for pkg: prefixes
Browse files Browse the repository at this point in the history
This commit adds support to filter image based on (unresolved)
packages. E.g.:
```
$ ./image-builder list-images --filter pkg:gdisk
...
rhel-9.6 type:vhd arch:x86_64
```

Based on an idea from Achilleas-k, many thanks.
  • Loading branch information
mvo5 committed Nov 14, 2024
1 parent cf436f8 commit 8aacd8e
Show file tree
Hide file tree
Showing 2 changed files with 32 additions and 4 deletions.
33 changes: 29 additions & 4 deletions pkg/imagefilter/filter.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import (

"github.com/gobwas/glob"

"github.com/osbuild/images/pkg/blueprint"
"github.com/osbuild/images/pkg/distro"
)

Expand Down Expand Up @@ -50,7 +51,7 @@ func newFilter(sl ...string) (*filter, error) {
}

var supportedFilters = []string{
"", "distro", "arch", "type", "bootmode",
"", "distro", "arch", "type", "bootmode", "pkg",
}

type term struct {
Expand All @@ -66,27 +67,51 @@ type filter struct {

// Matches returns true if the given (distro,arch,imgType) tuple matches
// the filter expressions
func (fl filter) Matches(distro distro.Distro, arch distro.Arch, imgType distro.ImageType) bool {
func (fl filter) Matches(dist distro.Distro, arch distro.Arch, imgType distro.ImageType) bool {
m := true

for _, term := range fl.terms {
switch term.prefix {
case "":
// no prefix, do a "fuzzy" search accross the common
// things users may want
m1 := term.pattern.Match(distro.Name())
m1 := term.pattern.Match(dist.Name())
m2 := term.pattern.Match(arch.Name())
m3 := term.pattern.Match(imgType.Name())
m = m && (m1 || m2 || m3)
case "distro":
m = m && term.pattern.Match(distro.Name())
m = m && term.pattern.Match(dist.Name())
case "arch":
m = m && term.pattern.Match(arch.Name())
case "type":
m = m && term.pattern.Match(imgType.Name())
// mostly here to show how flexible this is
case "bootmode":
m = m && term.pattern.Match(imgType.BootMode().String())
case "pkg":
m = m && containsPackages(imgType, term.pattern)
}
}
return m
}

func containsPackages(imgType distro.ImageType, pattern glob.Glob) bool {
var bp blueprint.Blueprint
manifest, _, err := imgType.Manifest(&bp, distro.ImageOptions{}, nil, 0)
if err != nil {
// XXX: some imgTypes like "iot-*", "edge-*" require a ostree
// url to get instanciated so we miss a bunch of types here
return false
}

for _, pkgSets := range manifest.GetPackageSetChains() {
for _, pkgSet := range pkgSets {
for _, pkg := range pkgSet.Include {
if pattern.Match(pkg) {
return true
}
}
}
}
return false
}
3 changes: 3 additions & 0 deletions pkg/imagefilter/filter_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,9 @@ func TestImageFilterFilter(t *testing.T) {
// bootmode: prefix
{[]string{"bootmode:uefi"}, "test-distro-1", "test_arch3", "qcow2", false},
{[]string{"bootmode:hybrid"}, "test-distro-1", "test_arch3", "qcow2", true},
// pkg: prefix
{[]string{"pkg:rando-no"}, "test-distro-1", "test_arch3", "qcow2", false},
{[]string{"pkg:dep-package1"}, "test-distro-1", "test_arch3", "qcow2", true},
// multiple filters are AND
{[]string{"distro:test-distro-1", "type:ami"}, "test-distro-1", "test_arch3", "qcow2", false},
{[]string{"distro:test-distro-1", "type:qcow2"}, "test-distro-1", "test_arch3", "qcow2", true},
Expand Down

0 comments on commit 8aacd8e

Please sign in to comment.