node/deps/npm/node_modules/libnpmfund/index.js
Ruy Adorno 4a22850d7f deps: upgrade npm to 7.13.0
PR-URL: https://github.com/nodejs/node/pull/38682
Reviewed-By: Rich Trott <rtrott@gmail.com>
Reviewed-By: Danielle Adams <adamzdanielle@gmail.com>
Reviewed-By: Trivikram Kamat <trivikr.dev@gmail.com>
Reviewed-By: Luigi Pinca <luigipinca@gmail.com>
Reviewed-By: Colin Ihrig <cjihrig@gmail.com>
2021-05-14 22:16:38 -07:00

197 lines
4.7 KiB
JavaScript

'use strict'
const URL = require('url').URL
const Arborist = require('@npmcli/arborist')
// supports object funding and string shorthand, or an array of these
// if original was an array, returns an array; else returns the lone item
function normalizeFunding (funding) {
const normalizeItem = item =>
typeof item === 'string' ? { url: item } : item
const sources = [].concat(funding || []).map(normalizeItem)
return Array.isArray(funding) ? sources : sources[0]
}
// Is the value of a `funding` property of a `package.json`
// a valid type+url for `npm fund` to display?
function isValidFunding (funding) {
if (!funding)
return false
if (Array.isArray(funding))
return funding.every(f => !Array.isArray(f) && isValidFunding(f))
try {
var parsed = new URL(funding.url || funding)
} catch (error) {
return false
}
if (
parsed.protocol !== 'https:' &&
parsed.protocol !== 'http:'
)
return false
return Boolean(parsed.host)
}
const empty = () => Object.create(null)
function readTree (tree, opts) {
let packageWithFundingCount = 0
const seen = new Set()
const { countOnly } = opts || {}
const _trailingDependencies = Symbol('trailingDependencies')
let filterSet
if (opts && opts.workspaces && opts.workspaces.length) {
const arb = new Arborist(opts)
filterSet = arb.workspaceDependencySet(tree, opts.workspaces)
}
function tracked (name, version) {
const key = String(name) + String(version)
if (seen.has(key))
return true
seen.add(key)
}
function retrieveDependencies (dependencies) {
const trailing = dependencies[_trailingDependencies]
if (trailing) {
return Object.assign(
empty(),
dependencies,
trailing
)
}
return dependencies
}
function hasDependencies (dependencies) {
return dependencies && (
Object.keys(dependencies).length ||
dependencies[_trailingDependencies]
)
}
function attachFundingInfo (target, funding) {
if (funding && isValidFunding(funding)) {
target.funding = normalizeFunding(funding)
packageWithFundingCount++
}
}
function getFundingDependencies (tree) {
const edges = tree && tree.edgesOut && tree.edgesOut.values()
if (!edges)
return empty()
const directDepsWithFunding = Array.from(edges).map(edge => {
if (!edge || !edge.to)
return empty()
const node = edge.to.target || edge.to
if (!node.package)
return empty()
if (filterSet && filterSet.size > 0 && !filterSet.has(node))
return empty()
const { name, funding, version } = node.package
// avoids duplicated items within the funding tree
if (tracked(name, version))
return empty()
const fundingItem = {}
if (version)
fundingItem.version = version
attachFundingInfo(fundingItem, funding)
return {
node,
fundingItem,
}
})
return directDepsWithFunding.reduce(
(res, { node, fundingItem }, i) => {
if (!fundingItem ||
fundingItem.length === 0 ||
!node)
return res
// recurse
const transitiveDependencies = node.edgesOut &&
node.edgesOut.size > 0 &&
getFundingDependencies(node)
// if we're only counting items there's no need
// to add all the data to the resulting object
if (countOnly)
return null
if (hasDependencies(transitiveDependencies)) {
fundingItem.dependencies =
retrieveDependencies(transitiveDependencies)
}
if (isValidFunding(fundingItem.funding))
res[node.package.name] = fundingItem
else if (hasDependencies(fundingItem.dependencies)) {
res[_trailingDependencies] =
Object.assign(
empty(),
res[_trailingDependencies],
fundingItem.dependencies
)
}
return res
}, countOnly ? null : empty())
}
const treeDependencies = getFundingDependencies(tree)
const result = {
length: packageWithFundingCount,
}
if (!countOnly) {
const name =
(tree && tree.package && tree.package.name) ||
(tree && tree.name)
result.name = name || (tree && tree.path)
if (tree && tree.package && tree.package.version)
result.version = tree.package.version
if (tree && tree.package && tree.package.funding)
result.funding = normalizeFunding(tree.package.funding)
result.dependencies = retrieveDependencies(treeDependencies)
}
return result
}
async function read (opts) {
const arb = new Arborist(opts)
const tree = await arb.loadActual(opts)
return readTree(tree, opts)
}
module.exports = {
read,
readTree,
normalizeFunding,
isValidFunding,
}