2020-05-20 22:31:16 +00:00
|
|
|
import * as jsyaml from 'js-yaml'
|
|
|
|
import * as minimatch from 'minimatch'
|
2020-07-11 15:17:56 +00:00
|
|
|
import {File, ChangeStatus} from './file'
|
|
|
|
|
|
|
|
// Type definition of object we expect to load from YAML
|
|
|
|
interface FilterYaml {
|
|
|
|
[name: string]: FilterItemYaml
|
|
|
|
}
|
|
|
|
type FilterItemYaml =
|
|
|
|
| string // Filename pattern, e.g. "path/to/*.js"
|
|
|
|
| {[changeTypes: string]: string} // Change status and filename, e.g. added|modified: "path/to/*.js"
|
|
|
|
| FilterItemYaml[] // Supports referencing another rule via YAML anchor
|
|
|
|
|
|
|
|
// Minimatch options used in all matchers
|
|
|
|
const MinimatchOptions: minimatch.IOptions = {
|
|
|
|
dot: true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Internal representation of one item in named filter rule
|
|
|
|
// Created as simplified form of data in FilterItemYaml
|
|
|
|
interface FilterRuleItem {
|
|
|
|
status?: ChangeStatus[] // Required change status of the matched files
|
|
|
|
matcher: minimatch.IMinimatch // Matches the filename
|
|
|
|
}
|
2020-05-20 22:31:16 +00:00
|
|
|
|
|
|
|
export default class Filter {
|
2020-07-11 15:17:56 +00:00
|
|
|
rules: {[key: string]: FilterRuleItem[]} = {}
|
2020-05-20 22:31:16 +00:00
|
|
|
|
2020-07-11 15:17:56 +00:00
|
|
|
// Creates instance of Filter and load rules from YAML if it's provided
|
|
|
|
constructor(yaml?: string) {
|
|
|
|
if (yaml) {
|
|
|
|
this.load(yaml)
|
2020-05-20 22:31:16 +00:00
|
|
|
}
|
2020-07-11 15:17:56 +00:00
|
|
|
}
|
2020-05-20 22:31:16 +00:00
|
|
|
|
2020-07-11 15:17:56 +00:00
|
|
|
// Load rules from YAML string
|
|
|
|
load(yaml: string): void {
|
2020-07-11 21:33:11 +00:00
|
|
|
if (!yaml) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-07-11 15:17:56 +00:00
|
|
|
const doc = jsyaml.safeLoad(yaml) as FilterYaml
|
|
|
|
if (typeof doc !== 'object') {
|
|
|
|
this.throwInvalidFormatError('Root element is not an object')
|
2020-05-21 11:46:48 +00:00
|
|
|
}
|
|
|
|
|
2020-07-11 15:17:56 +00:00
|
|
|
for (const [key, item] of Object.entries(doc)) {
|
|
|
|
this.rules[key] = this.parseFilterItemYaml(item)
|
2020-05-20 22:31:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-11 15:17:56 +00:00
|
|
|
// Returns dictionary with match result per rule
|
|
|
|
match(files: File[]): {[key: string]: boolean} {
|
2020-05-20 22:31:16 +00:00
|
|
|
const result: {[key: string]: boolean} = {}
|
|
|
|
for (const [key, patterns] of Object.entries(this.rules)) {
|
2020-07-11 15:17:56 +00:00
|
|
|
const match = files.some(file =>
|
|
|
|
patterns.some(
|
|
|
|
rule => (rule.status === undefined || rule.status.includes(file.status)) && rule.matcher.match(file.filename)
|
|
|
|
)
|
|
|
|
)
|
2020-05-20 22:31:16 +00:00
|
|
|
result[key] = match
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2020-07-11 15:17:56 +00:00
|
|
|
private parseFilterItemYaml(item: FilterItemYaml): FilterRuleItem[] {
|
|
|
|
if (Array.isArray(item)) {
|
|
|
|
return flat(item.map(i => this.parseFilterItemYaml(i)))
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof item === 'string') {
|
|
|
|
return [{status: undefined, matcher: new minimatch.Minimatch(item, MinimatchOptions)}]
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof item === 'object') {
|
|
|
|
return Object.entries(item).map(([key, pattern]) => {
|
|
|
|
if (typeof key !== 'string' || typeof pattern !== 'string') {
|
|
|
|
this.throwInvalidFormatError(
|
|
|
|
`Expected [key:string]= pattern:string, but [${key}:${typeof key}]= ${pattern}:${typeof pattern} found`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
status: key
|
|
|
|
.split('|')
|
|
|
|
.map(x => x.trim())
|
|
|
|
.filter(x => x.length > 0)
|
|
|
|
.map(x => x.toLowerCase()) as ChangeStatus[],
|
|
|
|
matcher: new minimatch.Minimatch(pattern, MinimatchOptions)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
this.throwInvalidFormatError(`Unexpected element type '${typeof item}'`)
|
|
|
|
}
|
|
|
|
|
|
|
|
private throwInvalidFormatError(message: string): never {
|
|
|
|
throw new Error(`Invalid filter YAML format: ${message}.`)
|
2020-05-20 22:31:16 +00:00
|
|
|
}
|
|
|
|
}
|
2020-06-19 21:39:06 +00:00
|
|
|
|
2020-07-11 15:17:56 +00:00
|
|
|
// Creates a new array with all sub-array elements concatenated
|
2020-06-19 21:39:06 +00:00
|
|
|
// In future could be replaced by Array.prototype.flat (supported on Node.js 11+)
|
2020-07-11 15:17:56 +00:00
|
|
|
function flat<T>(arr: T[][]): T[] {
|
|
|
|
return arr.reduce((acc, val) => acc.concat(val), [])
|
2020-06-19 21:39:06 +00:00
|
|
|
}
|