2015-12-28 20:03:29 +00:00
|
|
|
// Copyright 2015 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 detectors exposes functions to register and use container
|
|
|
|
// information extractors.
|
|
|
|
package detectors
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/coreos/clair/database"
|
|
|
|
)
|
|
|
|
|
|
|
|
// The NamespaceDetector interface defines a way to detect a Namespace from input data.
|
|
|
|
// A namespace is usually made of an Operating System name and its version.
|
|
|
|
type NamespaceDetector interface {
|
|
|
|
// Detect detects a Namespace and its version from input data.
|
|
|
|
Detect(map[string][]byte) *database.Namespace
|
|
|
|
// GetRequiredFiles returns the list of files required for Detect, without
|
|
|
|
// leading /.
|
|
|
|
GetRequiredFiles() []string
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
namespaceDetectorsLock sync.Mutex
|
|
|
|
namespaceDetectors = make(map[string]NamespaceDetector)
|
|
|
|
)
|
|
|
|
|
|
|
|
// RegisterNamespaceDetector provides a way to dynamically register an implementation of a
|
|
|
|
// NamespaceDetector.
|
|
|
|
//
|
|
|
|
// If RegisterNamespaceDetector is called twice with the same name if NamespaceDetector is nil,
|
|
|
|
// or if the name is blank, it panics.
|
|
|
|
func RegisterNamespaceDetector(name string, f NamespaceDetector) {
|
|
|
|
if name == "" {
|
|
|
|
panic("Could not register a NamespaceDetector with an empty name")
|
|
|
|
}
|
|
|
|
if f == nil {
|
|
|
|
panic("Could not register a nil NamespaceDetector")
|
|
|
|
}
|
|
|
|
|
|
|
|
namespaceDetectorsLock.Lock()
|
|
|
|
defer namespaceDetectorsLock.Unlock()
|
|
|
|
|
|
|
|
if _, alreadyExists := namespaceDetectors[name]; alreadyExists {
|
|
|
|
panic(fmt.Sprintf("Detector '%s' is already registered", name))
|
|
|
|
}
|
|
|
|
namespaceDetectors[name] = f
|
|
|
|
}
|
|
|
|
|
2016-04-21 09:01:18 +00:00
|
|
|
// DetectNamespaces finds the namespaces of the layer by using every registered NamespaceDetector.
|
|
|
|
func DetectNamespaces(data map[string][]byte) (namespaces []database.Namespace) {
|
2015-12-28 20:03:29 +00:00
|
|
|
for _, detector := range namespaceDetectors {
|
|
|
|
if namespace := detector.Detect(data); namespace != nil {
|
2016-04-21 09:01:18 +00:00
|
|
|
namespaces = append(namespaces, *namespace)
|
2015-12-28 20:03:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-21 09:01:18 +00:00
|
|
|
return
|
2015-12-28 20:03:29 +00:00
|
|
|
}
|
|
|
|
|
2016-04-21 09:01:18 +00:00
|
|
|
// GetRequiredFilesNamespace returns the list of files required for DetectNamespaces for every
|
2015-12-28 20:03:29 +00:00
|
|
|
// registered NamespaceDetector, without leading /.
|
|
|
|
func GetRequiredFilesNamespace() (files []string) {
|
|
|
|
for _, detector := range namespaceDetectors {
|
|
|
|
files = append(files, detector.GetRequiredFiles()...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|