@atomist/sdm
Version:
Atomist Software Delivery Machine SDK
160 lines (143 loc) • 5.67 kB
text/typescript
/*
* Copyright © 2020 Atomist, Inc.
*
* 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.
*/
import { ProjectReview, ReviewComment, Severity } from "@atomist/automation-client/lib/operations/review/ReviewResult";
import { isLocalProject } from "@atomist/automation-client/lib/project/local/LocalProject";
import { Project } from "@atomist/automation-client/lib/project/Project";
import { NoParameters } from "@atomist/automation-client/lib/SmartParameters";
import { logger } from "@atomist/automation-client/lib/util/logger";
import { codeLine, italic } from "@atomist/slack-messages";
import * as _ from "lodash";
import { spawnPromise } from "../../../api-helper/misc/child_process";
import { AutoInspectRegistration } from "../../../api/registration/AutoInspectRegistration";
import { CodeInspection, CodeInspectionRegistration } from "../../../api/registration/CodeInspectionRegistration";
import { DefaultNpmAuditOptions, NpmAuditOptions } from "../autofix/npmAuditAutofix";
import { IsNode } from "../pushtest/nodePushTests";
export interface NpmAuditAdvisory {
module_name: string;
vulnerable_versions: string;
severity: "info" | "low" | "moderate" | "high" | "critical";
title: string;
findings: Array<{ version: string; paths: string[] }>;
cves: string[];
url: string;
recommendation: string;
}
export interface NpmAuditResult {
actions: any[];
advisories: { [id: string]: NpmAuditAdvisory };
}
export const npmAuditReviewCategory = "npm audit";
function npmAuditReviewComment(detail: string, rule: string, severity: Severity = "info"): ReviewComment {
return {
category: npmAuditReviewCategory,
detail,
severity,
subcategory: rule,
};
}
export function mapNpmAuditResultsToReviewComments(npmAuditOutput: string): ReviewComment[] {
let results: NpmAuditResult;
try {
results = JSON.parse(npmAuditOutput);
} catch (e) {
logger.error(`Failed to parse npm audit output '${npmAuditOutput}': ${e.message}`);
return [];
}
return _.map(results.advisories, v => {
const rule = `${v.module_name}:${v.vulnerable_versions}`;
let details = `[${v.title}](${v.url})`;
if (!!v.recommendation) {
details = `${details} ${italic(v.recommendation.trim())}`;
}
if (!!v.cves && v.cves.length > 0) {
details = `${details} - ` + v.cves.map(c => `[${c}](https://nvd.nist.gov/vuln/detail/${c})`).join(" ");
}
if (!!v.findings && v.findings.length > 0) {
const findings = v.findings.map(
f =>
`\n - ${codeLine(`${v.module_name}:${f.version}`)}: ${(f.paths || [])
.map(p => `\n - ${codeLine(p)}`)
.join("")}`,
);
details = `${details} ${findings.join("")}`;
}
let severity: Severity;
switch (v.severity) {
case "info":
case "low":
severity = "info";
break;
case "moderate":
severity = "warn";
break;
case "high":
case "critical":
severity = "error";
break;
}
return npmAuditReviewComment(details, rule, severity);
});
}
export function runNpmAuditOnProject(
options: NpmAuditOptions = DefaultNpmAuditOptions,
): CodeInspection<ProjectReview, NoParameters> {
return async (p: Project) => {
const review: ProjectReview = { repoId: p.id, comments: [] };
if (!isLocalProject(p)) {
logger.error(`Project ${p.name} is not a local project`);
return review;
}
const cwd = p.baseDir;
const args = ["audit", "--json"];
if (options.packageLockOnly === true) {
args.push("--package-lock-only");
}
try {
const npmAuditResult = await spawnPromise("npm", args, { cwd });
if (npmAuditResult.stderr) {
logger.debug(`npm audit standard error from ${p.name}: ${npmAuditResult.stderr}`);
}
const comments = mapNpmAuditResultsToReviewComments(npmAuditResult.stdout);
review.comments.push(...comments);
} catch (e) {
logger.error(`Failed to run npm audit: ${e.message}`);
}
return review;
};
}
/**
* Provide a code inspection that runs `npm audit` and returns a
* ProjectReview.
*/
export function npmAuditInspection(
options: NpmAuditOptions = DefaultNpmAuditOptions,
): CodeInspectionRegistration<ProjectReview, NoParameters> {
return {
name: "NpmAuditInspection",
description: "Run npm audit on project",
inspection: runNpmAuditOnProject(options),
intent: "npm audit",
};
}
/**
* Provide an auto inspect registration that runs `npm audit` and
* returns a ProjectReview.
*/
export const NpmAuditAutoInspectRegistration: AutoInspectRegistration<ProjectReview, NoParameters> = {
name: "NpmAuditAutoInspection",
inspection: runNpmAuditOnProject(),
pushTest: IsNode,
};