-
Notifications
You must be signed in to change notification settings - Fork 274
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* Revert updates to dependency versions. * Add package-lock.json. * Remove root package lock. * Try to get node_modules output. * Cleanup. * Fix make.ks. * Add third party notice generator. * Update docs. * Add third party notice. * Copy node_modules to _build. * Copy ThirdPartyNotices.txt to _build directory. * Prune dev dependencies in _build. * Remove node_modules copy, shouldn't need to be there. * Cleanup. * Clean. * Clean.
- Loading branch information
1 parent
e046740
commit 50b2c9b
Showing
9 changed files
with
1,667 additions
and
10 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,169 @@ | ||
/** | ||
* Run from the root of the vsts-tasks repo. | ||
* Usage: `node generate-third-party-notice.js <task name>` | ||
*/ | ||
const fs = require('fs'); | ||
const os = require('os'); | ||
const path = require('path'); | ||
|
||
const log = { | ||
info(message) { | ||
console.log(`[INFO] ${message}`); | ||
}, | ||
warning(message) { | ||
console.log(`[WARNING] ${message}`); | ||
}, | ||
error(message) { | ||
console.error(`[ERROR] ${message}`) | ||
} | ||
}; | ||
|
||
/** Log `label: ${value}` and pass the value through. */ | ||
function trace(label, value) { | ||
log.info(`${label}: ${value}`); | ||
return value; | ||
} | ||
|
||
/** | ||
* Read `packagePath`'s package.json and deserialize it. | ||
* @param packagePath Absolute path to the NPM package | ||
* @returns Package manifest information parsed from the package's package.json | ||
*/ | ||
function readPackageJson(packagePath) { | ||
log.info(`Reading the package.json for ${packagePath} ...`); | ||
const contents = fs.readFileSync(path.join(packagePath, 'package.json'), { encoding: 'utf-8' }); | ||
return JSON.parse(contents); | ||
} | ||
|
||
/** | ||
* Get the name of the file containing the license for `packagePath`. | ||
* @param packagePath Absolute path to the NPM package | ||
* @returns Absolute path to the license file, or `null` if the license file can't be found | ||
*/ | ||
function findLicense(packagePath) { | ||
log.info(`Finding the license for '${packagePath}'`); | ||
const children = fs.readdirSync(packagePath); | ||
const licenseNames = [ | ||
'LICENSE', | ||
'LICENSE.md', | ||
'LICENSE.txt', | ||
'LICENSE-MIT.txt' | ||
].map(x => x.toLowerCase()); | ||
const candidates = children.filter(x => licenseNames.includes(x.toLowerCase())); | ||
if (!candidates || candidates.length === 0) { | ||
log.warning(`Could not find a license for ${packagePath}`); | ||
return null; | ||
} else { | ||
//console.log(JSON.stringify(candidates)); | ||
if (candidates.length > 1) { | ||
log.warning(`Found multiple license files for ${packagePath}: ${candidates.join(', ')}`); | ||
} | ||
return trace('Found license', path.join(packagePath, candidates[0])); | ||
} | ||
} | ||
|
||
/** | ||
* Scan the contents of the 'node_modules' directory for license information. | ||
* @param modulesRoot NPM package installation directory to scan | ||
* @returns Iterable of objects: `name` x `url` x `licenseText` | ||
*/ | ||
function* collectLicenseInfo(modulesRoot) { | ||
const packagePaths = fs.readdirSync(modulesRoot).map(x => path.join(modulesRoot, x)); | ||
for (let absolutePath of packagePaths) { | ||
log.info(`Collecting license information from ${absolutePath} ...`); | ||
const name = (() => { | ||
const basename = path.basename(absolutePath); | ||
if (path.dirname(absolutePath).endsWith('@types')) { | ||
return `@types/${basename}`; | ||
} else { | ||
return basename; | ||
} | ||
})(); | ||
|
||
if (name === '.bin') { | ||
continue; | ||
} | ||
|
||
if (name === '@types') { | ||
yield* collectLicenseInfo(absolutePath); | ||
continue; | ||
} | ||
|
||
const manifest = readPackageJson(absolutePath); | ||
const license = findLicense(absolutePath); | ||
|
||
let licenseText; | ||
if (license) { | ||
licenseText = fs.readFileSync(license, { encoding: 'utf-8' }); | ||
} else { | ||
licenseText = 'No license text available.'; | ||
} | ||
|
||
yield { | ||
name: name, | ||
url: manifest.repository.url, | ||
licenseText: licenseText | ||
}; | ||
} | ||
} | ||
|
||
/** Generate the third party notice line-by-line. */ | ||
function* thirdPartyNotice(libName, licenseInfo) { | ||
// Preamble | ||
yield ''; | ||
yield 'THIRD-PARTY SOFTWARE NOTICES AND INFORMATION'; | ||
yield 'Do Not Translate or Localize'; | ||
yield ''; | ||
yield `This Visual Studio Team Services extension (${libName}) is based on or incorporates material from the projects listed below (Third Party IP). The original copyright notice and the license under which Microsoft received such Third Party IP, are set forth below. Such licenses and notices are provided for informational purposes only. Microsoft licenses the Third Party IP to you under the licensing terms for the Visual Studio Team Services extension. Microsoft reserves all other rights not expressly granted under this agreement, whether by implication, estoppel or otherwise.`; | ||
yield ''; | ||
|
||
// Enumerated modules | ||
let num = 1; | ||
for (let item of licenseInfo) { | ||
if (item.url) { | ||
yield `${num}.\t${item.name} (${item.url})`; | ||
} else { | ||
yield `${num}.\t${item.name}`; | ||
} | ||
num += 1; | ||
} | ||
|
||
yield ''; | ||
yield ''; | ||
|
||
// Module licenses | ||
for (let item of licenseInfo) { | ||
yield `%% ${item.name} NOTICES, INFORMATION, AND LICENSE BEGIN HERE`; | ||
yield '========================================='; | ||
yield item.licenseText.trim(); | ||
yield '========================================='; | ||
yield `END OF ${item.name} NOTICES, INFORMATION, AND LICENSE`; | ||
yield ''; | ||
} | ||
} | ||
|
||
function writeLines(writeStream, lines) { | ||
const writeLine = (line) => { | ||
writeStream.write(line); | ||
writeStream.write(os.EOL); | ||
}; | ||
|
||
for (let line of lines) { | ||
writeLine(line); | ||
} | ||
} | ||
|
||
function main(args) { | ||
try { | ||
const nodeModuleDir = path.join(__dirname, 'node_modules'); | ||
const licenseInfo = Array.from(collectLicenseInfo(nodeModuleDir)); | ||
|
||
const writeStream = fs.createWriteStream(path.join(__dirname, 'ThirdPartyNotice.txt')); | ||
writeLines(writeStream, thirdPartyNotice('vsts-task-lib', licenseInfo)); | ||
writeStream.end(); | ||
} catch (e) { | ||
log.error(e.message); | ||
} | ||
} | ||
|
||
main(process.argv); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.