-
Notifications
You must be signed in to change notification settings - Fork 86
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
rpm: implement an RPM filescanner to discern RPM filepaths
Using the filepaths discovered by the RPM filescanner we can judge whether or not a language package has been installed via RPM or not. Signed-off-by: crozzy <[email protected]>
- Loading branch information
Showing
15 changed files
with
672 additions
and
50 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,165 @@ | ||
package rpm | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
"io" | ||
"io/fs" | ||
"os" | ||
"path" | ||
"runtime/trace" | ||
|
||
"github.com/quay/zlog" | ||
|
||
"github.com/quay/claircore" | ||
"github.com/quay/claircore/indexer" | ||
"github.com/quay/claircore/rpm/bdb" | ||
"github.com/quay/claircore/rpm/ndb" | ||
"github.com/quay/claircore/rpm/sqlite" | ||
) | ||
|
||
const ( | ||
scannerName = "rpm" | ||
scannerVersion = "1" | ||
scannerKind = "file" | ||
) | ||
|
||
var ( | ||
_ indexer.FileScanner = (*FileScanner)(nil) | ||
_ indexer.VersionedScanner = (*FileScanner)(nil) | ||
) | ||
|
||
type FileScanner struct{} | ||
|
||
func (*FileScanner) Name() string { return scannerName } | ||
|
||
func (*FileScanner) Version() string { return scannerVersion } | ||
|
||
func (*FileScanner) Kind() string { return scannerKind } | ||
|
||
func (s *FileScanner) Scan(ctx context.Context, layer *claircore.Layer) ([]claircore.File, error) { | ||
if err := ctx.Err(); err != nil { | ||
return nil, err | ||
} | ||
defer trace.StartRegion(ctx, "Scanner.Scan").End() | ||
trace.Log(ctx, "layer", layer.Hash.String()) | ||
ctx = zlog.ContextWithValues(ctx, | ||
"component", "rpm/FileScanner.Scan", | ||
"version", s.Version(), | ||
"layer", layer.Hash.String()) | ||
zlog.Debug(ctx).Msg("start") | ||
defer zlog.Debug(ctx).Msg("done") | ||
|
||
sys, err := layer.FS() | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: unable to open layer: %w", err) | ||
} | ||
|
||
found := make([]foundDB, 0) | ||
if err := fs.WalkDir(sys, ".", findDBs(ctx, &found, sys)); err != nil { | ||
return nil, fmt.Errorf("rpm: error walking fs: %w", err) | ||
} | ||
if len(found) == 0 { | ||
return nil, nil | ||
} | ||
|
||
files := []claircore.File{} | ||
done := map[string]struct{}{} | ||
|
||
zlog.Debug(ctx).Int("count", len(found)).Msg("found possible databases") | ||
for _, db := range found { | ||
ctx := zlog.ContextWithValues(ctx, "db", db.String()) | ||
zlog.Debug(ctx).Msg("examining database") | ||
if _, ok := done[db.Path]; ok { | ||
zlog.Debug(ctx).Msg("already seen, skipping") | ||
continue | ||
} | ||
done[db.Path] = struct{}{} | ||
|
||
var nat nativeDB // see native_db.go:/nativeDB | ||
switch db.Kind { | ||
case kindSQLite: | ||
r, err := sys.Open(path.Join(db.Path, `rpmdb.sqlite`)) | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: error reading sqlite db: %w", err) | ||
} | ||
defer func() { | ||
if err := r.Close(); err != nil { | ||
zlog.Warn(ctx).Err(err).Msg("unable to close sqlite db") | ||
} | ||
}() | ||
f, err := os.CreateTemp(os.TempDir(), `rpmdb.sqlite.*`) | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: error reading sqlite db: %w", err) | ||
} | ||
defer func() { | ||
if err := os.Remove(f.Name()); err != nil { | ||
zlog.Error(ctx).Err(err).Msg("unable to unlink sqlite db") | ||
} | ||
if err := f.Close(); err != nil { | ||
zlog.Warn(ctx).Err(err).Msg("unable to close sqlite db") | ||
} | ||
}() | ||
zlog.Debug(ctx).Str("file", f.Name()).Msg("copying sqlite db out of FS") | ||
if _, err := io.Copy(f, r); err != nil { | ||
return nil, fmt.Errorf("rpm: error reading sqlite db: %w", err) | ||
} | ||
if err := f.Sync(); err != nil { | ||
return nil, fmt.Errorf("rpm: error reading sqlite db: %w", err) | ||
} | ||
sdb, err := sqlite.Open(f.Name()) | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: error reading sqlite db: %w", err) | ||
} | ||
defer sdb.Close() | ||
nat = sdb | ||
case kindBDB: | ||
f, err := sys.Open(path.Join(db.Path, `Packages`)) | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: error reading bdb db: %w", err) | ||
} | ||
defer f.Close() | ||
r, done, err := mkAt(ctx, db.Kind, f) | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: error reading bdb db: %w", err) | ||
} | ||
defer done() | ||
var bpdb bdb.PackageDB | ||
if err := bpdb.Parse(r); err != nil { | ||
return nil, fmt.Errorf("rpm: error parsing bdb db: %w", err) | ||
} | ||
nat = &bpdb | ||
case kindNDB: | ||
f, err := sys.Open(path.Join(db.Path, `Packages.db`)) | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: error reading ndb db: %w", err) | ||
} | ||
defer f.Close() | ||
r, done, err := mkAt(ctx, db.Kind, f) | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: error reading ndb db: %w", err) | ||
} | ||
defer done() | ||
var npdb ndb.PackageDB | ||
if err := npdb.Parse(r); err != nil { | ||
return nil, fmt.Errorf("rpm: error parsing ndb db: %w", err) | ||
} | ||
nat = &npdb | ||
default: | ||
panic("programmer error: bad kind: " + db.Kind.String()) | ||
} | ||
if err := nat.Validate(ctx); err != nil { | ||
zlog.Warn(ctx). | ||
Err(err). | ||
Msg("rpm: invalid native DB") | ||
continue | ||
} | ||
fs, err := filesFromDB(ctx, db.String(), nat) | ||
if err != nil { | ||
return nil, fmt.Errorf("rpm: error reading native db: %w", err) | ||
} | ||
files = append(files, fs...) | ||
} | ||
|
||
return files, nil | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,75 @@ | ||
package rpm | ||
|
||
import ( | ||
"context" | ||
"os" | ||
"path/filepath" | ||
"testing" | ||
|
||
"github.com/quay/zlog" | ||
|
||
"github.com/quay/claircore" | ||
"github.com/quay/claircore/test" | ||
) | ||
|
||
var testcases = []struct { | ||
name string | ||
filename string | ||
expectedFiles int | ||
}{ | ||
{ | ||
name: "java layer", | ||
filename: "cdc13a947214994058941dee5dab876369896ec672defa07694cec6dd3fc7ca2", | ||
expectedFiles: 82, | ||
}, | ||
{ | ||
name: "open jdk layer", | ||
filename: "f68995d3d7382737a1ee41fb69ca9369693173dba4263233621f4defcb29c4bd", | ||
expectedFiles: 218, | ||
}, | ||
} | ||
|
||
func TestFileScannerLayer(t *testing.T) { | ||
ctx := context.Background() | ||
var s FileScanner | ||
desc := claircore.LayerDescription{ | ||
Digest: test.RandomSHA256Digest(t).String(), | ||
URI: "file:///dev/null", | ||
MediaType: test.MediaType, | ||
Headers: make(map[string][]string), | ||
} | ||
|
||
for _, tt := range testcases { | ||
t.Run(tt.name, func(t *testing.T) { | ||
ctx := zlog.Test(ctx, t) | ||
f, err := os.Open(filepath.Join(`testdata/layers`, tt.filename)) | ||
if err != nil { | ||
t.Fatal(err) | ||
} | ||
t.Cleanup(func() { | ||
if err := f.Close(); err != nil { | ||
t.Error(err) | ||
} | ||
}) | ||
var l claircore.Layer | ||
if err := l.Init(ctx, &desc, f); err != nil { | ||
t.Fatal(err) | ||
} | ||
t.Cleanup(func() { | ||
if err := l.Close(); err != nil { | ||
t.Error(err) | ||
} | ||
}) | ||
|
||
got, err := s.Scan(ctx, &l) | ||
if err != nil { | ||
t.Error(err) | ||
} | ||
t.Logf("found %d files", len(got)) | ||
if len(got) != tt.expectedFiles { | ||
t.Fatalf("expected %d files but got %d", tt.expectedFiles, len(got)) | ||
} | ||
t.Log(got) | ||
}) | ||
} | ||
} |
Oops, something went wrong.