// 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 oracle implements a vulnerability source updater using the // Oracle Linux OVAL Database. package oracle import ( "bufio" "encoding/xml" "errors" "io" "net/http" "regexp" "strconv" "strings" "gopkg.in/yaml.v2" "github.com/coreos/pkg/capnslog" "github.com/coreos/clair/database" "github.com/coreos/clair/ext/versionfmt" "github.com/coreos/clair/ext/versionfmt/rpm" "github.com/coreos/clair/ext/vulnsrc" "github.com/coreos/clair/pkg/commonerr" ) const ( firstOracle5ELSA = 20070057 ovalURI = "https://linux.oracle.com/oval/" elsaFilePrefix = "com.oracle.elsa-" updaterFlag = "oracleUpdater" ) var ( ignoredCriterions = []string{ " is signed with the Oracle Linux", ".ksplice1.", } elsaRegexp = regexp.MustCompile(`com.oracle.elsa-(\d+).xml`) log = capnslog.NewPackageLogger("github.com/coreos/clair", "ext/vulnsrc/oracle") ) type oval struct { Definitions []definition `xml:"definitions>definition"` } type definition struct { Title string `xml:"metadata>title"` Description string `xml:"metadata>description"` References []reference `xml:"metadata>reference"` Criteria criteria `xml:"criteria"` Severity string `xml:"metadata>advisory>severity"` } type reference struct { Source string `xml:"source,attr"` URI string `xml:"ref_url,attr"` } type criteria struct { Operator string `xml:"operator,attr"` Criterias []*criteria `xml:"criteria"` Criterions []criterion `xml:"criterion"` } type criterion struct { Comment string `xml:"comment,attr"` } type Config struct { Enabled bool } type updater struct{} func init() { vulnsrc.RegisterUpdater("oracle", &updater{}) } func (u *updater) Configure(config *vulnsrc.Config) (bool, error) { var fetcherConfig Config // If no configuration for this fetcher, assume enabled if _, ok := config.Params["oracle"]; !ok { return true, nil } yamlConfig, err := yaml.Marshal(config.Params["oracle"]) if err != nil { return false, errors.New("Invalid configuration for Oracle Linux fetcher.") } err = yaml.Unmarshal(yamlConfig, &fetcherConfig) if err != nil { return false, errors.New("Invalid configuration for Oracle Linux fetcher.") } if fetcherConfig.Enabled == true { return true, nil } else { log.Infof("Oracle Linux fetcher disabled.") return false, nil } } func (u *updater) Update(datastore database.Datastore) (resp vulnsrc.UpdateResponse, err error) { log.Info("fetching Oracle Linux vulnerabilities") // Get the first ELSA we have to manage. flagValue, err := datastore.GetKeyValue(updaterFlag) if err != nil { return resp, err } firstELSA, err := strconv.Atoi(flagValue) if firstELSA == 0 || err != nil { firstELSA = firstOracle5ELSA } // Fetch the update list. r, err := http.Get(ovalURI) if err != nil { log.Errorf("could not download Oracle's update list: %s", err) return resp, commonerr.ErrCouldNotDownload } defer r.Body.Close() // Get the list of ELSAs that we have to process. var elsaList []int scanner := bufio.NewScanner(r.Body) for scanner.Scan() { line := scanner.Text() r := elsaRegexp.FindStringSubmatch(line) if len(r) == 2 { elsaNo, _ := strconv.Atoi(r[1]) if elsaNo > firstELSA { elsaList = append(elsaList, elsaNo) } } } for _, elsa := range elsaList { // Download the ELSA's XML file. r, err := http.Get(ovalURI + elsaFilePrefix + strconv.Itoa(elsa) + ".xml") if err != nil { log.Errorf("could not download Oracle's update file: %s", err) return resp, commonerr.ErrCouldNotDownload } // Parse the XML. vs, err := parseELSA(r.Body) if err != nil { return resp, err } // Collect vulnerabilities. for _, v := range vs { resp.Vulnerabilities = append(resp.Vulnerabilities, v) } } // Set the flag if we found anything. if len(elsaList) > 0 { resp.FlagName = updaterFlag resp.FlagValue = strconv.Itoa(elsaList[len(elsaList)-1]) } else { log.Debug("no Oracle Linux update.") } return resp, nil } func (u *updater) Clean() {} func parseELSA(ovalReader io.Reader) (vulnerabilities []database.Vulnerability, err error) { // Decode the XML. var ov oval err = xml.NewDecoder(ovalReader).Decode(&ov) if err != nil { log.Errorf("could not decode Oracle's XML: %s", err) err = commonerr.ErrCouldNotParse return } // Iterate over the definitions and collect any vulnerabilities that affect // at least one package. for _, definition := range ov.Definitions { pkgs := toFeatureVersions(definition.Criteria) if len(pkgs) > 0 { vulnerability := database.Vulnerability{ Name: name(definition), Link: link(definition), Severity: severity(definition), Description: description(definition), } for _, p := range pkgs { vulnerability.FixedIn = append(vulnerability.FixedIn, p) } vulnerabilities = append(vulnerabilities, vulnerability) } } return } func getCriterions(node criteria) [][]criterion { // Filter useless criterions. var criterions []criterion for _, c := range node.Criterions { ignored := false for _, ignoredItem := range ignoredCriterions { if strings.Contains(c.Comment, ignoredItem) { ignored = true break } } if !ignored { criterions = append(criterions, c) } } if node.Operator == "AND" { return [][]criterion{criterions} } else if node.Operator == "OR" { var possibilities [][]criterion for _, c := range criterions { possibilities = append(possibilities, []criterion{c}) } return possibilities } return [][]criterion{} } func getPossibilities(node criteria) [][]criterion { if len(node.Criterias) == 0 { return getCriterions(node) } var possibilitiesToCompose [][][]criterion for _, criteria := range node.Criterias { possibilitiesToCompose = append(possibilitiesToCompose, getPossibilities(*criteria)) } if len(node.Criterions) > 0 { possibilitiesToCompose = append(possibilitiesToCompose, getCriterions(node)) } var possibilities [][]criterion if node.Operator == "AND" { for _, possibility := range possibilitiesToCompose[0] { possibilities = append(possibilities, possibility) } for _, possibilityGroup := range possibilitiesToCompose[1:] { var newPossibilities [][]criterion for _, possibility := range possibilities { for _, possibilityInGroup := range possibilityGroup { var p []criterion p = append(p, possibility...) p = append(p, possibilityInGroup...) newPossibilities = append(newPossibilities, p) } } possibilities = newPossibilities } } else if node.Operator == "OR" { for _, possibilityGroup := range possibilitiesToCompose { for _, possibility := range possibilityGroup { possibilities = append(possibilities, possibility) } } } return possibilities } func toFeatureVersions(criteria criteria) []database.FeatureVersion { // There are duplicates in Oracle .xml files. // This map is for deduplication. featureVersionParameters := make(map[string]database.FeatureVersion) possibilities := getPossibilities(criteria) for _, criterions := range possibilities { var ( featureVersion database.FeatureVersion osVersion int err error ) // Attempt to parse package data from trees of criterions. for _, c := range criterions { if strings.Contains(c.Comment, " is installed") { const prefixLen = len("Oracle Linux ") osVersion, err = strconv.Atoi(strings.TrimSpace(c.Comment[prefixLen : prefixLen+strings.Index(c.Comment[prefixLen:], " ")])) if err != nil { log.Warningf("could not parse Oracle Linux release version from: '%s'.", c.Comment) } } else if strings.Contains(c.Comment, " is earlier than ") { const prefixLen = len(" is earlier than ") featureVersion.Feature.Name = strings.TrimSpace(c.Comment[:strings.Index(c.Comment, " is earlier than ")]) version := c.Comment[strings.Index(c.Comment, " is earlier than ")+prefixLen:] err := versionfmt.Valid(rpm.ParserName, version) if err != nil { log.Warningf("could not parse package version '%s': %s. skipping", version, err.Error()) } else { featureVersion.Version = version } } } featureVersion.Feature.Namespace.Name = "oracle" + ":" + strconv.Itoa(osVersion) featureVersion.Feature.Namespace.VersionFormat = rpm.ParserName if featureVersion.Feature.Namespace.Name != "" && featureVersion.Feature.Name != "" && featureVersion.Version != "" { featureVersionParameters[featureVersion.Feature.Namespace.Name+":"+featureVersion.Feature.Name] = featureVersion } else { log.Warningf("could not determine a valid package from criterions: %v", criterions) } } // Convert the map to slice. var featureVersionParametersArray []database.FeatureVersion for _, fv := range featureVersionParameters { featureVersionParametersArray = append(featureVersionParametersArray, fv) } return featureVersionParametersArray } func description(def definition) (desc string) { // It is much more faster to proceed like this than using a Replacer. desc = strings.Replace(def.Description, "\n\n\n", " ", -1) desc = strings.Replace(desc, "\n\n", " ", -1) desc = strings.Replace(desc, "\n", " ", -1) return } func name(def definition) string { return strings.TrimSpace(def.Title[:strings.Index(def.Title, ": ")]) } func link(def definition) (link string) { for _, reference := range def.References { if reference.Source == "elsa" { link = reference.URI break } } return } func severity(def definition) database.Severity { switch strings.ToLower(def.Severity) { case "n/a": return database.NegligibleSeverity case "low": return database.LowSeverity case "moderate": return database.MediumSeverity case "important": return database.HighSeverity case "critical": return database.CriticalSeverity default: log.Warningf("could not determine vulnerability severity from: %s.", def.Severity) return database.UnknownSeverity } }