MixAuditJsonParser.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 The OWASP Foundation. All Rights Reserved.
 */
package org.owasp.dependencycheck.data.elixir;

import org.owasp.dependencycheck.analyzer.exception.AnalysisException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.annotation.concurrent.NotThreadSafe;

import javax.json.stream.JsonParsingException;
import java.io.Reader;
import java.util.ArrayList;
import java.util.List;
import javax.json.Json;
import javax.json.JsonArray;
import javax.json.JsonException;
import javax.json.JsonObject;
import javax.json.JsonReader;
import javax.json.JsonString;

/**
 * Parses json output from `mix_audit --format json`.
 *
 * @author Christoph Sassenberg
 */
@NotThreadSafe
public class MixAuditJsonParser {

    /**
     * A key in the mix json file.
     */
    static final String PASS_FAIL_KEY = "pass";
    /**
     * A key in the mix json file.
     */
    static final String RESULTS_KEY = "vulnerabilities";
    /**
     * A key in the mix json file.
     */
    static final String ADVISORY_KEY = "advisory";
    /**
     * A key in the mix json file.
     */
    static final String DEPENDENCY_KEY = "dependency";

    /**
     * The JsonReader for parsing JSON.
     */
    private final JsonReader jsonReader;

    /**
     * The List of MixAuditResults found.
     */
    private final List<MixAuditResult> mixAuditResults;

    /**
     * Whether the mix audit passed or failed.
     */
    private boolean mixAuditPass;

    /**
     * The LOGGER
     */
    private static final Logger LOGGER = LoggerFactory.getLogger(MixAuditJsonParser.class);

    /**
     * Creates a MixAuditJsonParser from a Reader.
     *
     * @param reader - the java.io.Reader to read the json character stream from
     */
    public MixAuditJsonParser(Reader reader) {
        LOGGER.debug("Creating a MixAuditJsonParser");
        this.jsonReader = Json.createReader(reader);
        this.mixAuditResults = new ArrayList<>();
        this.mixAuditPass = false;
    }

    /**
     * Process the input stream to create the list of dependencies.
     *
     * @throws AnalysisException thrown when there is an error parsing the
     * results of `mix_audit --format json`
     */
    public void process() throws AnalysisException {
        LOGGER.debug("Beginning mix_audit json output processing");
        try {
            final JsonObject output = jsonReader.readObject();
            if (output.containsKey(PASS_FAIL_KEY)) {
                this.mixAuditPass = output.getBoolean(PASS_FAIL_KEY);
            }

            if (output.containsKey(RESULTS_KEY) && output.isNull(RESULTS_KEY)) {
                LOGGER.debug("Found vulnerabilities");
            }
            final JsonArray results = output.getJsonArray(RESULTS_KEY);
            for (JsonObject result : results.getValuesAs(JsonObject.class)) {
                final JsonObject advisory = result.getJsonObject(ADVISORY_KEY);
                final JsonObject dependency = result.getJsonObject(DEPENDENCY_KEY);
                final ArrayList<String> patchedVersions = new ArrayList<>();

                for (JsonString patchedVersion : advisory.getJsonArray("patched_versions").getValuesAs(JsonString.class)) {
                    patchedVersions.add(patchedVersion.getString());
                }

                final MixAuditResult r = new MixAuditResult(
                        advisory.getString("id"),
                        advisory.getString("cve"),
                        advisory.getString("title"),
                        advisory.getString("description"),
                        advisory.getString("disclosure_date"),
                        advisory.getString("url"),
                        patchedVersions,
                        dependency.getString("lockfile"),
                        dependency.getString("package"),
                        dependency.getString("version")
                );

                this.mixAuditResults.add(r);
            }
        } catch (JsonParsingException jsonpe) {
            throw new AnalysisException("Error parsing stream", jsonpe);
        } catch (JsonException jsone) {
            throw new AnalysisException("Error reading stream", jsone);
        } catch (IllegalStateException ise) {
            throw new AnalysisException("Illegal state while parsing mix_audit output", ise);
        } catch (ClassCastException cce) {
            throw new AnalysisException("JSON not exactly matching output of `mix_audit --format json`", cce);
        }
    }

    /**
     * Gets the list of results.
     *
     * @return the list of results
     */
    public List<MixAuditResult> getResults() {
        return mixAuditResults;
    }
}