2018-09-17 21:09:02 +00:00
#!/usr/bin/env node
const { GitProcess } = require ( 'dugite' )
const childProcess = require ( 'child_process' )
2018-09-20 05:41:01 +00:00
const fs = require ( 'fs' )
2018-09-17 21:09:02 +00:00
const klaw = require ( 'klaw' )
const minimist = require ( 'minimist' )
const path = require ( 'path' )
const SOURCE _ROOT = path . normalize ( path . dirname ( _ _dirname ) )
2018-10-24 18:25:13 +00:00
const DEPOT _TOOLS = path . resolve ( SOURCE _ROOT , '..' , 'third_party' , 'depot_tools' )
2018-09-17 21:09:02 +00:00
const BLACKLIST = new Set ( [
2019-06-19 20:56:58 +00:00
[ 'shell' , 'browser' , 'mac' , 'atom_application.h' ] ,
[ 'shell' , 'browser' , 'mac' , 'atom_application_delegate.h' ] ,
[ 'shell' , 'browser' , 'resources' , 'win' , 'resource.h' ] ,
[ 'shell' , 'browser' , 'notifications' , 'mac' , 'notification_center_delegate.h' ] ,
2019-09-06 05:52:54 +00:00
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'atom_bundle_mover.h' ] ,
2019-06-19 20:56:58 +00:00
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'atom_menu_controller.h' ] ,
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'atom_ns_window.h' ] ,
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'atom_ns_window_delegate.h' ] ,
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'atom_preview_item.h' ] ,
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'atom_touch_bar.h' ] ,
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'atom_inspectable_web_contents_view.h' ] ,
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'event_dispatching_window.h' ] ,
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'NSColor+Hex.h' ] ,
[ 'shell' , 'browser' , 'ui' , 'cocoa' , 'NSString+ANSI.h' ] ,
[ 'shell' , 'common' , 'node_includes.h' ] ,
2019-02-13 23:24:28 +00:00
[ 'spec' , 'static' , 'jquery-2.0.3.min.js' ] ,
[ 'spec' , 'ts-smoke' , 'electron' , 'main.ts' ] ,
[ 'spec' , 'ts-smoke' , 'electron' , 'renderer.ts' ] ,
[ 'spec' , 'ts-smoke' , 'runner.js' ]
2018-09-17 21:09:02 +00:00
] . map ( tokens => path . join ( SOURCE _ROOT , ... tokens ) ) )
function spawnAndCheckExitCode ( cmd , args , opts ) {
opts = Object . assign ( { stdio : 'inherit' } , opts )
const status = childProcess . spawnSync ( cmd , args , opts ) . status
if ( status ) process . exit ( status )
}
2019-05-02 12:05:37 +00:00
function cpplint ( args ) {
const result = childProcess . spawnSync ( 'cpplint.py' , args , { encoding : 'utf8' } )
// cpplint.py writes EVERYTHING to stderr, including status messages
if ( result . stderr ) {
for ( const line of result . stderr . split ( /[\r\n]+/ ) ) {
if ( line . length && ! line . startsWith ( 'Done processing ' ) && line !== 'Total errors found: 0' ) {
console . warn ( line )
}
}
}
if ( result . status ) {
process . exit ( result . status )
}
}
2020-02-04 20:19:40 +00:00
function isObjCHeader ( filename ) {
return /\/(mac|cocoa)\// . test ( filename )
}
2018-09-17 21:09:02 +00:00
const LINTERS = [ {
key : 'c++' ,
2019-12-05 09:46:34 +00:00
roots : [ 'shell' ] ,
2020-02-04 20:19:40 +00:00
test : filename => filename . endsWith ( '.cc' ) || ( filename . endsWith ( '.h' ) && ! isObjCHeader ( filename ) ) ,
2018-09-17 21:09:02 +00:00
run : ( opts , filenames ) => {
2018-10-16 05:59:45 +00:00
if ( opts . fix ) {
spawnAndCheckExitCode ( 'python' , [ 'script/run-clang-format.py' , '--fix' , ... filenames ] )
} else {
spawnAndCheckExitCode ( 'python' , [ 'script/run-clang-format.py' , ... filenames ] )
}
2019-05-02 12:05:37 +00:00
cpplint ( filenames )
}
} , {
key : 'objc' ,
2019-06-19 20:56:58 +00:00
roots : [ 'shell' ] ,
2019-05-02 12:05:37 +00:00
test : filename => filename . endsWith ( '.mm' ) ,
run : ( opts , filenames ) => {
if ( opts . fix ) {
spawnAndCheckExitCode ( 'python' , [ 'script/run-clang-format.py' , '--fix' , ... filenames ] )
} else {
spawnAndCheckExitCode ( 'python' , [ 'script/run-clang-format.py' , ... filenames ] )
2018-09-19 13:42:03 +00:00
}
2019-05-02 12:05:37 +00:00
const filter = [
'-readability/casting' ,
'-whitespace/braces' ,
'-whitespace/indent' ,
'-whitespace/parens'
]
cpplint ( [ '--extensions=mm' , ` --filter= ${ filter . join ( ',' ) } ` , ... filenames ] )
2018-09-17 21:09:02 +00:00
}
} , {
key : 'python' ,
roots : [ 'script' ] ,
test : filename => filename . endsWith ( '.py' ) ,
run : ( opts , filenames ) => {
2018-10-24 18:25:13 +00:00
const rcfile = path . join ( DEPOT _TOOLS , 'pylintrc' )
2018-09-17 21:09:02 +00:00
const args = [ '--rcfile=' + rcfile , ... filenames ]
const env = Object . assign ( { PYTHONPATH : path . join ( SOURCE _ROOT , 'script' ) } , process . env )
spawnAndCheckExitCode ( 'pylint.py' , args , { env } )
}
} , {
key : 'javascript' ,
2019-11-01 20:37:02 +00:00
roots : [ 'lib' , 'spec' , 'spec-main' , 'script' , 'default_app' ] ,
ignoreRoots : [ 'spec/node_modules' , 'spec-main/node_modules' ] ,
2019-02-06 18:27:20 +00:00
test : filename => filename . endsWith ( '.js' ) || filename . endsWith ( '.ts' ) ,
2018-09-17 21:09:02 +00:00
run : ( opts , filenames ) => {
const cmd = path . join ( SOURCE _ROOT , 'node_modules' , '.bin' , 'eslint' )
2019-02-06 18:27:20 +00:00
const args = [ '--cache' , '--ext' , '.js,.ts' , ... filenames ]
2018-09-17 21:09:02 +00:00
if ( opts . fix ) args . unshift ( '--fix' )
spawnAndCheckExitCode ( cmd , args , { cwd : SOURCE _ROOT } )
}
2018-10-03 23:03:26 +00:00
} , {
key : 'gn' ,
roots : [ '.' ] ,
test : filename => filename . endsWith ( '.gn' ) || filename . endsWith ( '.gni' ) ,
run : ( opts , filenames ) => {
const allOk = filenames . map ( filename => {
2018-10-24 18:25:13 +00:00
const env = Object . assign ( {
CHROMIUM _BUILDTOOLS _PATH : path . resolve ( SOURCE _ROOT , '..' , 'buildtools' ) ,
DEPOT _TOOLS _WIN _TOOLCHAIN : '0'
} , process . env )
// Users may not have depot_tools in PATH.
env . PATH = ` ${ env . PATH } ${ path . delimiter } ${ DEPOT _TOOLS } `
2018-10-03 23:03:26 +00:00
const args = [ 'format' , filename ]
if ( ! opts . fix ) args . push ( '--dry-run' )
2018-10-24 18:25:13 +00:00
const result = childProcess . spawnSync ( 'gn' , args , { env , stdio : 'inherit' , shell : true } )
2018-10-03 23:03:26 +00:00
if ( result . status === 0 ) {
return true
} else if ( result . status === 2 ) {
console . log ( ` GN format errors in " ${ filename } ". Run 'gn format " ${ filename } "' or rerun with --fix to fix them. ` )
return false
} else {
console . log ( ` Error running 'gn format --dry-run " ${ filename } "': exit code ${ result . status } ` )
return false
}
} ) . every ( x => x )
if ( ! allOk ) {
process . exit ( 1 )
}
}
2019-06-19 17:48:15 +00:00
} , {
key : 'patches' ,
roots : [ 'patches' ] ,
test : ( ) => true ,
2019-11-04 19:04:18 +00:00
run : ( opts , filenames ) => {
2019-06-19 17:48:15 +00:00
const patchesDir = path . resolve ( _ _dirname , '../patches' )
for ( const patchTarget of fs . readdirSync ( patchesDir ) ) {
const targetDir = path . resolve ( patchesDir , patchTarget )
// If the config does not exist that is OK, we just skip this dir
const targetConfig = path . resolve ( targetDir , 'config.json' )
if ( ! fs . existsSync ( targetConfig ) ) continue
const config = JSON . parse ( fs . readFileSync ( targetConfig , 'utf8' ) )
for ( const key of Object . keys ( config ) ) {
// The directory the config points to should exist
const targetPatchesDir = path . resolve ( _ _dirname , '../../..' , key )
if ( ! fs . existsSync ( targetPatchesDir ) ) throw new Error ( ` target patch directory: " ${ targetPatchesDir } " does not exist ` )
// We need a .patches file
const dotPatchesPath = path . resolve ( targetPatchesDir , '.patches' )
if ( ! fs . existsSync ( dotPatchesPath ) ) throw new Error ( ` .patches file: " ${ dotPatchesPath } " does not exist ` )
// Read the patch list
const patchFileList = fs . readFileSync ( dotPatchesPath , 'utf8' ) . trim ( ) . split ( '\n' )
const patchFileSet = new Set ( patchFileList )
patchFileList . reduce ( ( seen , file ) => {
if ( seen . has ( file ) ) {
throw new Error ( ` ' ${ file } ' is listed in ${ dotPatchesPath } more than once ` )
}
return seen . add ( file )
} , new Set ( ) )
if ( patchFileList . length !== patchFileSet . size ) throw new Error ( 'each patch file should only be in the .patches file once' )
for ( const file of fs . readdirSync ( targetPatchesDir ) ) {
// Ignore the .patches file and READMEs
if ( file === '.patches' || file === 'README.md' ) continue
if ( ! patchFileSet . has ( file ) ) {
throw new Error ( ` Expected the .patches file at " ${ dotPatchesPath } " to contain a patch file (" ${ file } ") present in the directory but it did not ` )
}
patchFileSet . delete ( file )
}
// If anything is left in this set, it means it did not exist on disk
if ( patchFileSet . size > 0 ) {
throw new Error ( ` Expected all the patch files listed in the .patches file at " ${ dotPatchesPath } " to exist but some did not: \n ${ JSON . stringify ( [ ... patchFileSet . values ( ) ] , null , 2 ) } ` )
}
}
}
2019-11-04 19:04:18 +00:00
let ok = true
filenames . filter ( f => f . endsWith ( '.patch' ) ) . forEach ( f => {
const patchText = fs . readFileSync ( f , 'utf8' )
2019-12-13 17:18:45 +00:00
if ( /^Subject: .*$\s+^diff/m . test ( patchText ) ) {
2019-11-04 19:04:18 +00:00
console . warn ( ` Patch file ' ${ f } ' has no description. Every patch must contain a justification for why the patch exists and the plan for its removal. ` )
ok = false
}
} )
if ( ! ok ) {
process . exit ( 1 )
}
2019-06-19 17:48:15 +00:00
}
2018-09-17 21:09:02 +00:00
} ]
function parseCommandLine ( ) {
let help
const opts = minimist ( process . argv . slice ( 2 ) , {
2019-06-19 17:48:15 +00:00
boolean : [ 'c++' , 'objc' , 'javascript' , 'python' , 'gn' , 'patches' , 'help' , 'changed' , 'fix' , 'verbose' , 'only' ] ,
2018-09-17 21:09:02 +00:00
alias : { 'c++' : [ 'cc' , 'cpp' , 'cxx' ] , javascript : [ 'js' , 'es' ] , python : 'py' , changed : 'c' , help : 'h' , verbose : 'v' } ,
unknown : arg => { help = true }
} )
if ( help || opts . help ) {
2019-01-21 22:46:32 +00:00
console . log ( 'Usage: script/lint.js [--cc] [--js] [--py] [-c|--changed] [-h|--help] [-v|--verbose] [--fix] [--only -- file1 file2]' )
2018-09-17 21:09:02 +00:00
process . exit ( 0 )
}
return opts
}
async function findChangedFiles ( top ) {
2018-09-19 13:40:57 +00:00
const result = await GitProcess . exec ( [ 'diff' , '--name-only' , '--cached' ] , top )
2018-09-17 21:09:02 +00:00
if ( result . exitCode !== 0 ) {
console . log ( 'Failed to find changed files' , GitProcess . parseError ( result . stderr ) )
process . exit ( 1 )
}
const relativePaths = result . stdout . split ( /\r\n|\r|\n/g )
const absolutePaths = relativePaths . map ( x => path . join ( top , x ) )
return new Set ( absolutePaths )
}
async function findMatchingFiles ( top , test ) {
return new Promise ( ( resolve , reject ) => {
const matches = [ ]
2019-04-30 20:59:47 +00:00
klaw ( top , {
filter : f => path . basename ( f ) !== '.bin'
} )
2018-09-17 21:09:02 +00:00
. on ( 'end' , ( ) => resolve ( matches ) )
. on ( 'data' , item => {
if ( test ( item . path ) ) {
matches . push ( item . path )
}
} )
} )
}
async function findFiles ( args , linter ) {
let filenames = [ ]
let whitelist = null
// build the whitelist
if ( args . changed ) {
whitelist = await findChangedFiles ( SOURCE _ROOT )
if ( ! whitelist . size ) {
2018-10-16 05:59:45 +00:00
return [ ]
2018-09-17 21:09:02 +00:00
}
2019-01-21 22:46:32 +00:00
} else if ( args . only ) {
2019-12-13 17:18:45 +00:00
whitelist = new Set ( args . _ . map ( p => path . resolve ( p ) ) )
2018-09-17 21:09:02 +00:00
}
// accumulate the raw list of files
for ( const root of linter . roots ) {
const files = await findMatchingFiles ( path . join ( SOURCE _ROOT , root ) , linter . test )
filenames . push ( ... files )
}
2018-09-20 05:41:01 +00:00
for ( const ignoreRoot of ( linter . ignoreRoots ) || [ ] ) {
const ignorePath = path . join ( SOURCE _ROOT , ignoreRoot )
if ( ! fs . existsSync ( ignorePath ) ) continue
const ignoreFiles = new Set ( await findMatchingFiles ( ignorePath , linter . test ) )
filenames = filenames . filter ( fileName => ! ignoreFiles . has ( fileName ) )
}
2018-09-17 21:09:02 +00:00
// remove blacklisted files
filenames = filenames . filter ( x => ! BLACKLIST . has ( x ) )
// if a whitelist exists, remove anything not in it
if ( whitelist ) {
filenames = filenames . filter ( x => whitelist . has ( x ) )
}
2018-10-16 05:59:45 +00:00
// it's important that filenames be relative otherwise clang-format will
// produce patches with absolute paths in them, which `git apply` will refuse
// to apply.
return filenames . map ( x => path . relative ( SOURCE _ROOT , x ) )
2018-09-17 21:09:02 +00:00
}
async function main ( ) {
const opts = parseCommandLine ( )
// no mode specified? run 'em all
2020-01-29 17:03:53 +00:00
if ( ! opts [ 'c++' ] && ! opts . javascript && ! opts . objc && ! opts . python && ! opts . gn && ! opts . patches ) {
opts [ 'c++' ] = opts . javascript = opts . objc = opts . python = opts . gn = opts . patches = true
2018-09-17 21:09:02 +00:00
}
const linters = LINTERS . filter ( x => opts [ x . key ] )
for ( const linter of linters ) {
const filenames = await findFiles ( opts , linter )
if ( filenames . length ) {
if ( opts . verbose ) { console . log ( ` linting ${ filenames . length } ${ linter . key } ${ filenames . length === 1 ? 'file' : 'files' } ` ) }
linter . run ( opts , filenames )
}
}
}
if ( process . mainModule === module ) {
2018-09-20 05:41:01 +00:00
main ( ) . catch ( ( error ) => {
console . error ( error )
process . exit ( 1 )
} )
2018-09-17 21:09:02 +00:00
}