UNPKG

snyk-nuget-plugin

Version:
178 lines 7.34 kB
"use strict"; 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