Coverage Report - org.owasp.dependencycheck.analyzer.OpenSSLAnalyzer
 
Classes in this File Line Coverage Branch Coverage Complexity
OpenSSLAnalyzer
91%
32/35
71%
10/14
2.25
 
 1  
 /*
 2  
  * This file is part of dependency-check-core.
 3  
  *
 4  
  * Licensed under the Apache License, Version 2.0 (the "License");
 5  
  * you may not use this file except in compliance with the License.
 6  
  * You may obtain a copy of the License at
 7  
  *
 8  
  *     http://www.apache.org/licenses/LICENSE-2.0
 9  
  *
 10  
  * Unless required by applicable law or agreed to in writing, software
 11  
  * distributed under the License is distributed on an "AS IS" BASIS,
 12  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 13  
  * See the License for the specific language governing permissions and
 14  
  * limitations under the License.
 15  
  *
 16  
  * Copyright (c) 2015 Institute for Defense Analyses. All Rights Reserved.
 17  
  */
 18  
 package org.owasp.dependencycheck.analyzer;
 19  
 
 20  
 import org.apache.commons.io.FileUtils;
 21  
 import org.owasp.dependencycheck.Engine;
 22  
 import org.owasp.dependencycheck.analyzer.exception.AnalysisException;
 23  
 import org.owasp.dependencycheck.dependency.Confidence;
 24  
 import org.owasp.dependencycheck.dependency.Dependency;
 25  
 import org.owasp.dependencycheck.utils.FileFilterBuilder;
 26  
 import org.owasp.dependencycheck.utils.Settings;
 27  
 
 28  
 import java.io.File;
 29  
 import java.io.FileFilter;
 30  
 import java.io.IOException;
 31  
 import java.nio.charset.Charset;
 32  
 import java.util.regex.Matcher;
 33  
 import java.util.regex.Pattern;
 34  
 import org.owasp.dependencycheck.exception.InitializationException;
 35  
 
 36  
 /**
 37  
  * Used to analyze OpenSSL source code present in the file system.
 38  
  *
 39  
  * @author Dale Visser
 40  
  */
 41  12
 public class OpenSSLAnalyzer extends AbstractFileTypeAnalyzer {
 42  
 
 43  
     /**
 44  
      * Hexadecimal.
 45  
      */
 46  
     private static final int HEXADECIMAL = 16;
 47  
     /**
 48  
      * Filename to analyze. All other .h files get removed from consideration.
 49  
      */
 50  
     private static final String OPENSSLV_H = "opensslv.h";
 51  
 
 52  
     /**
 53  
      * Filter that detects files named "__init__.py".
 54  
      */
 55  1
     private static final FileFilter OPENSSLV_FILTER = FileFilterBuilder.newInstance().addFilenames(OPENSSLV_H).build();
 56  
     /**
 57  
      * Open SSL Version number pattern.
 58  
      */
 59  1
     private static final Pattern VERSION_PATTERN = Pattern.compile(
 60  
             "define\\s+OPENSSL_VERSION_NUMBER\\s+0x([0-9a-zA-Z]{8})L", Pattern.DOTALL
 61  
             | Pattern.CASE_INSENSITIVE);
 62  
     /**
 63  
      * The offset of the major version number.
 64  
      */
 65  
     private static final int MAJOR_OFFSET = 28;
 66  
     /**
 67  
      * The mask for the minor version number.
 68  
      */
 69  
     private static final long MINOR_MASK = 0x0ff00000L;
 70  
     /**
 71  
      * The offset of the minor version number.
 72  
      */
 73  
     private static final int MINOR_OFFSET = 20;
 74  
     /**
 75  
      * The max for the fix version.
 76  
      */
 77  
     private static final long FIX_MASK = 0x000ff000L;
 78  
     /**
 79  
      * The offset for the fix version.
 80  
      */
 81  
     private static final int FIX_OFFSET = 12;
 82  
     /**
 83  
      * The mask for the patch version.
 84  
      */
 85  
     private static final long PATCH_MASK = 0x00000ff0L;
 86  
     /**
 87  
      * The offset for the patch version.
 88  
      */
 89  
     private static final int PATCH_OFFSET = 4;
 90  
     /**
 91  
      * Number of letters.
 92  
      */
 93  
     private static final int NUM_LETTERS = 26;
 94  
     /**
 95  
      * The status mask.
 96  
      */
 97  
     private static final int STATUS_MASK = 0x0000000f;
 98  
 
 99  
     /**
 100  
      * Returns the open SSL version as a string.
 101  
      *
 102  
      * @param openSSLVersionConstant The open SSL version
 103  
      * @return the version of openssl
 104  
      */
 105  
     static String getOpenSSLVersion(long openSSLVersionConstant) {
 106  9
         final long major = openSSLVersionConstant >>> MAJOR_OFFSET;
 107  9
         final long minor = (openSSLVersionConstant & MINOR_MASK) >>> MINOR_OFFSET;
 108  9
         final long fix = (openSSLVersionConstant & FIX_MASK) >>> FIX_OFFSET;
 109  9
         final long patchLevel = (openSSLVersionConstant & PATCH_MASK) >>> PATCH_OFFSET;
 110  9
         final String patch = 0 == patchLevel || patchLevel > NUM_LETTERS ? "" : String.valueOf((char) (patchLevel + 'a' - 1));
 111  9
         final int statusCode = (int) (openSSLVersionConstant & STATUS_MASK);
 112  9
         final String status = 0xf == statusCode ? "" : (0 == statusCode ? "-dev" : "-beta" + statusCode);
 113  9
         return String.format("%d.%d.%d%s%s", major, minor, fix, patch, status);
 114  
     }
 115  
 
 116  
     /**
 117  
      * Returns the name of the Python Package Analyzer.
 118  
      *
 119  
      * @return the name of the analyzer
 120  
      */
 121  
     @Override
 122  
     public String getName() {
 123  19
         return "OpenSSL Source Analyzer";
 124  
     }
 125  
 
 126  
     /**
 127  
      * Tell that we are used for information collection.
 128  
      *
 129  
      * @return INFORMATION_COLLECTION
 130  
      */
 131  
     @Override
 132  
     public AnalysisPhase getAnalysisPhase() {
 133  6
         return AnalysisPhase.INFORMATION_COLLECTION;
 134  
     }
 135  
 
 136  
     /**
 137  
      * Returns the set of supported file extensions.
 138  
      *
 139  
      * @return the set of supported file extensions
 140  
      */
 141  
     @Override
 142  
     protected FileFilter getFileFilter() {
 143  859
         return OPENSSLV_FILTER;
 144  
     }
 145  
 
 146  
     /**
 147  
      * No-op initializer implementation.
 148  
      *
 149  
      * @throws InitializationException never thrown
 150  
      */
 151  
     @Override
 152  
     protected void initializeFileTypeAnalyzer() throws InitializationException {
 153  
         // Nothing to do here.
 154  4
     }
 155  
 
 156  
     /**
 157  
      * Analyzes python packages and adds evidence to the dependency.
 158  
      *
 159  
      * @param dependency the dependency being analyzed
 160  
      * @param engine the engine being used to perform the scan
 161  
      * @throws AnalysisException thrown if there is an unrecoverable error
 162  
      * analyzing the dependency
 163  
      */
 164  
     @Override
 165  
     protected void analyzeDependency(Dependency dependency, Engine engine)
 166  
             throws AnalysisException {
 167  1
         final File file = dependency.getActualFile();
 168  1
         final String parentName = file.getParentFile().getName();
 169  1
         boolean found = false;
 170  1
         final String contents = getFileContents(file);
 171  1
         if (!contents.isEmpty()) {
 172  1
             final Matcher matcher = VERSION_PATTERN.matcher(contents);
 173  1
             if (matcher.find()) {
 174  2
                 dependency.getVersionEvidence().addEvidence(OPENSSLV_H, "Version Constant",
 175  1
                         getOpenSSLVersion(Long.parseLong(matcher.group(1), HEXADECIMAL)), Confidence.HIGH);
 176  1
                 found = true;
 177  
             }
 178  
         }
 179  1
         if (found) {
 180  1
             dependency.setDisplayFileName(parentName + File.separatorChar + OPENSSLV_H);
 181  1
             dependency.getVendorEvidence().addEvidence(OPENSSLV_H, "Vendor", "OpenSSL", Confidence.HIGHEST);
 182  1
             dependency.getProductEvidence().addEvidence(OPENSSLV_H, "Product", "OpenSSL", Confidence.HIGHEST);
 183  
         } else {
 184  0
             engine.getDependencies().remove(dependency);
 185  
         }
 186  1
     }
 187  
 
 188  
     /**
 189  
      * Retrieves the contents of a given file.
 190  
      *
 191  
      * @param actualFile the file to read
 192  
      * @return the contents of the file
 193  
      * @throws AnalysisException thrown if there is an IO Exception
 194  
      */
 195  
     private String getFileContents(final File actualFile)
 196  
             throws AnalysisException {
 197  
         try {
 198  1
             return FileUtils.readFileToString(actualFile, Charset.defaultCharset()).trim();
 199  0
         } catch (IOException e) {
 200  0
             throw new AnalysisException(
 201  
                     "Problem occurred while reading dependency file.", e);
 202  
         }
 203  
     }
 204  
 
 205  
     /**
 206  
      * Returns the setting for the analyzer enabled setting key.
 207  
      *
 208  
      * @return the setting for the analyzer enabled setting key
 209  
      */
 210  
     @Override
 211  
     protected String getAnalyzerEnabledSettingKey() {
 212  6
         return Settings.KEYS.ANALYZER_OPENSSL_ENABLED;
 213  
     }
 214  
 }