Skip to content

Commit

Permalink
Catch picomatch error if invalid filter is given (#1453).
Browse files Browse the repository at this point in the history
  • Loading branch information
raineorshine committed Sep 17, 2024
1 parent 4b79c06 commit 2a6df8c
Show file tree
Hide file tree
Showing 5 changed files with 42 additions and 18 deletions.
8 changes: 5 additions & 3 deletions src/lib/filterAndReject.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,9 @@ import { Maybe } from '../types/Maybe'
import { VersionSpec } from '../types/VersionSpec'

/**
* Creates a filter function from a given filter string. Supports
* strings, wildcards, comma-or-space-delimited lists, and regexes.
* Creates a filter function from a given filter string.
* Supports strings, wildcards, comma-or-space-delimited lists, and regexes.
* The filter function *may* throw an exception if the filter pattern is invalid.
*
* @param [filterPattern]
* @returns
Expand Down Expand Up @@ -65,8 +66,9 @@ function composeFilter(filterPattern: FilterPattern): (name: string, versionSpec
// limit the arity to 1 to avoid passing the value
return predicate
}

/**
* Composes a filter function from filter, reject, filterVersion, and rejectVersion patterns.
* Composes a filter function from filter, reject, filterVersion, and rejectVersion patterns. The filter function *may* throw an exception if the filter pattern is invalid.
*
* @param [filter]
* @param [reject]
Expand Down
2 changes: 1 addition & 1 deletion src/lib/filterObject.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { Index } from '../types/IndexType'
import keyValueBy from './keyValueBy'

/** Filters an object by a predicate. */
/** Filters an object by a predicate. Does not catch exceptions thrown by the predicate. */
const filterObject = <T>(obj: Index<T>, predicate: (key: string, value: T) => boolean) =>
keyValueBy(obj, (key, value) => (predicate(key, value) ? { [key]: value } : null))

Expand Down
24 changes: 15 additions & 9 deletions src/lib/getCurrentDependencies.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import { VersionSpec } from '../types/VersionSpec'
import filterAndReject from './filterAndReject'
import filterObject from './filterObject'
import { keyValueBy } from './keyValueBy'
import programError from './programError'
import resolveDepSections from './resolveDepSections'

/** Returns true if spec1 is greater than spec2, ignoring invalid version ranges. */
Expand Down Expand Up @@ -51,15 +52,20 @@ function getCurrentDependencies(pkgData: PackageFile = {}, options: Options = {}

// filter & reject dependencies and versions
const workspacePackageMap = keyValueBy(options.workspacePackages || [])
const filteredDependencies = filterObject(
filterObject(allDependencies, name => !workspacePackageMap[name]),
filterAndReject(
options.filter || null,
options.reject || null,
options.filterVersion || null,
options.rejectVersion || null,
),
)
let filteredDependencies: Index<VersionSpec> = {}
try {
filteredDependencies = filterObject(
filterObject(allDependencies, name => !workspacePackageMap[name]),
filterAndReject(
options.filter || null,
options.reject || null,
options.filterVersion || null,
options.rejectVersion || null,
),
)
} catch (err: any) {
programError(options, 'Invalid filter: ' + err.message || err)
}

return filteredDependencies
}
Expand Down
16 changes: 12 additions & 4 deletions src/lib/getInstalledPackages.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { Index } from '../types/IndexType'
import { Options } from '../types/Options'
import { Version } from '../types/Version'
import { VersionSpec } from '../types/VersionSpec'
Expand Down Expand Up @@ -29,10 +30,17 @@ async function getInstalledPackages(options: Options = {}) {

// filter out undefined packages or those with a wildcard
const filterFunction = filterAndReject(options.filter, options.reject, options.filterVersion, options.rejectVersion)
return filterObject(
packages,
(dep: VersionSpec, version: Version) => !!version && !isWildPart(version) && filterFunction(dep, version),
)
let filteredPackages: Index<VersionSpec> = {}
try {
filteredPackages = filterObject(
packages,
(dep: VersionSpec, version: Version) => !!version && !isWildPart(version) && filterFunction(dep, version),
)
} catch (err: any) {
programError(options, 'Invalid filter: ' + err.message || err)
}

return filteredPackages
}

export default getInstalledPackages
10 changes: 9 additions & 1 deletion test/filter.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -191,14 +191,22 @@ describe('filter', () => {
upgraded.should.have.property('fp-and-or')
})

it('trim and ignore empty filter', async () => {
it('trim and ignore empty array', async () => {
const upgraded = (await ncu({
packageData: await fs.readFile(path.join(__dirname, 'test-data/ncu/package2.json'), 'utf-8'),
filter: [],
})) as Index<string>
upgraded.should.have.property('lodash.map')
upgraded.should.have.property('lodash.filter')
})

it('empty string is invalid', async () => {
const promise = ncu({
packageData: await fs.readFile(path.join(__dirname, 'test-data/ncu/package2.json'), 'utf-8'),
filter: ',test',
})
promise.should.eventually.be.rejectedWith('Invalid filter: Expected pattern to be a non-empty string')
})
})

describe('cli', () => {
Expand Down

0 comments on commit 2a6df8c

Please sign in to comment.