1457 lines
258 KiB
JavaScript
1457 lines
258 KiB
JavaScript
|
"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
|