2017-01-13 21:48:12 +00:00
|
|
|
// Copyright 2017 clair authors
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
// Package featurens exposes functions to dynamically register methods for
|
|
|
|
// determining a namespace for features present in an image layer.
|
|
|
|
package featurens
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
|
2017-05-04 17:21:25 +00:00
|
|
|
log "github.com/sirupsen/logrus"
|
2017-01-13 21:48:12 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
|
|
|
|
"github.com/coreos/clair/database"
|
|
|
|
"github.com/coreos/clair/pkg/tarutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
detectorsM sync.RWMutex
|
|
|
|
detectors = make(map[string]Detector)
|
|
|
|
)
|
|
|
|
|
|
|
|
// Detector represents an ability to detect a namespace used for organizing
|
|
|
|
// features present in an image layer.
|
|
|
|
type Detector interface {
|
|
|
|
// Detect attempts to determine a Namespace from a FilesMap of an image
|
|
|
|
// layer.
|
|
|
|
Detect(tarutil.FilesMap) (*database.Namespace, error)
|
|
|
|
|
2017-01-13 23:49:17 +00:00
|
|
|
// RequiredFilenames returns the list of files required to be in the FilesMap
|
2017-01-13 21:48:12 +00:00
|
|
|
// provided to the Detect method.
|
2017-01-13 23:49:17 +00:00
|
|
|
//
|
|
|
|
// Filenames must not begin with "/".
|
2017-01-13 21:48:12 +00:00
|
|
|
RequiredFilenames() []string
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterDetector makes a detector available by the provided name.
|
|
|
|
//
|
|
|
|
// If called twice with the same name, the name is blank, or if the provided
|
|
|
|
// Detector is nil, this function panics.
|
|
|
|
func RegisterDetector(name string, d Detector) {
|
|
|
|
if name == "" {
|
|
|
|
panic("namespace: could not register a Detector with an empty name")
|
|
|
|
}
|
|
|
|
if d == nil {
|
|
|
|
panic("namespace: could not register a nil Detector")
|
|
|
|
}
|
|
|
|
|
|
|
|
detectorsM.Lock()
|
|
|
|
defer detectorsM.Unlock()
|
|
|
|
|
|
|
|
if _, dup := detectors[name]; dup {
|
|
|
|
panic("namespace: RegisterDetector called twice for " + name)
|
|
|
|
}
|
|
|
|
|
|
|
|
detectors[name] = d
|
|
|
|
}
|
|
|
|
|
2017-05-12 20:59:17 +00:00
|
|
|
// Detect iterators through all registered Detectors and returns all non-nil detected namespaces
|
2017-07-26 23:22:29 +00:00
|
|
|
func Detect(files tarutil.FilesMap, detectorNames []string) ([]database.Namespace, error) {
|
2017-01-13 21:48:12 +00:00
|
|
|
detectorsM.RLock()
|
|
|
|
defer detectorsM.RUnlock()
|
2017-06-22 15:41:18 +00:00
|
|
|
namespaces := map[string]*database.Namespace{}
|
2017-07-26 23:22:29 +00:00
|
|
|
for _, name := range detectorNames {
|
|
|
|
if detector, ok := detectors[name]; ok {
|
|
|
|
namespace, err := detector.Detect(files)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).WithField("name", name).Warning("failed while attempting to detect namespace")
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if namespace != nil {
|
|
|
|
log.WithFields(log.Fields{"name": name, "namespace": namespace.Name}).Debug("detected namespace")
|
|
|
|
namespaces[namespace.Name] = namespace
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.WithField("Name", name).Warn("Unknown namespace detector")
|
2017-01-13 21:48:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-22 15:41:18 +00:00
|
|
|
nslist := []database.Namespace{}
|
|
|
|
for _, ns := range namespaces {
|
|
|
|
nslist = append(nslist, *ns)
|
|
|
|
}
|
|
|
|
return nslist, nil
|
2017-01-13 21:48:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// RequiredFilenames returns the total list of files required for all
|
|
|
|
// registered Detectors.
|
2017-07-26 23:22:29 +00:00
|
|
|
func RequiredFilenames(detectorNames []string) (files []string) {
|
2017-01-13 23:49:17 +00:00
|
|
|
detectorsM.RLock()
|
|
|
|
defer detectorsM.RUnlock()
|
|
|
|
|
2017-01-13 21:48:12 +00:00
|
|
|
for _, detector := range detectors {
|
|
|
|
files = append(files, detector.RequiredFilenames()...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-26 23:22:29 +00:00
|
|
|
// ListDetectors returns the names of all registered namespace detectors.
|
|
|
|
func ListDetectors() []string {
|
|
|
|
r := []string{}
|
|
|
|
for name := range detectors {
|
|
|
|
r = append(r, name)
|
|
|
|
}
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2017-01-13 23:49:17 +00:00
|
|
|
// TestData represents the data used to test an implementation of Detector.
|
2017-01-13 21:48:12 +00:00
|
|
|
type TestData struct {
|
|
|
|
Files tarutil.FilesMap
|
|
|
|
ExpectedNamespace *database.Namespace
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestDetector runs a Detector on each provided instance of TestData and
|
|
|
|
// asserts the output to be equal to the expected output.
|
|
|
|
func TestDetector(t *testing.T, d Detector, testData []TestData) {
|
|
|
|
for _, td := range testData {
|
|
|
|
namespace, err := d.Detect(td.Files)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
if namespace == nil {
|
|
|
|
assert.Equal(t, td.ExpectedNamespace, namespace)
|
|
|
|
} else {
|
|
|
|
assert.Equal(t, td.ExpectedNamespace.Name, namespace.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|