b66ec762a3
Handle scanning of openSUSE and SUSE Linux Enterprise images. Signed-off-by: Flavio Castelli <fcastelli@suse.com>
474 lines
13 KiB
Go
474 lines
13 KiB
Go
// 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 suse implements a vulnerability source updater using the
|
|
// SUSE Linux and openSUSE OVAL Database.
|
|
package suse
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/xml"
|
|
"io"
|
|
"net/http"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
"fmt"
|
|
|
|
"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 (
|
|
ovalURI = "http://ftp.suse.com/pub/projects/security/oval/"
|
|
)
|
|
|
|
var (
|
|
ignoredCriterions []string
|
|
suseOpenSUSEInstalledCommentRegexp = regexp.MustCompile(`(SUSE Linux Enterprise |openSUSE ).*is installed`)
|
|
suseInstalledCommentRegexp = regexp.MustCompile(`SUSE Linux Enterprise[A-Za-z\s]*? (\d+)[\w\s]*?(SP(\d+))? is installed`)
|
|
)
|
|
|
|
type oval struct {
|
|
Timestamp string `xml:"generator>timestamp"`
|
|
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"`
|
|
}
|
|
|
|
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 flavor int
|
|
|
|
const (
|
|
SUSE flavor = iota
|
|
OpenSUSE
|
|
)
|
|
|
|
type updater struct {
|
|
Name string
|
|
NamespaceName string
|
|
FilePrefix string
|
|
UpdaterFlag string
|
|
FileRegexp *regexp.Regexp
|
|
}
|
|
|
|
func newUpdater(f flavor) updater {
|
|
var up updater
|
|
|
|
switch f {
|
|
case SUSE:
|
|
up.Name = "SUSE Linux"
|
|
up.NamespaceName = "sles"
|
|
up.FilePrefix = "suse.linux.enterprise."
|
|
up.UpdaterFlag = "SUSEUpdater"
|
|
up.FileRegexp = regexp.MustCompile(`suse.linux.enterprise.(\d+).xml`)
|
|
case OpenSUSE:
|
|
up.Name = "openSUSE"
|
|
up.NamespaceName = "opensuse"
|
|
up.FilePrefix = "opensuse.leap."
|
|
up.UpdaterFlag = "openSUSEUpdater"
|
|
up.FileRegexp = regexp.MustCompile(`opensuse.leap.(\d+\.*\d*).xml`)
|
|
default:
|
|
panic("Unrecognized flavor")
|
|
}
|
|
|
|
return up
|
|
}
|
|
|
|
func init() {
|
|
suseUpdater := newUpdater(SUSE)
|
|
openSUSEUpdater := newUpdater(OpenSUSE)
|
|
vulnsrc.RegisterUpdater("suse", &suseUpdater)
|
|
vulnsrc.RegisterUpdater("opensuse", &openSUSEUpdater)
|
|
}
|
|
|
|
func (u *updater) Update(datastore database.Datastore) (resp vulnsrc.UpdateResponse, err error) {
|
|
log.WithField("package", u.Name).Info("Start fetching vulnerabilities")
|
|
|
|
tx, err := datastore.Begin()
|
|
if err != nil {
|
|
return resp, err
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
// openSUSE and SUSE have one single xml file for all the products, there are no incremental
|
|
// xml files. We store into the database the value of the generation timestamp
|
|
// of the latest file we parsed.
|
|
flagValue, ok, err := tx.FindKeyValue(u.UpdaterFlag)
|
|
if err != nil {
|
|
return resp, err
|
|
}
|
|
log.WithField("flagvalue", flagValue)
|
|
|
|
if !ok {
|
|
flagValue = "0"
|
|
}
|
|
|
|
// this contains the modification time of the most recent
|
|
// file expressed as unix time (int64)
|
|
latestOval, _ := strconv.ParseInt(flagValue, 10, 64)
|
|
|
|
// Fetch the update list.
|
|
r, err := http.Get(ovalURI)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
defer r.Body.Close()
|
|
|
|
var ovalFiles []string
|
|
var generationTimes []int64
|
|
|
|
scanner := bufio.NewScanner(r.Body)
|
|
for scanner.Scan() {
|
|
line := scanner.Text()
|
|
r := u.FileRegexp.FindStringSubmatch(line)
|
|
if len(r) != 2 {
|
|
continue
|
|
}
|
|
|
|
ovalFile := ovalURI + u.FilePrefix + r[1] + ".xml"
|
|
log.WithFields(
|
|
log.Fields{
|
|
"ovalFile": ovalFile,
|
|
"updater": u.Name,
|
|
}).Debug("file to check")
|
|
|
|
// do not fetch the entire file to get the value of the
|
|
// creation time. Rely on the "latest modified time"
|
|
// value of the file hosted on the remote server.
|
|
timestamp, err := getLatestModifiedTime(ovalFile)
|
|
if err != nil {
|
|
log.WithError(err).WithField("ovalFile", ovalFile).Warning("Ignoring OVAL file")
|
|
}
|
|
|
|
if timestamp > latestOval {
|
|
ovalFiles = append(ovalFiles, ovalFile)
|
|
}
|
|
}
|
|
|
|
for _, oval := range ovalFiles {
|
|
// Download the oval XML file.
|
|
r, err := http.Get(oval)
|
|
if err != nil {
|
|
log.WithError(err).Error("could not download", u.Name, "update list")
|
|
return resp, commonerr.ErrCouldNotDownload
|
|
}
|
|
|
|
match := u.FileRegexp.FindStringSubmatch(oval)
|
|
if len(match) != 2 {
|
|
log.Error("Skipping ", oval, "because it's not possible to extract osVersion")
|
|
continue
|
|
}
|
|
osVersion := match[1]
|
|
|
|
// Parse the XML.
|
|
vs, generationTime, err := parseOval(r.Body, u.NamespaceName, osVersion)
|
|
if err != nil {
|
|
return resp, err
|
|
}
|
|
generationTimes = append(generationTimes, generationTime)
|
|
|
|
// Collect vulnerabilities.
|
|
for _, v := range vs {
|
|
resp.Vulnerabilities = append(resp.Vulnerabilities, v)
|
|
}
|
|
}
|
|
|
|
// Set the flag if we found anything.
|
|
if len(generationTimes) > 0 {
|
|
resp.FlagName = u.UpdaterFlag
|
|
resp.FlagValue = strconv.FormatInt(latest(generationTimes), 10)
|
|
} else {
|
|
log.WithField("package", u.Name).Debug("no update")
|
|
}
|
|
|
|
return resp, nil
|
|
}
|
|
|
|
// Get the latest modification time of a remote file
|
|
// expressed as unix time
|
|
func getLatestModifiedTime(url string) (int64, error) {
|
|
resp, err := http.Head(url)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
last_modified := resp.Header.Get("Last-Modified")
|
|
if len(last_modified) == 0 {
|
|
return 0, fmt.Errorf("last modified header missing")
|
|
}
|
|
|
|
// "Thu, 30 Nov 2017 03:07:57 GMT
|
|
layout := "Mon, 2 Jan 2006 15:04:05 MST"
|
|
timestamp, err := time.Parse(layout, last_modified)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
return timestamp.Unix(), nil
|
|
}
|
|
|
|
func latest(values []int64) (ret int64) {
|
|
for _, element := range values {
|
|
if element > ret {
|
|
ret = element
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (u *updater) Clean() {}
|
|
|
|
func parseOval(ovalReader io.Reader, osFlavor, osVersion string) (vulnerabilities []database.VulnerabilityWithAffected, generationTime int64, err error) {
|
|
// Decode the XML.
|
|
var ov oval
|
|
err = xml.NewDecoder(ovalReader).Decode(&ov)
|
|
if err != nil {
|
|
log.WithError(err).Error("could not decode XML")
|
|
err = commonerr.ErrCouldNotParse
|
|
return
|
|
}
|
|
|
|
// timestamp format 2017-10-23T04:07:14
|
|
layout := "2006-1-2T15:04:05"
|
|
timestamp, err := time.Parse(layout, ov.Timestamp)
|
|
if err != nil {
|
|
return
|
|
}
|
|
generationTime = timestamp.Unix()
|
|
|
|
// Iterate over the definitions and collect any vulnerabilities that affect
|
|
// at least one package.
|
|
for _, definition := range ov.Definitions {
|
|
pkgs := toFeatureVersions(definition.Criteria, osFlavor, osVersion)
|
|
if len(pkgs) > 0 {
|
|
vulnerability := database.VulnerabilityWithAffected{
|
|
Vulnerability: database.Vulnerability{
|
|
Name: name(definition),
|
|
Link: link(definition),
|
|
//TODO: handle that once openSUSE/SLE OVAL files have severity info
|
|
Severity: database.UnknownSeverity,
|
|
Description: description(definition),
|
|
},
|
|
}
|
|
for _, p := range pkgs {
|
|
vulnerability.Affected = append(vulnerability.Affected, 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, osFlavor, osVersion string) []database.AffectedFeature {
|
|
// There are duplicates in SUSE .xml files.
|
|
// This map is for deduplication.
|
|
featureVersionParameters := make(map[string]database.AffectedFeature)
|
|
|
|
possibilities := getPossibilities(criteria)
|
|
for _, criterions := range possibilities {
|
|
var featureVersion database.AffectedFeature
|
|
|
|
// Attempt to parse package data from trees of criterions.
|
|
for _, c := range criterions {
|
|
if match := suseInstalledCommentRegexp.FindStringSubmatch(c.Comment); match != nil {
|
|
if len(match) != 4 {
|
|
log.WithField("comment", c.Comment).Warning("could not extract sles name and version from comment")
|
|
} else {
|
|
osVersion = match[1]
|
|
if match[3] != "" {
|
|
osVersion = fmt.Sprintf("%s.%s", osVersion, match[3])
|
|
}
|
|
}
|
|
}
|
|
|
|
if suseOpenSUSEInstalledCommentRegexp.FindStringSubmatch(c.Comment) == nil && strings.HasSuffix(c.Comment, " is installed") {
|
|
name, version, err := splitPackageNameAndVersion(c.Comment[:len(c.Comment)-13])
|
|
if err != nil {
|
|
log.WithError(err).WithField("comment", c.Comment).Warning("Could not extract package name and version from comment")
|
|
} else {
|
|
featureVersion.FeatureName = name
|
|
version := version
|
|
err := versionfmt.Valid(rpm.ParserName, version)
|
|
if err != nil {
|
|
log.WithError(err).WithField("version", version).Warning("could not parse package version. skipping")
|
|
} else {
|
|
featureVersion.AffectedVersion = version
|
|
if version != versionfmt.MaxVersion {
|
|
featureVersion.FixedInVersion = version
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
featureVersion.Namespace.Name = fmt.Sprintf("%s:%s", osFlavor, osVersion)
|
|
featureVersion.Namespace.VersionFormat = rpm.ParserName
|
|
|
|
if featureVersion.Namespace.Name != "" && featureVersion.FeatureName != "" && featureVersion.AffectedVersion != "" && featureVersion.FixedInVersion != "" {
|
|
featureVersionParameters[featureVersion.Namespace.Name+":"+featureVersion.FeatureName] = featureVersion
|
|
} else {
|
|
log.WithField("criterions", fmt.Sprintf("%v", criterions)).Warning("could not determine a valid package from criterions")
|
|
}
|
|
}
|
|
|
|
// Convert the map to slice.
|
|
var featureVersionParametersArray []database.AffectedFeature
|
|
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)
|
|
}
|
|
|
|
func link(def definition) (link string) {
|
|
for _, reference := range def.References {
|
|
if reference.Source == "CVE" {
|
|
link = reference.URI
|
|
break
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func splitPackageNameAndVersion(fullname string) (name, version string, err error) {
|
|
re := regexp.MustCompile(`-\d+\.`)
|
|
|
|
matches := re.FindStringSubmatchIndex(fullname)
|
|
|
|
if matches == nil {
|
|
err = fmt.Errorf("Cannot extract package name and version from %s", fullname)
|
|
} else {
|
|
name = fullname[:matches[0]]
|
|
version = fullname[matches[0]+1:]
|
|
}
|
|
|
|
return
|
|
}
|