CveItemOperator.java
/*
* This file is part of dependency-check-core.
*
* 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.
*
* Copyright (c) 2020 OWASP Foundation. All Rights Reserved.
*/
package org.owasp.dependencycheck.data.nvdcve;
import io.github.jeremylong.openvulnerability.client.nvd.Config;
import java.util.Objects;
import java.util.stream.Collectors;
import org.owasp.dependencycheck.data.nvd.ecosystem.Ecosystem;
import io.github.jeremylong.openvulnerability.client.nvd.DefCveItem;
import io.github.jeremylong.openvulnerability.client.nvd.LangString;
import io.github.jeremylong.openvulnerability.client.nvd.Node;
import java.util.List;
import org.owasp.dependencycheck.dependency.VulnerableSoftware;
/**
*
* Utility for processing {@linkplain DefCveItem} in order to extract key values
* like textual description and ecosystem type.
*
* @author skjolber
*/
public class CveItemOperator {
/**
* The filter for 2.3 CPEs in the CVEs - we don't import unless we get a
* match.
*/
private final String cpeStartsWithFilter;
/**
* Constructs a new CVE Item Operator utility.
*
* @param cpeStartsWithFilter the filter to use for CPE entries
*/
public CveItemOperator(String cpeStartsWithFilter) {
this.cpeStartsWithFilter = cpeStartsWithFilter;
}
/**
* Extracts the english description from the CVE object.
*
* @param cve the CVE data
* @return the English descriptions from the CVE object
*/
public String extractDescription(DefCveItem cve) {
return cve.getCve().getDescriptions().stream().filter((desc)
-> "en".equals(desc.getLang())).map(LangString::getValue).collect(Collectors.joining(" "));
}
//CSOFF: MissingSwitchDefault
/**
* Attempts to determine the ecosystem based on the vendor, product and
* targetSw.
*
* @param baseEcosystem the base ecosystem
* @param vendor the vendor
* @param product the product
* @param targetSw the target software
* @return the ecosystem if one is identified
*/
private String extractEcosystem(String baseEcosystem, String vendor, String product, String targetSw) {
//TODO the following was added to reduce the need for the slow UPDATE_ECOSYSTEM2 query
// the following should be analyzed to determine if an ecosystem should be returned.
// Note that these all have 'bindings' in the description of a vulnerability in more than
// one case these were related to language bindings; as such the list need to be reviewed and refined.
if (("mysql".equals(vendor) && "mysql".equals(product))
|| ("postgresql".equals(vendor) && "postgresql".equals(product))
|| ("picketlink".equals(vendor) && "picketlink".equals(product))
|| ("libxl_project".equals(vendor) && "libxl".equals(product))
|| ("ocaml".equals(vendor) && "postgresql-ocaml".equals(product))
|| ("curses_project".equals(vendor) && "curses".equals(product))
|| ("dalekjs".equals(vendor) && "dalekjs".equals(product))
|| ("microsoft".equals(vendor) && "internet_explorer".equals(product))
|| ("jenkins".equals(vendor) && "ssh_credentials".equals(product))
|| ("kubernetes".equals(vendor) && "kubernetes".equals(product))
|| ("gnome".equals(vendor) && "nautilus-python".equals(product))
|| ("apache".equals(vendor) && "qpid_proton".equals(product))
|| ("mysql-ocaml".equals(vendor) && "mysql-ocaml".equals(product))
|| ("google".equals(vendor) && "chrome".equals(product))
|| ("canonical".equals(vendor) && "ltsp_display_manager".equals(product))
|| ("gnome".equals(vendor) && "vala".equals(product))
|| ("apple".equals(vendor) && "safari".equals(product))
|| ("mapbox".equals(vendor) && "npm-test-sqlite3-trunk".equals(product))
|| ("apple".equals(vendor) && "webkit".equals(product))
|| ("mozilla".equals(vendor) && "firefox".equals(product))
|| ("apache".equals(vendor) && "thrift".equals(product))
|| ("apache".equals(vendor) && "qpid".equals(product))
|| ("mozilla".equals(vendor) && "thunderbird".equals(product))
|| ("mozilla".equals(vendor) && "firefox_esr".equals(product))
|| ("redhat".equals(vendor) && "jboss_amq_clients_2".equals(product))
|| ("node-opencv_project".equals(vendor) && "node-opencv".equals(product))
|| ("mozilla".equals(vendor) && "seamonkey".equals(product))
|| ("mozilla".equals(vendor) && "thunderbird_esr".equals(product))
|| ("mnet_soft_factory".equals(vendor) && "nodemanager_professional".equals(product))
|| ("mozilla".equals(vendor) && "mozilla_suite".equals(product))
|| ("theforeman".equals(vendor) && "hammer_cli".equals(product))
|| ("ibm".equals(vendor) && "websphere_application_server".equals(product))
|| ("sap".equals(vendor) && "hana_extend_application_services".equals(product))
|| ("apache".equals(vendor) && "zookeeper".equals(product))) {
return null;
}
if ("ibm".equals(vendor)
&& "java".equals(product)) {
return Ecosystem.NATIVE;
}
if ("oracle".equals(vendor)
&& "vm".equals(product)) {
return Ecosystem.NATIVE;
}
switch (targetSw) {
case "asp.net"://.net
case "c#"://.net
case ".net"://.net
case "dotnetnuke"://.net
return Ecosystem.DOTNET;
case "android"://android
case "java"://java
return Ecosystem.JAVA;
case "c/c++"://c++
case "borland_c++"://c++
case "visual_c++"://c++
case "gnu_c++"://c++
case "linux_kernel"://native
case "linux"://native
case "unix"://native
case "suse_linux"://native
case "redhat_enterprise_linux"://native
case "debian"://native
return Ecosystem.NATIVE;
case "coldfusion"://coldfusion
return Ecosystem.COLDFUSION;
case "ios"://ios
case "iphone"://ios
case "ipad"://ios
case "iphone_os"://ios
return Ecosystem.IOS;
case "jquery"://javascript
return Ecosystem.JAVASCRIPT;
case "node.js"://node.js
case "nodejs"://node.js
return Ecosystem.NODEJS;
case "perl"://perl
return Ecosystem.PERL;
case "joomla!"://php
case "joomla"://php
case "mybb"://php
case "simplesamlphp"://php
case "craft_cms"://php
case "moodle"://php
case "phpcms"://php
case "buddypress"://php
case "typo3"://php
case "php"://php
case "wordpress"://php
case "drupal"://php
case "mediawiki"://php
case "symfony"://php
case "openpne"://php
case "vbulletin3"://php
case "vbulletin4"://php
return Ecosystem.PHP;
case "python"://python
return Ecosystem.PYTHON;
case "ruby"://ruby
return Ecosystem.RUBY;
}
return baseEcosystem;
}
//CSON: MissingSwitchDefault
/**
* Attempts to determine the ecosystem based on the vendor, product and
* targetSw.
*
* @param baseEcosystem the base ecosystem
* @param parsedCpe the CPE identifier
* @return the ecosystem if one is identified
*/
public String extractEcosystem(String baseEcosystem, VulnerableSoftware parsedCpe) {
return extractEcosystem(baseEcosystem, parsedCpe.getVendor(), parsedCpe.getProduct(), parsedCpe.getTargetSw());
}
/**
* Determines if the CVE entry is rejected.
*
* @param description the CVE description
* @return <code>true</code> if the CVE was rejected; otherwise
* <code>false</code>
*/
public boolean isRejected(String description) {
return description.startsWith("** REJECT **") || description.startsWith("DO NOT USE THIS CANDIDATE NUMBER");
}
/**
* Tests the CVE's CPE entries against the starts with filter. In general
* this limits the CVEs imported to just application level vulnerabilities.
*
* @param cve the CVE entry to examine
* @return <code>true</code> if the CVE affects CPEs identified by the
* configured CPE Starts with filter
*/
boolean testCveCpeStartWithFilter(final DefCveItem cve) {
if (cve.getCve().getConfigurations() != null) {
//cycle through to see if this is a CPE we care about (use the CPE filters
return cve.getCve().getConfigurations().stream()
.map(Config::getNodes)
.flatMap(List::stream)
.filter(Objects::nonNull)
.map(Node::getCpeMatch)
.filter(Objects::nonNull)
.flatMap(List::stream)
.filter(cpe -> cpe != null && cpe.getCriteria() != null)
.anyMatch(cpe -> cpe.getCriteria().startsWith(cpeStartsWithFilter));
}
return false;
}
}