Skip to content

Commit 2b48234

Browse files
Update architecture docs
1 parent 73e9042 commit 2b48234

File tree

9 files changed

+92
-69
lines changed

9 files changed

+92
-69
lines changed

docs/Architecture/Editors.md

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
# Editors
2+
3+
Editor lint configurations are converted by `src/converters/editorConfigs/convertEditorConfig.ts`.
4+
Any setting that matches a known built-in TSLint setting will be replaced with the ESLint equivalent.
5+
6+
For now, only VS Code editor settings are accounted for.
7+
Eventually this will be refactored to allow other editors such as Atom.
8+
9+
1. An existing editor configuration is read from disk.
10+
2. If the existing configuration is not found or errored, nothing else needs to be done.
11+
3. Configuration settings are converted to their ESLint equivalents.
12+
4. Those ESLint equivalents are written to the configuration file.
13+
5. Results from converting are reported to the user.

docs/Architecture/Linters.md

+24-11
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,26 @@
11
# Linters
22

3-
3+
TSLint-to-ESLint linter configuration conversion is the first root-level converter run.
44
Within `src/converters/lintConfigs/convertLintConfig.ts`, the following steps occur:
55

6-
1. Existing configurations are read from disk
7-
2. TSLint rules are converted into their ESLint configurations
8-
3. ESLint configurations are summarized based on extended ESLint and TSLint presets
9-
- 3a. If no output rules conflict with `eslint-config-prettier`, it's added in
10-
- 3b. Any ESLint rules that are configured the same as an extended preset are trimmed
11-
4. The summarized configuration is written to the output config file
12-
5. Files to transform comments in have source text rewritten using the cached rule conversion results
13-
6. A summary of the results is printed to the user's console
6+
1. Raw TSLint rules are mapped to their ESLint equivalents.
7+
2. Those ESLint equivalents are deduplicated and relevant preset(s) detected.
8+
3. Those deduplicated rules and metadata are written to the output configuration file.
9+
4. A summary of conversion results is printed, along with any now-missing packages.
10+
11+
## Rule Conversion
1412

15-
### Conversion Results
13+
The parts of linter configurations most users focus on are the rule converters.
14+
Those are run by `src/converters/lintConfigs/rules/convertRules.ts`, which takes the following steps on each original TSLint rule:
1615

17-
The overall configuration generated by steps 2-3 and printed in 4-5 contains the following information:
16+
1. The raw TSLint rule is converted to a standardized format.
17+
2. The appropriate converter is run for the rule.
18+
3. If the rule is missing or the conversion failed, this is marked.
19+
4. For each output rule equivalent given by the conversion:
20+
* The output rule name is added to the TSLint rule's equivalency set.
21+
* The TSLint rule's config severity is mapped to its ESLint equivalent.
22+
* If this is the first time the output ESLint rule is seen, it's directly marked as converted.
23+
* If not, a rule merger is run to combine it with its existing output settings.
1824

1925
### Rule Converters
2026

@@ -42,3 +48,10 @@ It's possible that one ESLint rule will be output by multiple converters.
4248
These are located in `src/rules/mergers/`, and keyed under their names by the map in `src/rules/mergers.ts`.
4349

4450
For example, `@typescript-eslint/ban-types` spreads both arguments' `types` members into one large `types` object.
51+
52+
## Package Summaries
53+
54+
ESLint configurations are summarized based on extended ESLint and TSLint presets.
55+
56+
- If no output rules conflict with `eslint-config-prettier`, it's added in.
57+
- Any ESLint rules that are configured the same as an extended preset are trimmed.

docs/Architecture/README.md

+16-47
Original file line numberDiff line numberDiff line change
@@ -1,57 +1,26 @@
11
# Architecture
22

