2020-04-24 19:29:53 -04:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2020-04-24 19:29:53 -04:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2021-06-03 09:59:20 -04:00
|
|
|
import {GitClient} from '../../utils/git/git-client';
|
2020-04-24 19:29:53 -04:00
|
|
|
import {FormatConfig} from '../config';
|
|
|
|
|
|
|
|
// A callback to determine if the formatter run found a failure in formatting.
|
2021-05-27 15:04:39 -04:00
|
|
|
export type CallbackFunc =
|
|
|
|
(file: string, code: number|NodeJS.Signals, stdout: string, stderr: string) => boolean;
|
2020-04-24 19:29:53 -04:00
|
|
|
|
|
|
|
// The actions a formatter can take.
|
|
|
|
export type FormatterAction = 'check'|'format';
|
|
|
|
|
|
|
|
// The metadata needed for running one of the `FormatterAction`s on a file.
|
|
|
|
interface FormatterActionMetadata {
|
|
|
|
commandFlags: string;
|
|
|
|
callback: CallbackFunc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The base class for formatters to run against provided files.
|
|
|
|
*/
|
|
|
|
export abstract class Formatter {
|
2021-06-03 09:59:20 -04:00
|
|
|
protected git = GitClient.get();
|
2020-04-24 19:29:53 -04:00
|
|
|
/**
|
|
|
|
* The name of the formatter, this is used for identification in logging and for enabling and
|
|
|
|
* configuring the formatter in the config.
|
|
|
|
*/
|
|
|
|
abstract name: string;
|
|
|
|
|
|
|
|
/** The full path file location of the formatter binary. */
|
|
|
|
abstract binaryFilePath: string;
|
|
|
|
|
|
|
|
/** Metadata for each `FormatterAction` available to the formatter. */
|
|
|
|
abstract actions: {
|
|
|
|
// An action performing a check of format without making any changes.
|
|
|
|
check: FormatterActionMetadata;
|
|
|
|
// An action to format files in place.
|
|
|
|
format: FormatterActionMetadata;
|
|
|
|
};
|
|
|
|
|
|
|
|
/** The default matchers for the formatter for filtering files to be formatted. */
|
|
|
|
abstract defaultFileMatcher: string[];
|
|
|
|
|
2021-04-26 15:45:02 -04:00
|
|
|
constructor(protected config: FormatConfig) {}
|
2020-04-24 19:29:53 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the command to execute the provided action, including both the binary
|
|
|
|
* and command line flags.
|
|
|
|
*/
|
|
|
|
commandFor(action: FormatterAction) {
|
|
|
|
switch (action) {
|
|
|
|
case 'check':
|
|
|
|
return `${this.binaryFilePath} ${this.actions.check.commandFlags}`;
|
|
|
|
case 'format':
|
|
|
|
return `${this.binaryFilePath} ${this.actions.format.commandFlags}`;
|
|
|
|
default:
|
|
|
|
throw Error('Unknown action type');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the callback for the provided action to determine if an action
|
|
|
|
* failed in formatting.
|
|
|
|
*/
|
|
|
|
callbackFor(action: FormatterAction) {
|
|
|
|
switch (action) {
|
|
|
|
case 'check':
|
|
|
|
return this.actions.check.callback;
|
|
|
|
case 'format':
|
|
|
|
return this.actions.format.callback;
|
|
|
|
default:
|
|
|
|
throw Error('Unknown action type');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Whether the formatter is enabled in the provided config. */
|
|
|
|
isEnabled() {
|
|
|
|
return !!this.config[this.name];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Retrieve the active file matcher for the formatter. */
|
|
|
|
getFileMatcher() {
|
|
|
|
return this.getFileMatcherFromConfig() || this.defaultFileMatcher;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieves the file matcher from the config provided to the constructor if provided.
|
|
|
|
*/
|
|
|
|
private getFileMatcherFromConfig(): string[]|undefined {
|
|
|
|
const formatterConfig = this.config[this.name];
|
|
|
|
if (typeof formatterConfig === 'boolean') {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
return formatterConfig.matchers;
|
|
|
|
}
|
|
|
|
}
|