Skip to content

Commit

Permalink
Merge pull request #1411 from schoenherrg/feature/filter-using-file
Browse files Browse the repository at this point in the history
Feature: Support Reading Filter Expressions from a File
  • Loading branch information
neolynx authored Dec 21, 2024
2 parents fea7acb + c6bb5f7 commit e5b8315
Show file tree
Hide file tree
Showing 19 changed files with 229 additions and 17 deletions.
4 changes: 2 additions & 2 deletions cmd/mirror_create.go
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ func aptlyMirrorCreate(cmd *commander.Command, args []string) error {
return fmt.Errorf("unable to create mirror: %s", err)
}

repo.Filter = context.Flags().Lookup("filter").Value.String()
repo.Filter = context.Flags().Lookup("filter").Value.String() // allows file/stdin with @
repo.FilterWithDeps = context.Flags().Lookup("filter-with-deps").Value.Get().(bool)
repo.SkipComponentCheck = context.Flags().Lookup("force-components").Value.Get().(bool)
repo.SkipArchitectureCheck = context.Flags().Lookup("force-architectures").Value.Get().(bool)
Expand Down Expand Up @@ -103,7 +103,7 @@ Example:
cmd.Flag.Bool("with-installer", false, "download additional not packaged installer files")
cmd.Flag.Bool("with-sources", false, "download source packages in addition to binary packages")
cmd.Flag.Bool("with-udebs", false, "download .udeb packages (Debian installer support)")
cmd.Flag.String("filter", "", "filter packages in mirror")
AddStringOrFileFlag(&cmd.Flag, "filter", "", "filter packages in mirror, use '@file' to read filter from file or '@-' for stdin")
cmd.Flag.Bool("filter-with-deps", false, "when filtering, include dependencies of matching packages as well")
cmd.Flag.Bool("force-components", false, "(only with component list) skip check that requested components are listed in Release file")
cmd.Flag.Bool("force-architectures", false, "(only with architecture list) skip check that requested architectures are listed in Release file")
Expand Down
4 changes: 2 additions & 2 deletions cmd/mirror_edit.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ func aptlyMirrorEdit(cmd *commander.Command, args []string) error {
context.Flags().Visit(func(flag *flag.Flag) {
switch flag.Name {
case "filter":
repo.Filter = flag.Value.String()
repo.Filter = flag.Value.String() // allows file/stdin with @
case "filter-with-deps":
repo.FilterWithDeps = flag.Value.Get().(bool)
case "with-installer":
Expand Down Expand Up @@ -104,7 +104,7 @@ Example:
}

cmd.Flag.String("archive-url", "", "archive url is the root of archive")
cmd.Flag.String("filter", "", "filter packages in mirror")
AddStringOrFileFlag(&cmd.Flag, "filter", "", "filter packages in mirror, use '@file' to read filter from file or '@-' for stdin")
cmd.Flag.Bool("filter-with-deps", false, "when filtering, include dependencies of matching packages as well")
cmd.Flag.Bool("ignore-signatures", false, "disable verification of Release file signatures")
cmd.Flag.Bool("with-installer", false, "download additional not packaged installer files")
Expand Down
7 changes: 6 additions & 1 deletion cmd/package_search.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,11 @@ func aptlyPackageSearch(cmd *commander.Command, args []string) error {
}

if len(args) == 1 {
q, err = query.Parse(args[0])
value, err := GetStringOrFileContent(args[0])
if err != nil {
return fmt.Errorf("unable to read package query from file %s: %w", args[0], err)
}
q, err = query.Parse(value)
if err != nil {
return fmt.Errorf("unable to search: %s", err)
}
Expand Down Expand Up @@ -49,6 +53,7 @@ func makeCmdPackageSearch() *commander.Command {
Long: `
Command search displays list of packages in whole DB that match package query.
Use '@file' to read query from file or '@-' for stdin.
If query is not specified, all the packages are displayed.
Example:
Expand Down
8 changes: 7 additions & 1 deletion cmd/package_show.go
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,11 @@ func aptlyPackageShow(cmd *commander.Command, args []string) error {
return commander.ErrCommandError
}

q, err := query.Parse(args[0])
value, err := GetStringOrFileContent(args[0])
if err != nil {
return fmt.Errorf("unable to read package query from file %s: %w", args[0], err)
}
q, err := query.Parse(value)
if err != nil {
return fmt.Errorf("unable to show: %s", err)
}
Expand Down Expand Up @@ -130,6 +134,8 @@ matching query. Information from Debian control file is displayed.
Optionally information about package files and
inclusion into mirrors/snapshots/local repos is shown.
Use '@file' to read query from file or '@-' for stdin.
Example:
$ aptly package show 'nginx-light_1.2.1-2.2+wheezy2_i386'
Expand Down
8 changes: 7 additions & 1 deletion cmd/repo_move.go
Original file line number Diff line number Diff line change
Expand Up @@ -110,7 +110,11 @@ func aptlyRepoMoveCopyImport(cmd *commander.Command, args []string) error {

queries := make([]deb.PackageQuery, len(args)-2)
for i := 0; i < len(args)-2; i++ {
queries[i], err = query.Parse(args[i+2])
value, err := GetStringOrFileContent(args[i+2])
if err != nil {
return fmt.Errorf("unable to read package query from file %s: %w", args[i+2], err)
}
queries[i], err = query.Parse(value)
if err != nil {
return fmt.Errorf("unable to %s: %s", command, err)
}
Expand Down Expand Up @@ -186,6 +190,8 @@ func makeCmdRepoMove() *commander.Command {
Command move moves packages matching <package-query> from local repo
<src-name> to local repo <dst-name>.
Use '@file' to read package queries from file or '@-' for stdin.
Example:
$ aptly repo move testing stable 'myapp (=0.1.12)'
Expand Down
8 changes: 7 additions & 1 deletion cmd/repo_remove.go
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,11 @@ func aptlyRepoRemove(cmd *commander.Command, args []string) error {

queries := make([]deb.PackageQuery, len(args)-1)
for i := 0; i < len(args)-1; i++ {
queries[i], err = query.Parse(args[i+1])
value, err := GetStringOrFileContent(args[i+1])
if err != nil {
return fmt.Errorf("unable to read package query from file %s: %w", args[i+1], err)
}
queries[i], err = query.Parse(value)
if err != nil {
return fmt.Errorf("unable to remove: %s", err)
}
Expand Down Expand Up @@ -81,6 +85,8 @@ Commands removes packages matching <package-query> from local repository
snapshots, they can be removed completely (including files) by running
'aptly db cleanup'.
Use '@file' to read package queries from file or '@-' for stdin.
Example:
$ aptly repo remove testing 'myapp (=0.1.12)'
Expand Down
8 changes: 7 additions & 1 deletion cmd/snapshot_filter.go
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,11 @@ func aptlySnapshotFilter(cmd *commander.Command, args []string) error {
// Initial queries out of arguments
queries := make([]deb.PackageQuery, len(args)-2)
for i, arg := range args[2:] {
queries[i], err = query.Parse(arg)
value, err := GetStringOrFileContent(arg)
if err != nil {
return fmt.Errorf("unable to read package query from file %s: %w", arg, err)
}
queries[i], err = query.Parse(value)
if err != nil {
return fmt.Errorf("unable to parse query: %s", err)
}
Expand Down Expand Up @@ -103,6 +107,8 @@ Command filter does filtering in snapshot <source>, producing another
snapshot <destination>. Packages could be specified simply
as 'package-name' or as package queries.
Use '@file' syntax to read package queries from file and '@-' to read from stdin.
Example:
$ aptly snapshot filter wheezy-main wheezy-required 'Priority (required)'
Expand Down
8 changes: 7 additions & 1 deletion cmd/snapshot_pull.go
Original file line number Diff line number Diff line change
Expand Up @@ -88,7 +88,11 @@ func aptlySnapshotPull(cmd *commander.Command, args []string) error {
// Initial queries out of arguments
queries := make([]deb.PackageQuery, len(args)-3)
for i, arg := range args[3:] {
queries[i], err = query.Parse(arg)
value, err := GetStringOrFileContent(arg)
if err != nil {
return fmt.Errorf("unable to read package query from file %s: %w", arg, err)
}
queries[i], err = query.Parse(value)
if err != nil {
return fmt.Errorf("unable to parse query: %s", err)
}
Expand Down Expand Up @@ -167,6 +171,8 @@ versions from <source> following dependencies. New snapshot <destination>
is created as a result of this process. Packages could be specified simply
as 'package-name' or as package queries.
Use '@file' syntax to read package queries from file and '@-' to read from stdin.
Example:
$ aptly snapshot pull wheezy-main wheezy-backports wheezy-new-xorg xorg-server-server
Expand Down
8 changes: 7 additions & 1 deletion cmd/snapshot_search.go
Original file line number Diff line number Diff line change
Expand Up @@ -78,7 +78,11 @@ func aptlySnapshotMirrorRepoSearch(cmd *commander.Command, args []string) error
list.PrepareIndex()

if len(args) == 2 {
q, err = query.Parse(args[1])
value, err := GetStringOrFileContent(args[1])
if err != nil {
return fmt.Errorf("unable to read package query from file %s: %w", args[1], err)
}
q, err = query.Parse(value)
if err != nil {
return fmt.Errorf("unable to search: %s", err)
}
Expand Down Expand Up @@ -134,6 +138,8 @@ Command search displays list of packages in snapshot that match package query
If query is not specified, all the packages are displayed.
Use '@file' syntax to read package query from file and '@-' to read from stdin.
Example:
$ aptly snapshot search wheezy-main '$Architecture (i386), Name (% *-dev)'
Expand Down
55 changes: 55 additions & 0 deletions cmd/string_or_file_flag.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
package cmd

import (
"io"
"os"
"strings"

"github.com/smira/flag"
)

// StringOrFileFlag is a custom flag type that can handle both string input and file input.
// If the input starts with '@', it is treated as a filename and the contents are read from the file.
// If the input is '@-', the contents are read from stdin.
type StringOrFileFlag struct {
value string
}

func (s *StringOrFileFlag) String() string {
return s.value
}

func (s *StringOrFileFlag) Set(value string) error {
var err error
s.value, err = GetStringOrFileContent(value)
return err
}

func (s *StringOrFileFlag) Get() any {
return s.value
}

func AddStringOrFileFlag(flagSet *flag.FlagSet, name string, value string, usage string) *StringOrFileFlag {
result := &StringOrFileFlag{value: value}
flagSet.Var(result, name, usage)
return result
}

func GetStringOrFileContent(value string) (string, error) {
if !strings.HasPrefix(value, "@") {
return value, nil
}

filename := strings.TrimPrefix(value, "@")
var data []byte
var err error
if filename == "-" { // Read from stdin
data, err = io.ReadAll(os.Stdin)
} else {
data, err = os.ReadFile(filename)
}
if err != nil {
return "", err
}
return string(data), nil
}
27 changes: 24 additions & 3 deletions man/aptly.1
Original file line number Diff line number Diff line change
Expand Up @@ -588,7 +588,7 @@ Options:
.
.TP
\-\fBfilter\fR=
filter packages in mirror
filter packages in mirror, use \(cq@file\(cq to read filter from file or \(cq@\-\(cq for stdin
.
.TP
\-\fBfilter\-with\-deps\fR
Expand Down Expand Up @@ -771,7 +771,7 @@ archive url is the root of archive
.
.TP
\-\fBfilter\fR=
filter packages in mirror
filter packages in mirror, use \(cq@file\(cq to read filter from file or \(cq@\-\(cq for stdin
.
.TP
\-\fBfilter\-with\-deps\fR
Expand Down Expand Up @@ -1016,6 +1016,9 @@ display list in machine\-readable format
Command move moves packages matching \fIpackage\-query\fR from local repo \fIsrc\-name\fR to local repo \fIdst\-name\fR\.
.
.P
Use \(cq@file\(cq to read package queries from file or \(cq@\-\(cq for stdin\.
.
.P
Example:
.
.P
Expand All @@ -1039,6 +1042,9 @@ follow dependencies when processing package\-spec
Commands removes packages matching \fIpackage\-query\fR from local repository \fIname\fR\. If removed packages are not referenced by other repos or snapshots, they can be removed completely (including files) by running \(cqaptly db cleanup\(cq\.
.
.P
Use \(cq@file\(cq to read package queries from file or \(cq@\-\(cq for stdin\.
.
.P
Example:
.
.P
Expand Down Expand Up @@ -1263,6 +1269,9 @@ $ aptly snapshot verify wheezy\-main wheezy\-contrib wheezy\-non\-free
Command pull pulls new packages along with its\(cq dependencies to snapshot \fIname\fR from snapshot \fIsource\fR\. Pull can upgrade package version in \fIname\fR with versions from \fIsource\fR following dependencies\. New snapshot \fIdestination\fR is created as a result of this process\. Packages could be specified simply as \(cqpackage\-name\(cq or as package queries\.
.
.P
Use \(cq@file\(cq syntax to read package queries from file and \(cq@\-\(cq to read from stdin\.
.
.P
Example:
.
.IP "" 4
Expand Down Expand Up @@ -1398,6 +1407,9 @@ Command search displays list of packages in snapshot that match package query
If query is not specified, all the packages are displayed\.
.
.P
Use \(cq@file\(cq syntax to read package query from file and \(cq@\-\(cq to read from stdin\.
.
.P
Example:
.
.IP "" 4
Expand Down Expand Up @@ -1428,6 +1440,9 @@ include dependencies into search results
Command filter does filtering in snapshot \fIsource\fR, producing another snapshot \fIdestination\fR\. Packages could be specified simply as \(cqpackage\-name\(cq or as package queries\.
.
.P
Use \(cq@file\(cq syntax to read package queries from file and \(cq@\-\(cq to read from stdin\.
.
.P
Example:
.
.IP "" 4
Expand Down Expand Up @@ -2206,7 +2221,7 @@ don\(cqt sign Release files with GPG
Command search displays list of packages in whole DB that match package query\.
.
.P
If query is not specified, all the packages are displayed\.
Use \(cq@file\(cq to read query from file or \(cq@\-\(cq for stdin\. If query is not specified, all the packages are displayed\.
.
.P
Example:
Expand Down Expand Up @@ -2235,6 +2250,9 @@ custom format for result printing
Command shows displays detailed meta\-information about packages matching query\. Information from Debian control file is displayed\. Optionally information about package files and inclusion into mirrors/snapshots/local repos is shown\.
.
.P
Use \(cq@file\(cq to read query from file or \(cq@\-\(cq for stdin\.
.
.P
Example:
.
.IP "" 4
Expand Down Expand Up @@ -2653,5 +2671,8 @@ Blake Kostner (https://github\.com/btkostner)
.IP "\[ci]" 4
Leigh London (https://github\.com/leighlondon)
.
.IP "\[ci]" 4
Gordian Schönherr (https://github\.com/schoenherrg)
.
.IP "" 0

2 changes: 1 addition & 1 deletion system/t03_help/MirrorCreateHelpTest_gold
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ Options:
-dep-follow-source: when processing dependencies, follow from binary to Source packages
-dep-follow-suggests: when processing dependencies, follow Suggests
-dep-verbose-resolve: when processing dependencies, print detailed logs
-filter="": filter packages in mirror
-filter=: filter packages in mirror, use '@file' to read filter from file or '@-' for stdin
-filter-with-deps: when filtering, include dependencies of matching packages as well
-force-architectures: (only with architecture list) skip check that requested architectures are listed in Release file
-force-components: (only with component list) skip check that requested components are listed in Release file
Expand Down
2 changes: 1 addition & 1 deletion system/t03_help/MirrorCreateTest_gold
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ Options:
-dep-follow-source: when processing dependencies, follow from binary to Source packages
-dep-follow-suggests: when processing dependencies, follow Suggests
-dep-verbose-resolve: when processing dependencies, print detailed logs
-filter="": filter packages in mirror
-filter=: filter packages in mirror, use '@file' to read filter from file or '@-' for stdin
-filter-with-deps: when filtering, include dependencies of matching packages as well
-force-architectures: (only with architecture list) skip check that requested architectures are listed in Release file
-force-components: (only with component list) skip check that requested components are listed in Release file
Expand Down
2 changes: 1 addition & 1 deletion system/t03_help/WrongFlagTest_gold
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ Options:
-dep-follow-source: when processing dependencies, follow from binary to Source packages
-dep-follow-suggests: when processing dependencies, follow Suggests
-dep-verbose-resolve: when processing dependencies, print detailed logs
-filter="": filter packages in mirror
-filter=: filter packages in mirror, use '@file' to read filter from file or '@-' for stdin
-filter-with-deps: when filtering, include dependencies of matching packages as well
-force-architectures: (only with architecture list) skip check that requested architectures are listed in Release file
-force-components: (only with component list) skip check that requested components are listed in Release file
Expand Down
4 changes: 4 additions & 0 deletions system/t04_mirror/CreateMirror36Test_gold
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
Downloading: http://repo.aptly.info/system-tests/archive.debian.org/debian-security/dists/stretch/updates/Release

Mirror [mirror36]: http://repo.aptly.info/system-tests/archive.debian.org/debian-security/ stretch/updates successfully added.
You can run 'aptly mirror update mirror36' to download repository contents.
22 changes: 22 additions & 0 deletions system/t04_mirror/CreateMirror36Test_mirror_show
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
Name: mirror36
Archive Root URL: http://repo.aptly.info/system-tests/archive.debian.org/debian-security/
Distribution: stretch/updates
Components: main
Architectures: amd64, arm64, armel, armhf, i386
Download Sources: no
Download .udebs: no
Filter: nginx | Priority (required)
Filter With Deps: no
Last update: never

Information from release file:
Acquire-By-Hash: yes
Architectures: amd64 arm64 armel armhf i386
Codename: stretch
Components: updates/main updates/contrib updates/non-free
Description: Debian 9 Security Updates

Label: Debian-Security
Origin: Debian
Suite: oldoldstable
Version: 9
Loading

0 comments on commit e5b8315

Please sign in to comment.