snyk-nuget-plugin
Version:
Snyk CLI NuGet plugin
178 lines • 7.34 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.cloneShallow = cloneShallow;
exports.fromFolderName = fromFolderName;
exports.parse = parse;
const fs = require("fs");
const path = require("path");
const debugModule = require("debug");
const nuspec_parser_1 = require("./nuspec-parser");
const invalid_folder_format_error_1 = require("../../errors/invalid-folder-format-error");
const debug = debugModule('snyk');
function cloneShallow(dep) {
// clone, without the dependencies
return {
dependencies: {},
name: dep.name,
version: dep.version,
};
}
function extractFromDotVersionNotation(expression) {
const regexParseResult = /(?=\S+)(?=\.{1})((\.\d+)+((-?\w+\.?\d*)|(\+?[0-9a-f]{5,40}))?)/.exec(expression);
if (regexParseResult == null) {
debug(`Failed to extract version from the folder: ${expression}. This is not supposed to happen and should be reported - the folders should always be in the form of [FolderName].[semantic version]`);
throw new invalid_folder_format_error_1.InvalidFolderFormatError(`Tried to parse package version from a folder name but failed. I received: ${expression}`);
}
const versionRef = regexParseResult?.[0];
const name = expression.split(versionRef)[0];
return {
name,
version: versionRef.slice(1),
};
}
function fromFolderName(folderName) {
debug('Extracting by folder name ' + folderName);
const info = extractFromDotVersionNotation(folderName);
return {
dependencies: {},
name: info.name,
version: info.version,
};
}
function injectPath(dep, packagesFolder) {
dep.path = dep.localPath
? path.resolve(packagesFolder, dep.localPath)
: path.resolve(packagesFolder, dep.name + '.' + dep.version);
if (dep.localPath) {
delete dep.localPath;
}
}
function scanInstalled(installedPackages, packagesFolder) {
const flattenedPackageList = {};
debug('Located ' + installedPackages.length + ' packages in manifest');
installedPackages.forEach((entry) => {
injectPath(entry, packagesFolder);
flattenedPackageList[entry.name] =
flattenedPackageList[entry.name] || entry;
debug('Entry: ' + entry.name + ' -> ' + entry.path);
});
try {
debug('Scanning local installed folders');
debug('Trying to read from installed packages folder: ' + packagesFolder);
fs.readdirSync(packagesFolder)
.map((folderName) => {
try {
return fromFolderName(folderName);
}
catch (error) {
debug('Unable to parse dependency from folder');
debug(error);
return;
}
})
.forEach((dep) => {
if (dep) {
injectPath(dep, packagesFolder);
// only add a package from packages folder if version is different
if (flattenedPackageList[dep.name] &&
flattenedPackageList[dep.name].version !== dep.version) {
// prefer found from packages folder (dep) over existing
debug('For package ' +
dep.name +
' the version ' +
flattenedPackageList[dep.name].version +
' was extracted from manifest file.' +
'\nWe are overwriting it with version ' +
dep.version +
' from the packages folder');
flattenedPackageList[dep.name] = dep;
}
}
});
}
catch (error) {
debug('Could not complete packages folder scanning');
debug(error);
}
return flattenedPackageList;
}
async function fetchNugetInformationFromPackages(flattenedPackageList, targetFramework) {
const nugetPackageInformation = [];
// begin collecting information from .nuget files on installed packages
debug('Trying to analyze .nuspec files');
for (const name of Object.keys(flattenedPackageList)) {
try {
const dep = flattenedPackageList[name];
debug('...' + name);
const resolved = await (0, nuspec_parser_1.parseNuspec)(dep, targetFramework);
nugetPackageInformation.push(resolved);
}
catch (e) {
debug('Failed parsing nuspec file');
debug(e);
// log but make sure to rethrow the error
// why? if we cannot parse nuspec file, we got nothing to do!
throw e;
}
}
return nugetPackageInformation;
}
function processNugetInformation(nuspecResolutionChain) {
const nuspecResolutions = {};
nuspecResolutionChain.forEach((resolution) => {
if (!resolution) {
return;
} // jscs:ignore
debug('.nuspec analyzed for ' + resolution.name);
nuspecResolutions[resolution.name] = resolution;
});
return nuspecResolutions;
}
function buildTree(node, requiredChildren, flattenedPackageList, nuspecResolutions) {
for (const requiredChild of requiredChildren) {
let transitiveDependency;
if (flattenedPackageList[requiredChild.name]) {
// fetch from repo
transitiveDependency = cloneShallow(flattenedPackageList[requiredChild.name]);
}
else {
// create as new (uninstalled)
transitiveDependency = {
dependencies: {},
name: requiredChild.name,
version: requiredChild.version,
};
}
const transitiveChildren = (nuspecResolutions[transitiveDependency.name] &&
nuspecResolutions[transitiveDependency.name].children) ||
[];
buildTree(transitiveDependency, transitiveChildren, flattenedPackageList, nuspecResolutions);
node.dependencies[transitiveDependency.name] = transitiveDependency;
}
}
async function parse(tree, manifest, targetFramework, packagesFolder) {
if (!targetFramework) {
throw new Error('No valid Dotnet target framework found');
}
const flattenedPackageList = scanInstalled(manifest, packagesFolder);
const nugetPackageInformation = await fetchNugetInformationFromPackages(flattenedPackageList, targetFramework);
const nuspecResolutions = processNugetInformation(nugetPackageInformation);
// .nuget parsing is complete, returned as array of promise resolutions
// now the flat list should be rebuilt as a tree
debug('Building dependency tree');
const nugetKeys = Object.keys(nuspecResolutions);
Object.keys(flattenedPackageList).forEach((packageName) => {
tree.dependencies[packageName] = cloneShallow(flattenedPackageList[packageName]);
});
if (nugetKeys.length > 0) {
// local folders scanned, build list from .nuspec
for (const key of nugetKeys) {
const resolution = nuspecResolutions[key];
const node = cloneShallow(flattenedPackageList[resolution.name]);
buildTree(node, resolution.children, flattenedPackageList, nuspecResolutions);
tree.dependencies[node.name] = node;
}
}
return tree;
}
//# sourceMappingURL=dotnet-framework-parser.js.map
;