2016-01-06 09:32:59 +00:00
|
|
|
// Copyright 2015, 2016 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 fetchers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/coreos/clair/database"
|
|
|
|
"github.com/coreos/clair/updater"
|
|
|
|
cerrors "github.com/coreos/clair/utils/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
archLinuxCVEURL = "https://wiki.archlinux.org/api.php?action=query&titles=CVE&format=txt&prop=revisions&rvlimit=1&rvprop=content"
|
2016-01-12 21:16:14 +00:00
|
|
|
archLinuxUpdaterFlag = "archLinuxUpdater"
|
2016-01-06 09:32:59 +00:00
|
|
|
tokensRegexp = "{|}|CVF|PKG|Pkg|pkg|\\[|\\]"
|
|
|
|
)
|
|
|
|
|
|
|
|
type SecurityAdvisory struct {
|
|
|
|
Name string
|
|
|
|
URL string
|
|
|
|
}
|
|
|
|
|
2016-01-12 21:16:14 +00:00
|
|
|
// ArchLinuxCVE represents a CVE for Arch Linux
|
|
|
|
type ArchLinuxCVE struct {
|
2016-01-06 09:32:59 +00:00
|
|
|
CVEID string
|
|
|
|
Package string
|
|
|
|
DisclosureDate string
|
|
|
|
AffectedVersion string
|
|
|
|
FixedInVersion string
|
|
|
|
ResponseTime string
|
|
|
|
Status string
|
|
|
|
ASAID SecurityAdvisory
|
|
|
|
}
|
|
|
|
|
2016-01-12 21:16:14 +00:00
|
|
|
// ArchLinuxFetcher implements updater.Fetcher for the Arch Linux CVE
|
2016-01-06 09:32:59 +00:00
|
|
|
// (See wiki : https://wiki.archlinux.org/index.php/CVE).
|
2016-01-12 21:16:14 +00:00
|
|
|
type ArchLinuxFetcher struct{}
|
2016-01-06 09:32:59 +00:00
|
|
|
|
|
|
|
func init() {
|
2016-01-12 21:16:14 +00:00
|
|
|
updater.RegisterFetcher("archlinux", &ArchLinuxFetcher{})
|
2016-01-06 09:32:59 +00:00
|
|
|
}
|
|
|
|
|
2016-01-12 21:16:14 +00:00
|
|
|
// FetchUpdate fetches vulnerability updates from the Arch Linux Security Tracker.
|
|
|
|
func (fetcher *ArchLinuxFetcher) FetchUpdate() (resp updater.FetcherResponse, err error) {
|
|
|
|
log.Info("fetching ArchLinux vulneratibilities")
|
2016-01-06 09:32:59 +00:00
|
|
|
|
|
|
|
r, err := http.Get(archLinuxCVEURL)
|
|
|
|
if err != nil {
|
2016-01-12 21:16:14 +00:00
|
|
|
log.Errorf("could not download ArchLinux CVE wiki content: %s", err)
|
2016-01-06 09:32:59 +00:00
|
|
|
return resp, cerrors.ErrCouldNotDownload
|
|
|
|
}
|
2016-01-12 17:31:54 +00:00
|
|
|
defer r.Body.Close()
|
2016-01-12 21:16:14 +00:00
|
|
|
flag, err := database.GetFlagValue(archLinuxUpdaterFlag)
|
2016-01-06 09:32:59 +00:00
|
|
|
if err != nil {
|
|
|
|
return resp, err
|
|
|
|
}
|
|
|
|
|
2016-01-12 21:16:14 +00:00
|
|
|
resp, err = parseArchLinuxWikiCVE(r.Body, flag)
|
2016-01-06 09:32:59 +00:00
|
|
|
if err != nil {
|
|
|
|
return resp, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
2016-01-12 21:16:14 +00:00
|
|
|
func parseArchLinuxWikiCVE(reader io.Reader, flag string) (resp updater.FetcherResponse, err error) {
|
2016-01-06 09:32:59 +00:00
|
|
|
scanner := bufio.NewScanner(reader)
|
|
|
|
re := regexp.MustCompile(tokensRegexp)
|
|
|
|
for scanner.Scan() {
|
|
|
|
line := scanner.Text()
|
|
|
|
if strings.Contains(line, "{{CVE|CVE") {
|
|
|
|
if !strings.Contains(line, "CVE-2014-????") {
|
2016-01-12 21:16:14 +00:00
|
|
|
cve := buildArchLinuxCVE(re.ReplaceAllString(line, ""))
|
2016-01-06 09:32:59 +00:00
|
|
|
vulnerability := &database.Vulnerability{
|
|
|
|
ID: cve.CVEID,
|
|
|
|
Link: cve.ASAID.URL,
|
|
|
|
Description: cve.ASAID.Name,
|
|
|
|
}
|
|
|
|
resp.Vulnerabilities = append(
|
|
|
|
resp.Vulnerabilities, vulnerability)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
2016-01-12 21:16:14 +00:00
|
|
|
func buildArchLinuxCVE(line string) ArchLinuxCVE {
|
2016-01-06 09:32:59 +00:00
|
|
|
data := strings.Split(strings.TrimSpace(line), "||")
|
|
|
|
sa := SecurityAdvisory{}
|
|
|
|
if len(data) == 8 {
|
|
|
|
dataSecurity := strings.Split(strings.TrimSpace(data[7]), " ")
|
|
|
|
if len(dataSecurity) == 2 {
|
|
|
|
sa.Name = dataSecurity[1]
|
|
|
|
sa.URL = dataSecurity[0]
|
|
|
|
} else {
|
|
|
|
sa.Name = data[7]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
title := data[0]
|
|
|
|
dataTitle := strings.Split(strings.TrimSpace(data[0]), "|")
|
|
|
|
if len(dataTitle) >= 1 {
|
|
|
|
title = dataTitle[2]
|
|
|
|
}
|
2016-01-12 21:16:14 +00:00
|
|
|
return ArchLinuxCVE{
|
2016-01-06 09:32:59 +00:00
|
|
|
CVEID: title,
|
|
|
|
Package: strings.Replace(strings.TrimSpace(data[1]), "|", "", -1),
|
|
|
|
DisclosureDate: strings.TrimSpace(data[2]),
|
|
|
|
AffectedVersion: strings.TrimSpace(data[3]),
|
|
|
|
FixedInVersion: strings.TrimSpace(data[4]),
|
|
|
|
ResponseTime: strings.TrimSpace(data[5]),
|
|
|
|
Status: strings.TrimSpace(data[6]),
|
|
|
|
ASAID: sa,
|
|
|
|
}
|
|
|
|
}
|