Skip to content

Commit

Permalink
clairctl: add import and export commands
Browse files Browse the repository at this point in the history
This commit brings airgap updater support into clair.

Signed-off-by: Hank Donnay <[email protected]>
  • Loading branch information
hdonnay committed Aug 31, 2020
1 parent 98c8ffd commit 050bc2d
Show file tree
Hide file tree
Showing 4 changed files with 221 additions and 0 deletions.
23 changes: 23 additions & 0 deletions cmd/clairctl/config.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
package main

import (
"os"

"gopkg.in/yaml.v3"

"github.com/quay/clair/v4/config"
)

func loadConfig(n string) (*config.Config, error) {
f, err := os.Open(n)
if err != nil {
return nil, err
}
defer f.Close()

var cfg config.Config
if err := yaml.NewDecoder(f).Decode(&cfg); err != nil {
return nil, err
}
return &cfg, nil
}
103 changes: 103 additions & 0 deletions cmd/clairctl/export.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,103 @@
package main

import (
"errors"
"io"
"os"
"regexp"

"github.com/quay/claircore/libvuln"
"github.com/quay/claircore/libvuln/driver"
"github.com/quay/claircore/updater"
_ "github.com/quay/claircore/updater/defaults"
"github.com/urfave/cli/v2"
)

// ExportCmd is the "export-updaters" subcommand.
var ExportCmd = &cli.Command{
Name: "export-updaters",
Description: "Run configured exporters and export to a file.",
Action: exportAction,
Usage: "run updaters and export results",
ArgsUsage: "[out]",
Flags: []cli.Flag{
// Strict can be used to check that updaters still work.
&cli.BoolFlag{
Name: "strict",
Usage: "Return non-zero exit when updaters report errors.",
},
&cli.PathFlag{
Name: "config",
Aliases: []string{"c"},
Usage: "clair configuration file",
Value: "config.yaml",
TakesFile: true,
},
},
}

func exportAction(c *cli.Context) error {
ctx := c.Context
var out io.Writer

// Setup the output file.
args := c.Args()
switch args.Len() {
case 0:
out = os.Stdout
case 1:
f, err := os.Create(args.First())
if err != nil {
return err
}
defer f.Close()
out = f
default:
return errors.New("too many arguments (wanted at most one)")
}

// Read and process the config file.
cfg, err := loadConfig(c.String("config"))
if err != nil {
return err
}
filter, err := regexp.Compile(cfg.Updaters.Filter)
if err != nil {
return err
}
cfgs := make(map[string]driver.ConfigUnmarshaler, len(cfg.Updaters.Config))
for name, node := range cfg.Updaters.Config {
cfgs[name] = node.Decode
}

cl, _, err := cfg.Client(nil)
if err != nil {
return err
}

u, err := libvuln.NewOfflineUpdater(cfgs, filter.MatchString, out)
if err != nil {
return err
}

defs := updater.Registered()
cfg.Updaters.FilterSets(defs)
if err := updater.Configure(ctx, defs, cfgs, cl); err != nil {
return err
}
ufs := make([]driver.UpdaterSetFactory, 0, len(defs))
for _, u := range defs {
ufs = append(ufs, u)
}

if err := u.RunUpdaters(ctx, ufs...); err != nil {
// Don't exit non-zero if we run into errors, unless the strict flag was
// provided.
code := 0
if c.Bool("strict") {
code = 1
}
return cli.Exit(err.Error(), code)
}
return nil
}
93 changes: 93 additions & 0 deletions cmd/clairctl/import.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,93 @@
package main

import (
"errors"
"fmt"
"io"
"net/http"
"net/url"
"os"

"github.com/jackc/pgx/v4/pgxpool"
"github.com/quay/claircore/libvuln"
"github.com/urfave/cli/v2"
)

// ImportCmd is the "import-updaters" subcommand.
var ImportCmd = &cli.Command{
Name: "import-updaters",
Description: "Import updates from a file.",
Action: importAction,
Usage: "import updates",
ArgsUsage: "in",
Flags: []cli.Flag{
&cli.PathFlag{
Name: "config",
Aliases: []string{"c"},
Usage: "clair configuration file",
Value: "config.yaml",
TakesFile: true,
},
},
}

func importAction(c *cli.Context) error {
ctx := c.Context
// Read and process the config file.
cfg, err := loadConfig(c.String("config"))
if err != nil {
return err
}

cl, _, err := cfg.Client(nil)
if err != nil {
return err
}

// Setup the input file.
args := c.Args()
if args.Len() != 1 {
return errors.New("need at least one argument")
}
inName := args.First()

var in io.Reader
u, uerr := url.Parse(inName)
f, ferr := os.Open(inName)
if f != nil {
defer f.Close()
}
switch {
case uerr == nil:
req, err := http.NewRequestWithContext(ctx, http.MethodGet, u.String(), nil)
if err != nil {
return err
}
res, err := cl.Do(req)
if res != nil {
defer res.Body.Close()
}
if err != nil {
return err
}
if res.StatusCode != http.StatusOK {
return fmt.Errorf("unexpected response: %d %s", res.StatusCode, res.Status)
}
in = res.Body
case ferr == nil:
in = f
default:
return fmt.Errorf("unable to make sense of argument %q", inName)
}

pool, err := pgxpool.Connect(ctx, cfg.Matcher.ConnString)
if err != nil {
return err
}
defer pool.Close()

if err := libvuln.OfflineImport(ctx, pool, in); err != nil {
return err
}
return nil
}
2 changes: 2 additions & 0 deletions cmd/clairctl/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,8 @@ func main() {
Commands: []*cli.Command{
ManifestCmd,
ReportCmd,
ExportCmd,
ImportCmd,
},
Flags: []cli.Flag{
&cli.BoolFlag{
Expand Down

0 comments on commit 050bc2d

Please sign in to comment.