// 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 clairpb import ( "encoding/json" "fmt" "github.com/coreos/clair/database" "github.com/coreos/clair/ext/versionfmt" ) // DatabaseDetectorTypeMapping maps the database detector type to the integer // enum proto. var DatabaseDetectorTypeMapping = map[database.DetectorType]Detector_DType{ database.NamespaceDetectorType: Detector_DType(1), database.FeatureDetectorType: Detector_DType(2), } // PagedVulnerableAncestriesFromDatabaseModel converts database // PagedVulnerableAncestries to api PagedVulnerableAncestries and assigns // indexes to ancestries. func PagedVulnerableAncestriesFromDatabaseModel(dbVuln *database.PagedVulnerableAncestries) (*PagedVulnerableAncestries, error) { if dbVuln == nil { return nil, nil } vuln, err := VulnerabilityFromDatabaseModel(dbVuln.Vulnerability) if err != nil { return nil, err } next := "" if !dbVuln.End { next = string(dbVuln.Next) } vulnAncestry := PagedVulnerableAncestries{ Vulnerability: vuln, CurrentPage: string(dbVuln.Current), NextPage: next, Limit: int32(dbVuln.Limit), } for index, ancestryName := range dbVuln.Affected { indexedAncestry := PagedVulnerableAncestries_IndexedAncestryName{ Name: ancestryName, Index: int32(index), } vulnAncestry.Ancestries = append(vulnAncestry.Ancestries, &indexedAncestry) } return &vulnAncestry, nil } // NotificationFromDatabaseModel converts database notification, old and new // vulnerabilities' paged vulnerable ancestries to be api notification. func NotificationFromDatabaseModel(dbNotification database.VulnerabilityNotificationWithVulnerable) (*GetNotificationResponse_Notification, error) { var ( noti GetNotificationResponse_Notification err error ) noti.Name = dbNotification.Name if !dbNotification.Created.IsZero() { noti.Created = fmt.Sprintf("%d", dbNotification.Created.Unix()) } if !dbNotification.Notified.IsZero() { noti.Notified = fmt.Sprintf("%d", dbNotification.Notified.Unix()) } if !dbNotification.Deleted.IsZero() { noti.Deleted = fmt.Sprintf("%d", dbNotification.Deleted.Unix()) } noti.Old, err = PagedVulnerableAncestriesFromDatabaseModel(dbNotification.Old) if err != nil { return nil, err } noti.New, err = PagedVulnerableAncestriesFromDatabaseModel(dbNotification.New) if err != nil { return nil, err } return ¬i, nil } // VulnerabilityFromDatabaseModel converts database Vulnerability to api Vulnerability. func VulnerabilityFromDatabaseModel(dbVuln database.Vulnerability) (*Vulnerability, error) { metaString := "" if dbVuln.Metadata != nil { metadataByte, err := json.Marshal(dbVuln.Metadata) if err != nil { return nil, err } metaString = string(metadataByte) } return &Vulnerability{ Name: dbVuln.Name, NamespaceName: dbVuln.Namespace.Name, Description: dbVuln.Description, Link: dbVuln.Link, Severity: string(dbVuln.Severity), Metadata: metaString, }, nil } // VulnerabilityWithFixedInFromDatabaseModel converts database VulnerabilityWithFixedIn to api Vulnerability. func VulnerabilityWithFixedInFromDatabaseModel(dbVuln database.VulnerabilityWithFixedIn) (*Vulnerability, error) { vuln, err := VulnerabilityFromDatabaseModel(dbVuln.Vulnerability) if err != nil { return nil, err } vuln.FixedBy = dbVuln.FixedInVersion return vuln, nil } // NamespacedFeatureFromDatabaseModel converts database namespacedFeature to api Feature. func NamespacedFeatureFromDatabaseModel(feature database.AncestryFeature) *Feature { version := feature.Feature.Version if version == versionfmt.MaxVersion { version = "None" } return &Feature{ Name: feature.Feature.Name, Namespace: &Namespace{ Name: feature.Namespace.Name, Detector: DetectorFromDatabaseModel(feature.NamespaceBy), }, VersionFormat: feature.Namespace.VersionFormat, Version: version, Detector: DetectorFromDatabaseModel(feature.FeatureBy), FeatureType: string(feature.Type), } } // DetectorFromDatabaseModel converts database detector to api detector. func DetectorFromDatabaseModel(detector database.Detector) *Detector { if !detector.Valid() { return nil } return &Detector{ Name: detector.Name, Version: detector.Version, Dtype: DatabaseDetectorTypeMapping[detector.DType], } } // DetectorsFromDatabaseModel converts database detectors to api detectors. func DetectorsFromDatabaseModel(dbDetectors []database.Detector) []*Detector { detectors := make([]*Detector, 0, len(dbDetectors)) for _, d := range dbDetectors { detectors = append(detectors, DetectorFromDatabaseModel(d)) } return detectors }