API/api.medcify.app/node_modules/snyk/dist/cli/895.index.js

1457 lines
258 KiB
JavaScript
Raw Normal View History

2022-09-26 06:11:44 +00:00
"use strict";
exports.id = 895;
exports.ids = [895];
exports.modules = {
/***/ 33111:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.InvalidArgumentError = exports.isIacShareResultsOptions = exports.assertIaCOptionsFlags = exports.UnsupportedEntitlementCommandError = exports.UnsupportedEntitlementFlagError = exports.FlagValueError = exports.FeatureFlagError = exports.FlagError = void 0;
const errors_1 = __webpack_require__(55191);
const args_1 = __webpack_require__(94765);
const error_utils_1 = __webpack_require__(36401);
const types_1 = __webpack_require__(94820);
const keys = [
'org',
'debug',
'insecure',
'detectionDepth',
'severityThreshold',
'rules',
'json',
'sarif',
'json-file-output',
'sarif-file-output',
'v',
'version',
'h',
'help',
'q',
'quiet',
'scan',
'report',
// Tags and attributes
'tags',
'project-tags',
'project-environment',
'project-lifecycle',
'project-business-criticality',
'target-reference',
'var-file',
// PolicyOptions
'ignore-policy',
'policy-path',
// Report options
'remote-repo-url',
'target-name',
];
const allowed = new Set(keys);
function camelcaseToDash(key) {
return key.replace(/[A-Z]/g, (m) => '-' + m.toLowerCase());
}
function getFlagName(key) {
const dashes = key.length === 1 ? '-' : '--';
const flag = camelcaseToDash(key);
return `${dashes}${flag}`;
}
class FlagError extends errors_1.CustomError {
constructor(key) {
const flag = getFlagName(key);
const msg = `Unsupported flag "${flag}" provided. Run snyk iac test --help for supported flags`;
super(msg);
this.code = types_1.IaCErrorCodes.FlagError;
this.strCode = error_utils_1.getErrorStringCode(this.code);
this.userMessage = msg;
}
}
exports.FlagError = FlagError;
class FeatureFlagError extends errors_1.CustomError {
constructor(key, featureFlag, hasSnykPreview) {
const flag = getFlagName(key);
let msg;
if (hasSnykPreview) {
msg = `Flag "${flag}" is only supported if feature flag '${featureFlag}' is enabled. The feature flag can be enabled via Snyk Preview if you are on the Enterprise Plan`;
}
else {
msg = `Flag "${flag}" is only supported if feature flag "${featureFlag}" is enabled. To enable it, please contact Snyk support.`;
}
super(msg);
this.code = types_1.IaCErrorCodes.FeatureFlagError;
this.strCode = error_utils_1.getErrorStringCode(this.code);
this.userMessage = msg;
}
}
exports.FeatureFlagError = FeatureFlagError;
class FlagValueError extends errors_1.CustomError {
constructor(key, value) {
const flag = getFlagName(key);
const msg = `Unsupported value "${value}" provided to flag "${flag}".\nSupported values are: ${SUPPORTED_TF_PLAN_SCAN_MODES.join(', ')}`;
super(msg);
this.code = types_1.IaCErrorCodes.FlagValueError;
this.strCode = error_utils_1.getErrorStringCode(this.code);
this.userMessage = msg;
}
}
exports.FlagValueError = FlagValueError;
class UnsupportedEntitlementFlagError extends errors_1.CustomError {
constructor(key, entitlementName) {
const flag = getFlagName(key);
super(`Unsupported flag: ${flag} - Missing the ${entitlementName} entitlement`);
this.code = types_1.IaCErrorCodes.UnsupportedEntitlementFlagError;
this.strCode = error_utils_1.getErrorStringCode(this.code);
this.userMessage = `Flag "${flag}" is currently not supported for this org. To enable it, please contact snyk support.`;
}
}
exports.UnsupportedEntitlementFlagError = UnsupportedEntitlementFlagError;
class UnsupportedEntitlementCommandError extends errors_1.CustomError {
constructor(key, entitlementName) {
super(`Unsupported command: ${key} - Missing the ${entitlementName} entitlement`);
this.code = types_1.IaCErrorCodes.UnsupportedEntitlementFlagError;
this.strCode = error_utils_1.getErrorStringCode(this.code);
this.userMessage = `Command "${key}" is currently not supported for this org. To enable it, please contact snyk support.`;
}
}
exports.UnsupportedEntitlementCommandError = UnsupportedEntitlementCommandError;
/**
* Validates the command line flags passed to the snyk iac test
* command. The current argument parsing is very permissive and
* allows unknown flags to be provided without validation.
*
* For snyk iac we need to explicitly validate the flags to avoid
* misconfigurations and typos. For example, if the --experimental
* flag were to be misspelled we would end up sending the client
* data to our backend rather than running it locally as intended.
* @param argv command line args passed to the process
*/
function assertIaCOptionsFlags(argv) {
// We process the process.argv so we don't get default values.
const parsed = args_1.args(argv);
for (const key of Object.keys(parsed.options)) {
// The _ property is a special case that contains non
// flag strings passed to the command line (usually files)
// and `iac` is the command provided.
if (key !== '_' && key !== 'iac' && !allowed.has(key)) {
throw new FlagError(key);
}
}
if (parsed.options.scan) {
assertTerraformPlanModes(parsed.options.scan);
}
}
exports.assertIaCOptionsFlags = assertIaCOptionsFlags;
const SUPPORTED_TF_PLAN_SCAN_MODES = [
types_1.TerraformPlanScanMode.DeltaScan,
types_1.TerraformPlanScanMode.FullScan,
];
function assertTerraformPlanModes(scanModeArgValue) {
if (!SUPPORTED_TF_PLAN_SCAN_MODES.includes(scanModeArgValue)) {
throw new FlagValueError('scan', scanModeArgValue);
}
}
function isIacShareResultsOptions(options) {
return options.iac && options.report;
}
exports.isIacShareResultsOptions = isIacShareResultsOptions;
class InvalidArgumentError extends errors_1.CustomError {
constructor(key) {
const flag = getFlagName(key);
const msg = `Invalid argument provided to flag "${flag}". Value must be a string`;
super(msg);
this.code = types_1.IaCErrorCodes.InvalidArgumentError;
this.strCode = error_utils_1.getErrorStringCode(this.code);
this.userMessage = msg;
}
}
exports.InvalidArgumentError = InvalidArgumentError;
/***/ }),
/***/ 36401:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.getErrorStringCode = void 0;
const types_1 = __webpack_require__(94820);
function getErrorStringCode(code) {
const errorName = types_1.IaCErrorCodes[code];
if (!errorName) {
return 'INVALID_IAC_ERROR';
}
let result = errorName.replace(/([A-Z])/g, '_$1');
if (result.charAt(0) === '_') {
result = result.substring(1);
}
return result.toUpperCase();
}
exports.getErrorStringCode = getErrorStringCode;
/***/ }),
/***/ 11693:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.FailedToGetIacOrgSettingsError = exports.getIacOrgSettings = void 0;
const types_1 = __webpack_require__(94820);
const config_1 = __webpack_require__(25425);
const is_ci_1 = __webpack_require__(10090);
const api_token_1 = __webpack_require__(95181);
const request_1 = __webpack_require__(52050);
const errors_1 = __webpack_require__(55191);
const error_utils_1 = __webpack_require__(36401);
function getIacOrgSettings(publicOrgId) {
const payload = {
method: 'get',
url: config_1.default.API + '/iac-org-settings',
json: true,
qs: { org: publicOrgId },
headers: {
'x-is-ci': is_ci_1.isCI(),
authorization: api_token_1.getAuthHeader(),
},
};
return new Promise((resolve, reject) => {
request_1.makeRequest(payload, (error, res) => {
if (error) {
return reject(error);
}
if (res.statusCode < 200 || res.statusCode > 299) {
return reject(new FailedToGetIacOrgSettingsError());
}
resolve(res.body);
});
});
}
exports.getIacOrgSettings = getIacOrgSettings;
class FailedToGetIacOrgSettingsError extends errors_1.CustomError {
constructor(message) {
super(message || 'Failed to fetch IaC organization settings');
this.code = types_1.IaCErrorCodes.FailedToGetIacOrgSettingsError;
this.strCode = error_utils_1.getErrorStringCode(this.code);
this.userMessage =
'We failed to fetch your organization settings, including custom severity overrides for infrastructure-as-code policies. Please run the command again with the `-d` flag and contact support@snyk.io with the contents of the output.';
}
}
exports.FailedToGetIacOrgSettingsError = FailedToGetIacOrgSettingsError;
/***/ }),
/***/ 47658:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.DescribeExclusiveArgumentError = void 0;
const custom_error_1 = __webpack_require__(17188);
const drift_1 = __webpack_require__(26445);
class DescribeExclusiveArgumentError extends custom_error_1.CustomError {
constructor() {
super('Please use only one of these arguments: ' +
drift_1.DescribeExclusiveArgs.join(', '));
}
}
exports.DescribeExclusiveArgumentError = DescribeExclusiveArgumentError;
/***/ }),
/***/ 37541:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.DescribeRequiredArgumentError = void 0;
const custom_error_1 = __webpack_require__(17188);
const chalk_1 = __webpack_require__(32589);
class DescribeRequiredArgumentError extends custom_error_1.CustomError {
constructor() {
super(chalk_1.default.red('Describe command require one of these arguments:\n' +
' --only-unmanaged: Report resources not under Terraform control\n' +
' --only-managed: Report resources from Terraform states that changed (may take a while)\n' +
' --all: Scan for managed and unmanaged resources (may take a while)'));
}
}
exports.DescribeRequiredArgumentError = DescribeRequiredArgumentError;
/***/ }),
/***/ 24030:
/***/ ((__unused_webpack_module, exports) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.default = `
/***/ }),
/***/ 19679:
/***/ ((__unused_webpack_module, exports) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.default = `<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="1560pt" height="2448pt" viewBox="0 0 1560 2448"><g enable-background="new"><g><g id="Layer-1" data-name="AAPL:Style"><g id="Layer-2" data-name="AAPL:StyleContent"><mask id="ma0"><g transform="matrix(1,0,0,.99999997,0,-4)"><use xlink:href="#im1" x="0" y="0" width="1560" height="2456"/></g></mask><symbol id="im1" viewBox="0 0 1560 2456"><image width="1560" height="2456" xlink:href="
/***/ }),
/***/ 26445:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.processHTMLOutput = exports.processAnalysis = exports.updateExcludeInPolicy = exports.driftignoreFromPolicy = exports.parseDriftAnalysisResults = exports.validateArgs = exports.DescribeRequiredArgs = exports.DescribeExclusiveArgs = void 0;
const fs = __webpack_require__(35747);
const describe_exclusive_argument_error_1 = __webpack_require__(47658);
const describe_required_argument_error_1 = __webpack_require__(37541);
const snyk_logo_1 = __webpack_require__(19679);
const snyk_favicon_1 = __webpack_require__(24030);
const output_1 = __webpack_require__(55659);
const driftctl_1 = __webpack_require__(3659);
exports.DescribeExclusiveArgs = [
'all',
'only-managed',
'drift',
'only-unmanaged',
];
exports.DescribeRequiredArgs = [
'all',
'only-managed',
'drift',
'only-unmanaged',
];
exports.validateArgs = (options) => {
if (options.kind === 'describe') {
return validateDescribeArgs(options);
}
};
const validateDescribeArgs = (options) => {
// Check that there is no more than one of the exclusive arguments
let count = 0;
for (const describeExclusiveArg of exports.DescribeExclusiveArgs) {
if (options[describeExclusiveArg]) {
count++;
}
}
if (count > 1) {
throw new describe_exclusive_argument_error_1.DescribeExclusiveArgumentError();
}
// Check we have one of the required arguments
count = 0;
for (const describeRequiredArgs of exports.DescribeRequiredArgs) {
if (options[describeRequiredArgs]) {
count++;
}
}
if (count === 0) {
throw new describe_required_argument_error_1.DescribeRequiredArgumentError();
}
};
exports.parseDriftAnalysisResults = (input) => {
return JSON.parse(input);
};
function driftignoreFromPolicy(policy) {
const excludeSection = 'iac-drift';
if (!policy || !policy.exclude || !(excludeSection in policy.exclude)) {
return [];
}
return policy.exclude[excludeSection];
}
exports.driftignoreFromPolicy = driftignoreFromPolicy;
exports.updateExcludeInPolicy = (policy, analysis, options) => {
var _a, _b, _c;
const excludedResources = driftignoreFromPolicy(policy);
const addResource = (res) => excludedResources.push(`${res.type}.${res.id}`);
if (!options['exclude-changed'] && analysis.summary.total_changed > 0) {
(_a = analysis.differences) === null || _a === void 0 ? void 0 : _a.forEach((change) => addResource(change.res));
}
if (!options['exclude-missing'] && analysis.summary.total_missing > 0) {
(_b = analysis.missing) === null || _b === void 0 ? void 0 : _b.forEach((res) => addResource(res));
}
if (!options['exclude-unmanaged'] && analysis.summary.total_unmanaged > 0) {
(_c = analysis.unmanaged) === null || _c === void 0 ? void 0 : _c.forEach((res) => addResource(res));
}
if (!policy.exclude) {
policy.exclude = {};
}
policy.exclude['iac-drift'] = excludedResources;
};
async function processAnalysis(options, describe) {
if (options.html || options['html-file-output']) {
// we use fmt for html output
const fmtResult = await driftctl_1.runDriftCTL({
options: { ...options, kind: 'fmt' },
input: describe.stdout,
});
const output = processHTMLOutput(options, fmtResult.stdout);
if (options.html) {
// html on stdout
return output;
}
// should return an empty string if we use the html-file-output flag
return '';
}
if (options.json) {
// json on stdout
return describe.stdout;
}
const analysis = exports.parseDriftAnalysisResults(describe.stdout);
return output_1.getHumanReadableAnalysis(options, analysis);
}
exports.processAnalysis = processAnalysis;
function processHTMLOutput(options, stdout) {
if (options.html) {
stdout = rebrandHTMLOutput(stdout);
}
if (options['html-file-output']) {
const data = fs.readFileSync(options['html-file-output'], {
encoding: 'utf8',
});
fs.writeFileSync(options['html-file-output'], rebrandHTMLOutput(data));
}
return stdout;
}
exports.processHTMLOutput = processHTMLOutput;
function rebrandHTMLOutput(data) {
// Replace favicon
const faviconReplaceRegex = new RegExp('(<link rel="shortcut icon")(.*)(\\/>)', 'g');
data = data.replace(faviconReplaceRegex, `<link rel="shortcut icon" type="image/x-icon" href="${snyk_favicon_1.default}" />`);
// Replace HTML title
const titleReplaceRegex = new RegExp('(<title>)(.*)(<\\/title>)', 'g');
data = data.replace(titleReplaceRegex, `<title>Snyk IaC drift report</title>`);
// Replace header brand logo
const logoReplaceRegex = new RegExp('(<div id="brand_logo">)((.|\\r|\\n)*?)(<\\/div>)', 'g');
data = data.replace(logoReplaceRegex, `<div id="brand_logo">${snyk_logo_1.default}</div>`);
return data;
}
/***/ }),
/***/ 3659:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
var _a;
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.runDriftCTL = exports.translateExitCode = exports.generateArgs = exports.driftctlVersion = exports.DCTL_EXIT_CODES = void 0;
const config_1 = __webpack_require__(25425);
const exit_codes_1 = __webpack_require__(80079);
const env_paths_1 = __webpack_require__(21766);
const metrics_1 = __webpack_require__(32971);
const analytics = __webpack_require__(82744);
const spinner_1 = __webpack_require__(86766);
const service_mappings_1 = __webpack_require__(16228);
const drift_1 = __webpack_require__(26445);
const debugLib = __webpack_require__(15158);
const request_1 = __webpack_require__(52050);
const path = __webpack_require__(85622);
const child_process = __webpack_require__(63129);
const fs = __webpack_require__(35747);
const os = __webpack_require__(12087);
const crypto = __webpack_require__(76417);
const file_utils_1 = __webpack_require__(52761);
const debug = debugLib('driftctl');
const cachePath = (_a = config_1.default.CACHE_PATH) !== null && _a !== void 0 ? _a : env_paths_1.default('snyk').cache;
exports.DCTL_EXIT_CODES = {
EXIT_IN_SYNC: 0,
EXIT_NOT_IN_SYNC: 1,
EXIT_ERROR: 2,
};
exports.driftctlVersion = 'v0.35.2';
const driftctlChecksums = {
driftctl_darwin_amd64: '1e21863bb99d104da8a1e33999563cc172ca51bb5b6ac7d3a4bd9678e0285067',
'driftctl_windows_386.exe': '20dc49a4faebfdbdf9f9198499ba2426ea6cda0666e82d81cd4e08a41516d5e0',
driftctl_darwin_arm64: '37b7a4c72f4db62b056f1b534eb93fbb8dd32c303719ed4af87d9bd4d903fcf6',
driftctl_linux_arm64: '3e6dabf013e097be1aac0e6e0f4ebcc3ada85a1906c6841d57fbe96c9ee9857e',
'driftctl_windows_arm.exe': '480c330fefdc6e1d58de943817556a1cd183d32d58b6cb20c8127cd3b8753dc4',
driftctl_linux_amd64: '80b7b99c343e1502f54321987c9f00fa3c381fbea819b1e440a0377b18706fb1',
'driftctl_windows_amd64.exe': 'bbc71047bd75e1bcd80752b7c03c27c0d8d7d93bec72a70eb8bc8220687805de',
'driftctl_windows_arm64.exe': 'be1a5564ec3351503caa16121d192ad8d8e8f287a5324939214b20177c7363e4',
driftctl_linux_arm: 'd04c911bdb86092743077bfbb025bfb8391978236bb381122594aeaa77f9a68f',
driftctl_linux_386: 'e720c2f3c25594c7b83ffb2123c461283589b6ee73b9a59c1c4f48bb2bac2775',
};
const dctlBaseUrl = 'https://static.snyk.io/cli/driftctl/';
const driftctlPath = path.join(cachePath, 'driftctl_' + exports.driftctlVersion);
const driftctlDefaultOptions = ['--no-version-check'];
let isBinaryDownloaded = false;
exports.generateArgs = async (options, driftIgnore) => {
if (options.kind === 'describe') {
return await generateScanFlags(options, driftIgnore);
}
if (options.kind === 'fmt') {
return generateFmtFlags(options);
}
throw 'Unsupported command';
};
const generateFmtFlags = (options) => {
const args = ['fmt', ...driftctlDefaultOptions];
if (options.json) {
args.push('--output');
args.push('json://stdout');
}
if (options.html) {
args.push('--output');
args.push('html://stdout');
}
if (options['html-file-output']) {
args.push('--output');
args.push('html://' + options['html-file-output']);
}
return args;
};
const generateScanFlags = async (options, driftIgnore) => {
const args = ['scan', ...driftctlDefaultOptions];
if (options.quiet) {
args.push('--quiet');
}
if (options.filter) {
args.push('--filter');
args.push(options.filter);
}
args.push('--output');
args.push('json://stdout');
if (options['fetch-tfstate-headers']) {
args.push('--headers');
args.push(options['fetch-tfstate-headers']);
}
if (options['tfc-token']) {
args.push('--tfc-token');
args.push(options['tfc-token']);
}
if (options['tfc-endpoint']) {
args.push('--tfc-endpoint');
args.push(options['tfc-endpoint']);
}
if (options['tf-provider-version']) {
args.push('--tf-provider-version');
args.push(options['tf-provider-version']);
}
if (options.strict) {
args.push('--strict');
}
if (options.deep || options.all) {
args.push('--deep');
}
if (options['only-managed'] || options.drift) {
args.push('--only-managed');
}
if (options['only-unmanaged']) {
args.push('--only-unmanaged');
}
if (options.driftignore) {
args.push('--driftignore');
args.push(options.driftignore);
}
if (options['tf-lockfile']) {
args.push('--tf-lockfile');
args.push(options['tf-lockfile']);
}
if (driftIgnore && driftIgnore.length > 0) {
args.push('--ignore');
args.push(driftIgnore.join(','));
}
let configDir = cachePath;
await file_utils_1.createDirIfNotExists(cachePath);
if (options['config-dir']) {
configDir = options['config-dir'];
}
args.push('--config-dir');
args.push(configDir);
if (options.from) {
const froms = options.from.split(',');
for (const f of froms) {
args.push('--from');
args.push(f);
}
}
let to = 'aws+tf';
if (options.to) {
to = options.to;
}
args.push('--to');
args.push(to);
if (options.service) {
const services = options.service.split(',');
service_mappings_1.verifyServiceMappingExists(services);
args.push('--ignore');
args.push(service_mappings_1.createIgnorePattern(services));
}
debug(args);
return args;
};
function translateExitCode(exitCode) {
switch (exitCode) {
case exports.DCTL_EXIT_CODES.EXIT_IN_SYNC:
return 0;
case exports.DCTL_EXIT_CODES.EXIT_NOT_IN_SYNC:
return exit_codes_1.EXIT_CODES.VULNS_FOUND;
case exports.DCTL_EXIT_CODES.EXIT_ERROR:
return exit_codes_1.EXIT_CODES.ERROR;
default:
debug('driftctl returned %d', exitCode);
return exit_codes_1.EXIT_CODES.ERROR;
}
}
exports.translateExitCode = translateExitCode;
exports.runDriftCTL = async ({ options, driftIgnore, input, stdio, }) => {
const path = await findOrDownload();
await drift_1.validateArgs(options);
const args = await exports.generateArgs(options, driftIgnore);
if (!stdio) {
stdio = ['pipe', 'pipe', 'inherit'];
}
debug('running driftctl %s ', args.join(' '));
const p = child_process.spawn(path, args, {
stdio,
env: { ...process.env, DCTL_IS_SNYK: 'true' },
});
let stdout = '';
return new Promise((resolve, reject) => {
var _a, _b, _c;
if (input) {
(_a = p.stdin) === null || _a === void 0 ? void 0 : _a.write(input);
(_b = p.stdin) === null || _b === void 0 ? void 0 : _b.end();
}
p.on('error', (error) => {
reject(error);
});
(_c = p.stdout) === null || _c === void 0 ? void 0 : _c.on('data', (data) => {
stdout += data;
});
p.on('exit', (code) => {
resolve({ code: translateExitCode(code), stdout });
});
});
};
async function findOrDownload() {
let dctl = await findDriftCtl();
if (isBinaryDownloaded) {
return dctl;
}
let downloadDuration = 0;
let binaryExist = true;
if (dctl === '') {
binaryExist = false;
try {
await file_utils_1.createDirIfNotExists(cachePath);
dctl = driftctlPath;
const duration = new metrics_1.TimerMetricInstance('driftctl_download');
duration.start();
await download(driftctlUrl(), dctl);
duration.stop();
downloadDuration = Math.round(duration.getValue() / 1000);
}
catch (err) {
return Promise.reject(err);
}
}
analytics.add('iac-drift-binary-already-exist', binaryExist);
analytics.add('iac-drift-binary-download-duration-seconds', downloadDuration);
isBinaryDownloaded = true;
return dctl;
}
async function findDriftCtl() {
// lookup in custom path contained in env var DRIFTCTL_PATH
let dctlPath = config_1.default.DRIFTCTL_PATH;
if (dctlPath != null) {
const exists = await file_utils_1.isExe(dctlPath);
if (exists) {
debug('Found driftctl in $DRIFTCTL_PATH: %s', dctlPath);
return dctlPath;
}
}
// lookup in app cache
dctlPath = driftctlPath;
const exists = await file_utils_1.isExe(dctlPath);
if (exists) {
debug('Found driftctl in cache: %s', dctlPath);
return dctlPath;
}
debug('driftctl not found');
return '';
}
async function download(url, destination) {
debug('downloading driftctl into %s', destination);
const payload = {
method: 'GET',
url: url,
output: destination,
follow: 3,
};
await spinner_1.spinner('Downloading...');
return new Promise((resolve, reject) => {
request_1.makeRequest(payload, function (err, res, body) {
try {
if (err) {
reject(new Error('Could not download driftctl from ' + url + ': ' + err));
return;
}
if (res.statusCode !== 200) {
reject(new Error('Could not download driftctl from ' + url + ': ' + res.statusCode));
return;
}
validateChecksum(body);
fs.writeFileSync(destination, body);
debug('File saved: ' + destination);
fs.chmodSync(destination, 0o744);
resolve(true);
}
finally {
spinner_1.spinner.clearAll();
}
});
});
}
function validateChecksum(body) {
// only validate if we downloaded the official driftctl binary
if (config_1.default.DRIFTCTL_URL || config_1.default.DRIFTCTL_PATH) {
return;
}
const computedHash = crypto
.createHash('sha256')
.update(body)
.digest('hex');
const givenHash = driftctlChecksums[driftctlFileName()];
if (computedHash != givenHash) {
throw new Error('Downloaded file has inconsistent checksum...');
}
}
function driftctlFileName() {
let platform = 'linux';
switch (os.platform()) {
case 'darwin':
platform = 'darwin';
break;
case 'win32':
platform = 'windows';
break;
}
let arch = 'amd64';
switch (os.arch()) {
case 'ia32':
case 'x32':
arch = '386';
break;
case 'arm':
arch = 'arm';
break;
case 'arm64':
arch = 'arm64';
break;
}
let ext = '';
switch (os.platform()) {
case 'win32':
ext = '.exe';
break;
}
return `driftctl_${platform}_${arch}${ext}`;
}
function driftctlUrl() {
if (config_1.default.DRIFTCTL_URL) {
return config_1.default.DRIFTCTL_URL;
}
return `${dctlBaseUrl}/${exports.driftctlVersion}/${driftctlFileName()}`;
}
/***/ }),
/***/ 55659:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.getHumanReadableAnalysis = void 0;
const service_mappings_1 = __webpack_require__(16228);
const chalk_1 = __webpack_require__(32589);
const common_1 = __webpack_require__(47724);
const jsondiffpatch_1 = __webpack_require__(79732);
function getHumanReadableAnalysis(option, analysis) {
let output = getHumanReadableHeader();
if (!option['only-unmanaged']) {
output += getHumanReadableManaged(analysis);
}
if (!option['only-managed'] && !option.drift) {
output += getHumanReadableUnmanaged(analysis);
}
output += getHumanReadableSummary(analysis);
return output;
}
exports.getHumanReadableAnalysis = getHumanReadableAnalysis;
function changeAsString(obj) {
if (typeof obj === 'string') {
return obj;
}
return JSON.stringify(obj);
}
function isJsonDiff(driftChange) {
if (!(typeof driftChange.from === 'string')) {
return false;
}
try {
JSON.parse(driftChange.from);
}
catch (e) {
return false;
}
return true;
}
function getNonJsonDiff(driftChange) {
let output = '';
switch (driftChange.type) {
case 'create':
output += chalk_1.default.green('+') + ' ' + driftChange.path.join('.') + ': ';
output +=
chalk_1.default.bold(changeAsString(driftChange.from)) +
' => ' +
chalk_1.default.green(changeAsString(driftChange.to));
break;
case 'update':
output += chalk_1.default.yellow('~') + ' ' + driftChange.path.join('.') + ': ';
output +=
chalk_1.default.bold(changeAsString(driftChange.from)) +
' => ' +
chalk_1.default.yellow(changeAsString(driftChange.to));
break;
case 'delete':
output += chalk_1.default.red('-') + ' ' + driftChange.path.join('.') + ': ';
output += chalk_1.default.red(changeAsString(driftChange.from));
break;
default:
output += driftChange.path.join('.') + ': ';
output +=
chalk_1.default.bold(changeAsString(driftChange.from)) +
' => ' +
chalk_1.default.bold(changeAsString(driftChange.to));
break;
}
output += '\n';
return output;
}
function getJsonDiff(driftChange) {
let output = '';
let from = null;
if (driftChange.from) {
from = JSON.parse(driftChange.from);
}
let to = null;
if (driftChange.to) {
to = JSON.parse(driftChange.to);
}
let diffStr = '';
const diffPatch = jsondiffpatch_1.create().diff(from, to);
if (diffPatch) {
diffStr = jsondiffpatch_1.console.format(diffPatch, from);
}
switch (driftChange.type) {
case 'create':
output += chalk_1.default.green('+') + ' ' + driftChange.path.join('.') + ':\n';
break;
case 'update':
output += chalk_1.default.yellow('~') + ' ' + driftChange.path.join('.') + ':\n';
break;
case 'delete':
output += chalk_1.default.red('-') + ' ' + driftChange.path.join('.') + ':\n';
break;
default:
output += driftChange.path.join('.') + ':\n';
break;
}
for (const elem of diffStr.split('\n')) {
output += addLine(common_1.leftPad(elem, 4));
}
return output;
}
function getHumanReadableDrift(analysis) {
var _a;
let output = '';
if (!analysis.differences || analysis.differences.length <= 0) {
return '';
}
const diffByStates = new Map();
for (const difference of analysis.differences) {
let statefile = 'Generated';
if (difference.res.source) {
statefile = difference.res.source.source;
}
if (!diffByStates.has(statefile)) {
diffByStates.set(statefile, {
diffByType: new Map(),
count: 0,
});
}
const hrDiffs = mustGet(diffByStates, statefile);
const type = difference.res.type;
if (!hrDiffs.diffByType.has(type)) {
hrDiffs.diffByType.set(type, []);
}
(_a = hrDiffs.diffByType.get(type)) === null || _a === void 0 ? void 0 : _a.push(difference);
hrDiffs.count++;
}
output += addLine(chalk_1.default.bold('Changed resources: ' + analysis.differences.length));
output += '\n';
for (const state of [...diffByStates.keys()].sort()) {
const hrDiffs = mustGet(diffByStates, state);
output += addLine(chalk_1.default.blue('State: ' +
chalk_1.default.bold(state) +
' [ Changed Resources: ' +
chalk_1.default.bold(hrDiffs.count.toString()) +
' ]'));
output += '\n';
for (const type of [...hrDiffs.diffByType.keys()].sort()) {
output += addLine(common_1.leftPad('Resource Type: ' + type, 2));
const diffs = mustGet(hrDiffs.diffByType, type);
for (const diff of diffs) {
output += common_1.leftPad('ID: ' + chalk_1.default.bold(diff.res.id), 4);
if (diff.res.human_readable_attributes &&
diff.res.human_readable_attributes.size > 0) {
for (const humanReadableAttribute of [
...diff.res.human_readable_attributes.keys(),
].sort()) {
output +=
' ' +
humanReadableAttribute +
': ' +
diff.res.human_readable_attributes.get(humanReadableAttribute);
}
}
output += '\n';
for (const driftChange of diff.changelog) {
output += common_1.leftPad('');
if (isJsonDiff(driftChange)) {
output += getJsonDiff(driftChange);
}
else {
output += getNonJsonDiff(driftChange);
}
}
output += '\n';
}
}
}
return output;
}
function getHumanReadableMissing(analysis) {
var _a;
let output = '';
if (!analysis.missing || analysis.missing.length <= 0) {
return '';
}
const missingByStates = new Map();
for (const missing of analysis.missing) {
let statefile = 'Generated';
if (missing.source) {
statefile = missing.source.source;
}
if (!missingByStates.has(statefile)) {
missingByStates.set(statefile, {
missingByType: new Map(),
count: 0,
});
}
const hrMissing = mustGet(missingByStates, statefile);
const type = missing.type;
if (!hrMissing.missingByType.has(type)) {
hrMissing.missingByType.set(type, []);
}
(_a = hrMissing.missingByType.get(type)) === null || _a === void 0 ? void 0 : _a.push(missing);
hrMissing.count++;
}
output += addLine(chalk_1.default.bold('Missing resources: ' + analysis.missing.length));
output += '\n';
for (const state of [...missingByStates.keys()].sort()) {
const hrMissing = mustGet(missingByStates, state);
output += addLine(chalk_1.default.blue('State: ' +
chalk_1.default.bold(state) +
' [ Missing Resources: ' +
chalk_1.default.bold(hrMissing.count.toString()) +
' ]'));
output += '\n';
for (const type of [...hrMissing.missingByType.keys()].sort()) {
output += addLine(common_1.leftPad('Resource Type: ' + type, 2));
const driftResources = mustGet(hrMissing.missingByType, type);
output += getHumanReadableResourceList(driftResources) + '\n';
}
}
return output;
}
function getHumanReadableManaged(analysis) {
let output = '';
if (analysis.differences && analysis.differences.length > 0) {
output += getHumanReadableDrift(analysis);
}
if (analysis.missing && analysis.missing.length > 0) {
output += getHumanReadableMissing(analysis);
}
return output;
}
function getHumanReadableResourceList(driftResources) {
let output = '';
for (const res of driftResources) {
output += common_1.leftPad('ID: ' + chalk_1.default.bold(res.id), 4);
if (res.human_readable_attributes &&
res.human_readable_attributes.size > 0) {
for (const humanReadableAttribute of [
...res.human_readable_attributes.keys(),
].sort()) {
output +=
' ' +
humanReadableAttribute +
': ' +
res.human_readable_attributes.get(humanReadableAttribute);
}
}
output += '\n';
}
return output;
}
function getHumanReadableUnmanaged(analysis) {
var _a;
let output = '';
if (!analysis.unmanaged || analysis.unmanaged.length <= 0) {
return '';
}
const unmanagedByServices = new Map();
for (const unmanaged of analysis.unmanaged) {
const service = service_mappings_1.findServiceMappingForType(unmanaged.type);
if (!unmanagedByServices.has(service)) {
unmanagedByServices.set(service, {
unmanagedByType: new Map(),
count: 0,
});
}
const hrUnmanaged = mustGet(unmanagedByServices, service);
const type = unmanaged.type;
if (!hrUnmanaged.unmanagedByType.has(type)) {
hrUnmanaged.unmanagedByType.set(type, []);
}
(_a = hrUnmanaged.unmanagedByType.get(type)) === null || _a === void 0 ? void 0 : _a.push(unmanaged);
hrUnmanaged.count++;
}
output += addLine(chalk_1.default.bold('Unmanaged resources: ' + analysis.unmanaged.length));
output += '\n';
for (let service of [...unmanagedByServices.keys()].sort()) {
const hrUnmanaged = mustGet(unmanagedByServices, service);
if (service === '') {
service = 'Unidentified';
}
output += addLine(chalk_1.default.blue('Service: ' +
chalk_1.default.bold(service) +
' [ Unmanaged Resources: ' +
chalk_1.default.bold(hrUnmanaged.count.toString()) +
' ]'));
output += '\n';
for (const type of [...hrUnmanaged.unmanagedByType.keys()].sort()) {
output += addLine(common_1.leftPad('Resource Type: ' + type, 2));
const driftResources = mustGet(hrUnmanaged.unmanagedByType, type);
output += getHumanReadableResourceList(driftResources) + '\n';
}
}
return output;
}
function getHumanReadableHeader() {
// TODO: driftctl to return number of states and supported resources?
let output = addLine(chalk_1.default.bold('Snyk Scanning Infrastructure As Code Discrepancies...'));
output += '\n';
output += addLine(common_1.leftPad('Info: Resources under IaC, but different to terraform states.', 2));
output += addLine(common_1.leftPad('Resolve: Reapply IaC resources or update into terraform.', 2));
output += '\n';
return output;
}
function getHumanReadableSummary(analysis) {
let output = addLine(chalk_1.default.bold('Test Summary'));
output += '\n';
// TODO: driftctl to return number of states
if (analysis.managed) {
output += addLine(common_1.leftPad('Managed Resources: ' + chalk_1.default.bold(analysis.managed.length.toString()), 2));
}
if (analysis.differences) {
output += addLine(common_1.leftPad('Changed Resources: ' +
chalk_1.default.bold(analysis.differences.length.toString()), 2));
}
if (analysis.missing) {
output += addLine(common_1.leftPad('Missing Resources: ' + chalk_1.default.bold(analysis.missing.length.toString()), 2));
}
if (analysis.unmanaged) {
output += addLine(common_1.leftPad('Unmanaged Resources: ' +
chalk_1.default.bold(analysis.unmanaged.length.toString()), 2));
}
output += '\n';
output += addLine(common_1.leftPad('IaC Coverage: ' + chalk_1.default.bold(analysis.coverage.toString() + '%'), 2));
output += addLine(common_1.leftPad('Info: To reach full coverage, remove resources or move it to Terraform.', 2));
output += '\n';
output += addLine(common_1.leftPad('Tip: Run --help to find out about commands and flags.', 2));
output += addLine(common_1.leftPad('Scanned with ' +
analysis.provider_name +
' provider version ' +
analysis.provider_version +
'. Use --tf-provider-version to update.', 6));
return output;
}
function addLine(line) {
return line + '\n';
}
// Used when we are sure the key exists because we just set it but typescript linter does not see that...
function mustGet(map, key) {
const value = map.get(key);
if (!value) {
throw new Error('Key does not exists');
}
return value;
}
/***/ }),
/***/ 52761:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.saveFile = exports.isArchive = exports.isFile = exports.createDirIfNotExists = exports.isExists = exports.isExe = void 0;
const tar = __webpack_require__(97998);
const fs_1 = __webpack_require__(35747);
const util_1 = __webpack_require__(31669);
async function isExe(path) {
try {
await fs_1.promises.access(path, fs_1.constants.X_OK);
return true;
}
catch (err) {
return false;
}
}
exports.isExe = isExe;
async function isExists(path) {
try {
await fs_1.promises.stat(path);
return true;
}
catch (err) {
return false;
}
}
exports.isExists = isExists;
async function createDirIfNotExists(path) {
const isDirExists = await isExists(path);
if (!isDirExists) {
fs_1.promises.mkdir(path, { recursive: true });
}
}
exports.createDirIfNotExists = createDirIfNotExists;
async function isFile(path) {
try {
return (await fs_1.promises.stat(path)).isFile();
}
catch (err) {
return false;
}
}
exports.isFile = isFile;
async function isArchive(path) {
try {
const tarList = util_1.promisify(tar.list);
await tarList({ file: path, strict: true });
return true;
}
catch (e) {
return false;
}
}
exports.isArchive = isArchive;
async function saveFile(dataBuffer, savePath) {
await fs_1.promises.writeFile(savePath, dataBuffer);
await fs_1.promises.chmod(savePath, 0o744);
}
exports.saveFile = saveFile;
/***/ }),
/***/ 16228:
/***/ ((__unused_webpack_module, exports, __webpack_require__) => {
Object.defineProperty(exports, "__esModule", ({ value: true }));
exports.InvalidServiceError = exports.createIgnorePatternWithMap = exports.createIgnorePattern = exports.findServiceMappingForType = exports.verifyServiceMappingExists = exports.services2resources = void 0;
const errors_1 = __webpack_require__(55191);
const types_1 = __webpack_require__(94820);
const error_utils_1 = __webpack_require__(36401);
exports.services2resources = new Map([
// Amazon
[
'aws_s3',
[
'aws_s3_bucket',
'aws_s3_bucket_analytics_configuration',
'aws_s3_bucket_inventory',
'aws_s3_bucket_metric',
'aws_s3_bucket_notification',
'aws_s3_bucket_policy',
],
],
[
'aws_ec2',
[
'aws_instance',
'aws_key_pair',
'aws_ami',
'aws_ebs_snapshot',
'aws_ebs_volume',
'aws_eip',
'aws_eip_association',
'aws_volume_attachment',
'aws_launch_configuration',
'aws_launch_template',
],
],
['aws_lambda', ['aws_lambda_function', 'aws_lambda_event_source_mapping']],
[
'aws_rds',
[
'aws_db_instance',
'aws_db_subnet_group',
'aws_rds_cluster',
'aws_rds_cluster_endpoint',
'aws_rds_cluster_instance',
],
],
['aws_route53', ['aws_route53_record', 'aws_route53_zone']],
[
'aws_iam',
[
'aws_iam_access_key',
'aws_iam_policy',
'aws_iam_policy_attachment',
'aws_iam_role',
'aws_iam_role_policy',
'aws_iam_role_policy_attachment',
'aws_iam_user',
'aws_iam_user_policy',
'aws_iam_user_policy_attachment',
],
],
[
'aws_vpc',
[
'aws_security_group',
'aws_security_group_rule',
'aws_subnet',
'aws_default_vpc',
'aws_vpc',
'aws_default_security_group',
'aws_route_table',
'aws_default_route_table',
'aws_route',
'aws_route_table_association',
'aws_nat_gateway',
'aws_internet_gateway',
],
],
[
'aws_api_gateway',
[
'aws_api_gateway_resource',
'aws_api_gateway_rest_api',
'aws_api_gateway_account',
'aws_api_gateway_api_key',
'aws_api_gateway_authorizer',
'aws_api_gateway_base_path_mapping',
'aws_api_gateway_domain_name',
'aws_api_gateway_gateway_response',
'aws_api_gateway_integration',
'aws_api_gateway_integration_response',
'aws_api_gateway_method',
'aws_api_gateway_method_response',
'aws_api_gateway_method_settings',
'aws_api_gateway_model',
'aws_api_gateway_request_validator',
'aws_api_gateway_rest_api_policy',
'aws_api_gateway_stage',
'aws_api_gateway_vpc_link',
],
],
[
'aws_apigatewayv2',
[
'aws_apigatewayv2_api',
'aws_apigatewayv2_api_mapping',
'aws_apigatewayv2_authorizer',
'aws_apigatewayv2_deployment',
'aws_apigatewayv2_domain_name',
'aws_apigatewayv2_integration',
'aws_apigatewayv2_integration_response',
'aws_apigatewayv2_model',
'aws_apigatewayv2_route',
'aws_apigatewayv2_route_response',
'aws_apigatewayv2_stage',
'aws_apigatewayv2_vpc_link',
],
],
['aws_sqs', ['aws_sqs_queue', 'aws_sqs_queue_policy']],
[
'aws_sns',
['aws_sns_topic', 'aws_sns_topic_policy', 'aws_sns_topic_subscription'],
],
['aws_ecr', ['aws_ecr_repository']],
['aws_cloudfront', ['aws_cloudfront_distribution']],
['aws_kms', ['aws_kms_key', 'aws_kms_alias']],
['aws_dynamodb', ['aws_dynamodb_table']],
// Azure
['azure_base', ['azurerm_resource_group']],
['azure_compute', ['azurerm_image', 'azurerm_ssh_public_key']],
['azure_storage', ['azurerm_storage_account', 'azurerm_storage_container']],
[
'azure_network',
[
'azurerm_resource_group',
'azurerm_subnet',
'azurerm_public_ip',
'azurerm_firewall',
'azurerm_route',
'azurerm_route_table',
'azurerm_network_security_group',
],
],
['azure_container', ['azurerm_container_registry']],
[
'azure_database',
['azurerm_postgresql_server', 'azurerm_postgresql_database'],
],
['azure_loadbalancer', ['azurerm_lb', 'azurerm_lb_rule']],
[
'azure_private_dns',
[
'azurerm_private_dns_a_record',
'azurerm_private_dns_aaaa_record',
'azurerm_private_dns_cname_record',
'azurerm_private_dns_mx_record',
'azurerm_private_dns_ptr_record',
'azurerm_private_dns_srv_record',
'azurerm_private_dns_txt_record',
'azurerm_private_dns_zone',
],
],
// Google
[
'google_cloud_platform',
[
'google_project_iam_binding',
'google_project_iam_member',
'google_project_iam_policy',
],
],
[
'google_cloud_storage',
[
'google_storage_bucket',
'google_storage_bucket_iam_binding',
'google_storage_bucket_iam_member',
'google_storage_bucket_iam_policy',
],
],
[
'google_compute_engine',
[
'google_compute_address',
'google_compute_disk',
'google_compute_global_address',
'google_compute_firewall',
'google_compute_health_check',
'google_compute_image',
'google_compute_instance',
'google_compute_instance_group',
'google_compute_network',
'google_compute_node_group',
'google_compute_router',
'google_compute_subnetwork',
],
],
['google_cloud_dns', ['google_dns_managed_zone']],
[
'google_cloud_bigtable',
['google_bigtable_instance', 'google_bigtable_table'],
],
[
'google_cloud_bigquery',
['google_bigquery_table', 'google_bigquery_dataset'],
],
['google_cloud_functions', ['google_cloudfunctions_function']],
['google_cloud_sql', ['google_sql_database_instance']],
['google_cloud_run', ['google_cloud_run_service']],
]);
function verifyServiceMappingExists(services) {
if (services.length == 0) {
throw new InvalidServiceError('');
}
for (const s of services) {
if (!exports.services2resources.has(s)) {
throw new InvalidServiceError(`We were unable to match service "${s}". Please provide a valid service name: ${existingServiceNames()}`);
}
}
}
exports.verifyServiceMappingExists = verifyServiceMappingExists;
function findServiceMappingForType(type) {
var _a;
for (const service of exports.services2resources.keys()) {
if ((_a = exports.services2resources.get(service)) === null || _a === void 0 ? void 0 : _a.includes(type)) {
return service;
}
}
return '';
}
exports.findServiceMappingForType = findServiceMappingForType;
function existingServiceNames() {
let res = '';
for (const s of exports.services2resources.keys()) {
res += `${s},`;
}
return res.substring(0, res.length - 1);
}
function createIgnorePattern(services) {
return createIgnorePatternWithMap(services, exports.services2resources);
}
exports.createIgnorePattern = createIgnorePattern;
function createIgnorePatternWithMap(services, serviceMap) {
let res = '*';
const seenResources = new Set();
for (const s of services) {
const resourcePatterns = serviceMap.get(s);
for (const rp of resourcePatterns || []) {
// A resource might belong to multiple services, skip it if already processed
if (seenResources.has(rp)) {
continue;
}
res += `,!${rp}`;
seenResources.add(rp);
}
}
return res;
}
exports.createIgnorePatternWithMap = createIgnorePatternWithMap;
class InvalidServiceError extends errors_1.CustomError {
constructor(msg) {
super(msg);
this.code = types_1.IaCErrorCodes.InvalidServiceError;
this.strCode = error_utils_1.getErrorStringCode(this.code);
this.userMessage = msg;
}
}
exports.InvalidServiceError = InvalidServiceError;
/***/ })
};
;
//# sourceMappingURL=895.index.js.map