atcheck/lsscanner/new_ubuntu/node_modules/npm/lib/utils/funding.js

152 lines
3.4 KiB
JavaScript
Raw Normal View History

2019-12-16 18:42:21 +01:00
'use strict'
const URL = require('url').URL
exports.getFundingInfo = getFundingInfo
exports.validFundingUrl = validFundingUrl
// Is the value of a `funding` property of a `package.json`
// a valid type+url for `npm fund` to display?
function validFundingUrl (funding) {
if (!funding) return false
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)
}
function getFundingInfo (idealTree, opts) {
let length = 0
const seen = new Set()
const { countOnly } = opts || {}
const empty = () => Object.create(null)
const _trailingDependencies = Symbol('trailingDependencies')
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 retrieveFunding (funding) {
return typeof funding === 'string'
? {
url: funding
}
: funding
}
function getFundingDependencies (tree) {
const deps = tree && tree.dependencies
if (!deps) return empty()
// broken into two steps to make sure items appearance
// within top levels takes precedence over nested ones
return (Object.keys(deps)).map((key) => {
const dep = deps[key]
const { name, funding, version } = dep
const fundingItem = {}
// avoids duplicated items within the funding tree
if (tracked(name, version)) return empty()
if (version) {
fundingItem.version = version
}
if (funding && validFundingUrl(funding)) {
fundingItem.funding = retrieveFunding(funding)
length++
}
return {
dep,
fundingItem
}
}).reduce((res, { dep, fundingItem }, i) => {
if (!fundingItem) return res
// recurse
const dependencies = dep.dependencies &&
Object.keys(dep.dependencies).length > 0 &&
getFundingDependencies(dep)
// 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(dependencies)) {
fundingItem.dependencies = retrieveDependencies(dependencies)
}
if (fundingItem.funding) {
res[dep.name] = fundingItem
} else if (fundingItem.dependencies) {
res[_trailingDependencies] =
Object.assign(
empty(),
res[_trailingDependencies],
fundingItem.dependencies
)
}
return res
}, empty())
}
const idealTreeDependencies = getFundingDependencies(idealTree)
const result = {
length
}
if (!countOnly) {
result.name = idealTree.name || idealTree.path
if (idealTree && idealTree.version) {
result.version = idealTree.version
}
if (idealTree && idealTree.funding) {
result.funding = retrieveFunding(idealTree.funding)
}
result.dependencies =
retrieveDependencies(idealTreeDependencies)
}
return result
}