1
1
import path from 'path' ;
2
2
import fs from 'fs' ;
3
- import pkgUp from 'eslint-module-utils/pkgUp' ;
4
3
import minimatch from 'minimatch' ;
5
4
import resolve from 'eslint-module-utils/resolve' ;
6
5
import moduleVisitor from 'eslint-module-utils/moduleVisitor' ;
6
+ import includes from 'array-includes' ;
7
7
import importType from '../core/importType' ;
8
8
import { getFilePackageName } from '../core/packagePath' ;
9
9
import docsUrl from '../docsUrl' ;
@@ -40,16 +40,41 @@ function extractDepFields(pkg) {
40
40
} ;
41
41
}
42
42
43
- function getPackageDepFields ( packageJsonPath , throwAtRead ) {
44
- if ( ! depFieldCache . has ( packageJsonPath ) ) {
45
- const depFields = extractDepFields ( readJSON ( packageJsonPath , throwAtRead ) ) ;
46
- depFieldCache . set ( packageJsonPath , depFields ) ;
43
+ function getPackageDepFields ( packageDir , considerInParents , requireInDir ) {
44
+ const cacheKey = JSON . stringify ( { packageDir, considerInParents } ) ;
45
+
46
+ if ( ! depFieldCache . has ( cacheKey ) ) {
47
+ // try in the current directory, erroring if the user explicitly specified this directory
48
+ // and reading fails
49
+ const parsedPackage = readJSON ( path . join ( packageDir , 'package.json' ) , requireInDir ) ;
50
+ const depFields = extractDepFields ( parsedPackage || { } ) ;
51
+
52
+ // If readJSON returned nothing, we want to keep searching since the current directory didn't
53
+ // have a package.json. Also keep searching if we're merging in some set of parents dependencies.
54
+ // However, if we're already at the root, stop.
55
+ if ( ( ! parsedPackage || considerInParents . length > 0 ) && packageDir !== path . parse ( packageDir ) . root ) {
56
+ const parentDepFields = getPackageDepFields ( path . dirname ( packageDir ) , considerInParents , false ) ;
57
+
58
+ Object . keys ( depFields ) . forEach ( depsKey => {
59
+ if (
60
+ ( depsKey === 'dependencies' && includes ( considerInParents , 'prod' ) ) ||
61
+ ( depsKey === 'devDependencies' && includes ( considerInParents , 'dev' ) ) ||
62
+ ( depsKey === 'peerDependencies' && includes ( considerInParents , 'peer' ) ) ||
63
+ ( depsKey === 'optionalDependencies' && includes ( considerInParents , 'optional' ) ) ||
64
+ ( depsKey === 'bundledDependencies' && includes ( considerInParents , 'bundled' ) )
65
+ ) {
66
+ Object . assign ( depFields [ depsKey ] , parentDepFields [ depsKey ] ) ;
67
+ }
68
+ } ) ;
69
+ }
70
+
71
+ depFieldCache . set ( cacheKey , depFields ) ;
47
72
}
48
73
49
- return depFieldCache . get ( packageJsonPath ) ;
74
+ return depFieldCache . get ( cacheKey ) ;
50
75
}
51
76
52
- function getDependencies ( context , packageDir ) {
77
+ function getDependencies ( context , packageDir , considerInParents ) {
53
78
let paths = [ ] ;
54
79
try {
55
80
const packageContent = {
@@ -71,22 +96,20 @@ function getDependencies(context, packageDir) {
71
96
if ( paths . length > 0 ) {
72
97
// use rule config to find package.json
73
98
paths . forEach ( dir => {
74
- const packageJsonPath = path . join ( dir , 'package.json' ) ;
75
- const _packageContent = getPackageDepFields ( packageJsonPath , true ) ;
99
+ const _packageContent = getPackageDepFields ( dir , considerInParents , true ) ;
76
100
Object . keys ( packageContent ) . forEach ( depsKey =>
77
101
Object . assign ( packageContent [ depsKey ] , _packageContent [ depsKey ] ) ,
78
102
) ;
79
103
} ) ;
80
104
} else {
81
- const packageJsonPath = pkgUp ( {
82
- cwd : context . getPhysicalFilename ? context . getPhysicalFilename ( ) : context . getFilename ( ) ,
83
- normalize : false ,
84
- } ) ;
85
-
86
105
// use closest package.json
87
106
Object . assign (
88
107
packageContent ,
89
- getPackageDepFields ( packageJsonPath , false ) ,
108
+ getPackageDepFields (
109
+ path . dirname ( context . getPhysicalFilename ? context . getPhysicalFilename ( ) : context . getFilename ( ) ) ,
110
+ considerInParents ,
111
+ false ,
112
+ ) ,
90
113
) ;
91
114
}
92
115
@@ -268,6 +291,7 @@ module.exports = {
268
291
'packageDir' : { 'type' : [ 'string' , 'array' ] } ,
269
292
'includeInternal' : { 'type' : [ 'boolean' ] } ,
270
293
'includeTypes' : { 'type' : [ 'boolean' ] } ,
294
+ 'considerInParents' : { 'type' : [ 'array' ] } ,
271
295
} ,
272
296
'additionalProperties' : false ,
273
297
} ,
@@ -277,7 +301,7 @@ module.exports = {
277
301
create ( context ) {
278
302
const options = context . options [ 0 ] || { } ;
279
303
const filename = context . getPhysicalFilename ? context . getPhysicalFilename ( ) : context . getFilename ( ) ;
280
- const deps = getDependencies ( context , options . packageDir ) || extractDepFields ( { } ) ;
304
+ const deps = getDependencies ( context , options . packageDir , options . considerInParents || [ ] ) || extractDepFields ( { } ) ;
281
305
282
306
const depsOptions = {
283
307
allowDevDeps : testConfig ( options . devDependencies , filename ) !== false ,
0 commit comments