3-
## CLI Architecture
3+
tslint-to-eslint-config is a heavily tested CLI app that uses rudimentary [dependency injection](https://wikipedia.org/wiki/Dependency_injection) to link functions together.
44

5-
1. CLI usage starts in `bin/tslint-to-eslint-config`, which immediately calls `src/cli/main.ts`.
6-
2. CLI settings are parsed and read in `src/cli/runCli.ts`.
7-
3. Linter configuration conversion is run by `src/conversion/convertLintConfig.ts`.
8-
4. Editor configuration conversion is run by `src/conversion/convertEditorConfig.ts`.
5+
The initial application flow starts in `bin/tslint-to-eslint-config`:
96

10-
## Linter Configuration Conversion
7+
1. `src/cli/main.ts`'s `main` is called with `process.argv` as the program arguments.
8+
* This file sets up all function dependencies and bound functions that call to each other.
9+
2. `src/cli/runCli.ts`'s `runCli` is called with those bound dependencies and raw arguments.
1110

12-
Within `src/conversion/convertLintConfig.ts`, the following steps occur:
11+
> See [Dependencies.md](./Dependencies.md) for more info on dependency bindings.
1312
14-
1. Existing configurations are read from disk
15-
2. TSLint rules are converted into their ESLint configurations
16-
3. ESLint configurations are summarized based on extended ESLint and TSLint presets
17-
- 3a. If no output rules conflict with `eslint-config-prettier`, it's added in
18-
- 3b. Any ESLint rules that are configured the same as an extended preset are trimmed
19-
4. The summarized configuration is written to the output config file
20-
5. Files to transform comments in have source text rewritten using the cached rule conversion results
21-
6. A summary of the results is printed to the user's console
13+
## CLI Runner
2214

23-
### Conversion Results
15+
1. CLI options are parsed from the raw arguments into a commands object.
16+
2. If the version should be printed, we do that and stop execution.
17+
3. Any existing linter and TypeScript configurations are read from disk.
18+
4. Each converter is run, halting execution if it fails.
2419

25-
The overall configuration generated by steps 2-3 and printed in 4-5 contains the following information:
20+
## Converters
2621

27-
### Rule Converters
22+
Within that flow, there are three "root-level" converters directly called by `runCli`, in order:
2823

29-
Each TSLint rule should output at least one ESLint rule as the equivalent.
30-
"Converters" for TSLint rules are located in `src/rules/converters/`, and keyed under their names by the map in `src/rules/converters.ts`.
31-
32-
Each converter for a TSLint rule takes an arguments object for the rule, and returns an array of objects containing:
33-
34-
- `rules`: At least one equivalent ESLint rule and options
35-
- `notices`: Any extra info that should be printed after conversion
36-
- `plugins`: Any plugins that should now be installed if not already
37-
38-
The `rules` output is an array of objects containing:
39-
40-
- `ruleName`: Equivalent ESLint rule name that should be enabled
41-
- `ruleArguments`: Any arguments for that ESLint rule
42-
43-
Multiple objects must be supported because some general-use TSLint rules can only be represented by two or more ESLint rules.
44-
For example, TSLint's `no-banned-terms` is represented by ESLint's `no-caller` and `no-eval`.
45-
46-
### Rule Mergers
47-
48-
It's possible that one ESLint rule will be output by multiple converters.
49-
"Mergers" for those ESLint rules should take in two configurations to the same rule and output the equivalent single configuration.
50-
These are located in `src/rules/mergers/`, and keyed under their names by the map in `src/rules/mergers.ts`.
51-
52-
For example, `@typescript-eslint/ban-types` spreads both arguments' `types` members into one large `types` object.
53-
54-
## Editor Configuration Conversion
55-
56-
Editor lint configurations are converted by `src/editorSettings/convertEditorSettings.ts`.
57-
Any setting that matches a known built-in TSLint setting will be replaced with the ESLint equivalent.
24+
1. **[Linters.md](./Linters.md)**: Converting from an original TSLint configuration to the equivalent TSLint configuration.
25+
2. **[Editors.md](./Editors.md)**: Creating new IDE settings for ESLint equivalent to any existing TSLint settings.
26+
3. **[Comments.md](./Comments.md)**: Converting inline `tslint:disable` lint disable comments to their `eslint-disable` equivalents.

src/cli/runCli.ts

+7
Original file line numberDiff line numberDiff line change
@@ -19,10 +19,14 @@ export type RunCliDependencies = {
1919
logger: Logger;
2020
};
2121

22+
/**
23+
* @see `/docs/Architecture/README.md` for documentation.
24+
*/
2225
export const runCli = async (
2326
dependencies: RunCliDependencies,
2427
rawArgv: string[],
2528
): Promise<ResultStatus> => {
29+
// 1. CLI options are parsed from the raw arguments into a commands object.
2630
const command = new Command()
2731
.storeOptionsAsProperties(false)
2832
.usage("[options] <file ...> --language [language]")
@@ -41,17 +45,20 @@ export const runCli = async (
4145
...command.parse(rawArgv).opts(),
4246
} as TSLintToESLintSettings;
4347

48+
// 2. If the version should be printed, we do that and stop execution.
4449
if (command.opts().version) {
4550
dependencies.logger.stdout.write(`${version}${EOL}`);
4651
return ResultStatus.Succeeded;
4752
}
4853

54+
// 3. Any existing linter and TypeScript configurations are read from disk.
4955
const originalConfigurations = await dependencies.findOriginalConfigurations(parsedArgv);
5056
if (originalConfigurations.status !== ResultStatus.Succeeded) {
5157
logErrorResult(originalConfigurations, dependencies.logger);
5258
return originalConfigurations.status;
5359
}
5460

61+
// 4. Each converter is run, halting execution if it fails.
5562
for (const converter of dependencies.converters) {
5663
const result = await tryConvertConfig(converter, parsedArgv, originalConfigurations.data);
5764
if (result.status !== ResultStatus.Succeeded) {

src/converters/editorConfigs/convertEditorConfig.ts

+13-8
Original file line numberDiff line numberDiff line change
@@ -22,29 +22,33 @@ export const convertEditorConfig = async (
2222
dependencies: ConvertEditorConfigDependencies,
2323
settings: TSLintToESLintSettings,
2424
): Promise<ResultWithStatus> => {
25-
const conversion = await dependencies.findEditorConfiguration(settings.editor);
26-
if (conversion === undefined) {
25+
// 1. An existing editor configuration is read from disk.
26+
const configuration = await dependencies.findEditorConfiguration(settings.editor);
27+
28+
// 2. If the existing configuration is not found or errored, nothing else needs to be done.
29+
if (configuration === undefined) {
2730
return {
2831
status: ResultStatus.Succeeded,
2932
};
3033
}
31-
32-
if (conversion.result instanceof Error) {
34+
if (configuration.result instanceof Error) {
3335
return {
34-
errors: [conversion.result],
36+
errors: [configuration.result],
3537
status: ResultStatus.Failed,
3638
};
3739
}
3840

41+
// 3. Configuration settings are converted to their ESLint equivalents.
3942
const settingConversionResults = dependencies.convertEditorSettings(
40-
conversion.result,
43+
configuration.result,
4144
settings,
4245
);
4346

47+
// 4. Those ESLint equivalents are written to the configuration file.
4448
const fileWriteError = await dependencies.writeEditorConfigConversionResults(
45-
conversion.configPath,
49+
configuration.configPath,
4650
settingConversionResults,
47-
conversion.result,
51+
configuration.result,
4852
);
4953
if (fileWriteError !== undefined) {
5054
return {
@@ -53,6 +57,7 @@ export const convertEditorConfig = async (
5357
};
5458
}
5559

60+
// 5. Results from converting are reported to the user.
5661
dependencies.reportEditorSettingConversionResults(settingConversionResults);
5762

5863
return {

src/converters/lintConfigs/convertLintConfig.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -25,18 +25,21 @@ export const convertLintConfig = async (
2525
originalConfigurations: AllOriginalConfigurations,
2626
ruleEquivalents: Map<string, string[]>,
2727
): Promise<ResultWithStatus> => {
28+
// 1. Raw TSLint rules are mapped to their ESLint equivalents.
2829
const ruleConversionResults = dependencies.convertRules(
2930
originalConfigurations.tslint.full.rules,
3031
ruleEquivalents,
3132
);
3233

34+
// 2. Those ESLint equivalents are deduplicated and relevant preset(s) detected.
3335
const summarizedConfiguration = await dependencies.summarizePackageRules(
3436
originalConfigurations.eslint,
3537
originalConfigurations.tslint,
3638
ruleConversionResults,
3739
settings.prettier,
3840
);
3941

42+
// 3. Those deduplicated rules and metadata are written to the output configuration file.
4043
const fileWriteError = await dependencies.writeConfigConversionResults(
4144
settings.config,
4245
summarizedConfiguration,
@@ -49,8 +52,8 @@ export const convertLintConfig = async (
4952
};
5053
}
5154

55+
// 4. A summary of conversion results is printed, along with any now-missing packages.
5256
await dependencies.reportConfigConversionResults(settings.config, summarizedConfiguration);
53-
5457
await dependencies.logMissingPackages(summarizedConfiguration, originalConfigurations.packages);
5558

5659
return {

src/converters/lintConfigs/rules/convertRules.ts

+13
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,10 @@ export type RuleConversionResults = {
2121
ruleEquivalents: Map<string, string[]>;
2222
};
2323

24+
/**
25+
* Converts raw TSLint rules to their ESLint equivalents.
26+
* @see `/docs/Architecture/Linting.md` for documentation.
27+
*/
2428
export const convertRules = (
2529
dependencies: ConvertRulesDependencies,
2630
rawTslintRules: TSLintConfigurationRules | undefined,
@@ -33,9 +37,13 @@ export const convertRules = (
3337

3438
if (rawTslintRules !== undefined) {
3539
for (const [ruleName, value] of Object.entries(rawTslintRules)) {
40+
// 1. The raw TSLint rule is converted to a standardized format.
3641
const tslintRule = formatRawTslintRule(ruleName, value);
42+
43+
// 2. The appropriate converter is run for the rule.
3744
const conversion = convertRule(tslintRule, dependencies.ruleConverters);
3845

46+
// 3. If the rule is missing or the conversion failed, this is marked.
3947
if (conversion === undefined) {
4048
if (tslintRule.ruleSeverity !== "off") {
4149
missing.push(tslintRule);
@@ -51,20 +59,25 @@ export const convertRules = (
5159

5260
const equivalents = new Set<string>();
5361

62+
// 4. For each output rule equivalent given by the conversion:
5463
for (const changes of conversion.rules) {
64+
// 4a. The output rule name is added to the TSLint rule's equivalency set.
5565
equivalents.add(changes.ruleName);
5666

67+
// 4b. The TSLint rule's config severity is mapped to its ESLint equivalent.
5768
const existingConversion = converted.get(changes.ruleName);
5869
const newConversion = {
5970
...changes,
6071
ruleSeverity: convertTSLintRuleSeverity(tslintRule.ruleSeverity),
6172
};
6273

74+
// 4c. If this is the first time the output ESLint rule is seen, it's directly marked as converted.
6375
if (existingConversion === undefined) {
6476
converted.set(changes.ruleName, newConversion);
6577
continue;
6678
}
6779

80+
// 4d. If not, a rule merger is run to combine it with its existing output settings.
6881
const merger = dependencies.ruleMergers.get(changes.ruleName);
6982
if (merger === undefined) {
7083
failed.push(ConversionError.forMerger(changes.ruleName));

src/converters/lintConfigs/rules/ruleConverters/no-banned-terms.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import { RuleConverter } from "../ruleConverter";
22

33
export const convertNoBannedTerms: RuleConverter = () => {
44
return {
5-
// This is mentioned in Architecture.md as a TSLint rule with two ESLint equivalents
5+
// This is mentioned in Architecture/Linters.md as a TSLint rule with two ESLint equivalents
66
rules: [
77
{
88
ruleName: "no-caller",

src/converters/lintConfigs/rules/ruleMergers/ban-types.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ export const mergeBanTypes: RuleMerger = (existingOptions, newOptions) => {
55
return [];
66
}
77

8-
// This is mentioned in Architecture.md as an ESLint rule with a merger
8+
// This is mentioned in Architecture/Linters.md as an ESLint rule with a merger
99
return [
1010
{
1111
types: {

0 commit comments

Comments
 (0)