codemod 1.2.0
codemod: ^1.2.0 copied to clipboard
Write and run automated code modifications on a codebase. Primarily geared towards updating and refactoring Dart code, but can modify any files.
1.2.0 #
- Add
PackageContextForTesttopackage:codemod/test.dartto help test suggestors that require a fully resolved AST from the analyzer (for example: suggestors using theAstVisitingSuggestormixin withshouldResolveAstenabled).
1.1.0 #
- Compatibility with Dart 3 and analyzer 6.
1.0.11 #
- Widen analyzer dependency range to include v3, v4, and v5.
1.0.10 #
- Update analyzer dependency to v2
1.0.4 #
- Switch to replacements for deprecated Dart CLIs
1.0.3 #
- Fix wildcard in GitHub CI
1.0.2 #
- Raise Dart SDK minimum to 2.11.0
1.0.1 #
- Include file path in error message when parsing a Dart file fails.
1.0.0 #
- Null-safety release.
AstVisitingSuggestor.contextwill throw aStateErrorif accessed outside one of the visitor methods.
0.3.1 #
- Fix invalid file path error on windows when applying patches.
0.3.0 #
-
Breaking:
runInteractiveCodemodandrunInteractiveCodemodSequenceare both now async. -
Breaking:
Suggestoris now a function typedef instead of a class. Addtionally, it takes the newFileContexttype as its only parameter (previously thegeneratePatchesmethod took aSourceFile) and now must return aStream<Patch>instead ofIterable<Patch>.If you were extending
Suggestor, you can either change the class to a function like so:final String licenseHeader = '...'; - class LicenseHeaderInserter implements Suggestor { - @override - bool shouldSkip(String sourceFileContents) => - sourceFileContents.trimLeft().startsWith(licenseHeader); - - @override - Iterable<Patch> generatePatches(SourceFile sourceFile) sync* { - yield Patch(sourceFile, sourceFile.span(0, 0), licenseHeader); - } - } + Stream<Patch> licenseHeaderInserter(FileContext context) async* { + // Skip if license header already exists. + if (context.sourceText.trimLeft().startsWith(licenseHeader)) return; + + yield Patch(licenseHeader, 0, 0); + }Or rename the
generatePatches()method tocall()so that the class is callable like a function:final String licenseHeader = '...'; - class LicenseHeaderInserter implements Suggestor { + class LicenseHeaderInserter { - @override bool shouldSkip(String sourceFileContents) => sourceFileContents.trimLeft().startsWith(licenseHeader); - @override - Iterable<Patch> generatePatches(SourceFile sourceFile) sync* { + Stream<Patch> call(FileContext context) async* { + if (shouldSkip(context.sourceText)) return; - yield Patch(sourceFile, sourceFile.span(0, 0), licenseHeader); + yield Patch(licenseHeader, 0, 0); } } -
Breaking: Simplify the
Patchclass to now only encapsulate the updated text, start offset, and end offset.- yield Patch(sourceFile, sourceFile.span(5, 10), 'updated text'); + yield Patch('updated text', 5, 10); -
Breaking: Rename
AstVisitingSuggestorMixintoAstVisitingSuggestorsince theMixinsuffix was redundant. -
Breaking: Remove
AggregateSuggestorclass in favor of anaggregate(Iterable<Suggestor> suggestors)function. -
Breaking: Move
applyPatchesfrom the mainpackage:codemod/codemod.dartentrypoint to the newpackage:codemod/test.dartentrypoint to make its intended usage clear. -
Add a
shouldResolveAst(FileContext context)method toAstVisitingSuggestor. Defaults to false (since resolving is slower), but can be overridden to true if the fully resolved AST is needed.- Add example of such a codemod: see example/is_even_or_odd_suggestor.dart
-
Add
package:codemod/test.dartentrypoint for testing suggestors. This entrypoint exports three functions:-
Future<FileContext> fileContextForTest(String name, String contents) -
void expectSuggestorGeneratesPatches(Suggestor suggestor, FileContext context, dynamic resultMatcher) -
String applyPatches(SourceFile sourceFile, Iterable<Patch> patches)The first two should be sufficient for testing most suggestors.
-
-
Use GitHub Actions for CI (remove Travis CI).
0.2.0 #
-
Breaking Change: remove the
FileQueryclass. TherunInteractiveCodemod()function now expects anIterable<File>instead of aFileQuery. This is intended to simplify consumption;package:globcan be used to easily query for the desired files. -
Breaking Change: remove
createPathFilter()- this was used withFileQueryand can be replaced by custom logic now thatrunInteractiveCodemod()accepts anIterable<File>. -
Breaking Change: remove
isDartFile()- usepackage:globinstead to target files with the.dartextension (e.g.Glob('**.dart')). -
Add a
filePathsFromGlob()utility function that takes aGlobinstance and returns the file paths matched by the glob, but filtered to exclude hidden files (e.g. files in.dart_tool/). This is intended to serve the most common use case of running codemods on Dart projects, e.g.:runInteractiveCodemod( filePathsFromGlob(Glob('**.dart', recursive: true)), suggestor);
-
Widen Analyzer dependency range to include
0.39.x. -
Make
Patchoverrideoperator ==andhashCodeso that instances can be compared for equality. If twoPatchinstances target the same span in the same file and have the sameupdatedText, they are considered equal. -
AstVisitingSuggestorMixinnow de-duplicates patches suggested for each source file. This can be useful for recursive and generalizing AST visitors that may end up suggesting duplicate patches in parts of the AST that get handled by multiplevisitmethods. -
Improve the error/help output when overlapping patches are found.
0.1.5 #
-
Widen Analyzer dependency range from
^0.37.0to>=0.37.0 <0.39.0. -
Exclude
build/folder when a codemod gets run.
0.1.4 #
- Prompts the user to either skip overlapping patches or quit when they are found.
0.1.3 #
-
Codemod authors can now augment the help output and the changes required output via
runInteractiveCodemod()andrunInteractiveCodemodSequence()using the optionaladditionalHelpOutputandchangesRequiredOutputparams.-
If
additionalHelpOutputis given, it will be printed to stderr after the default help output when the codemod is run with the-h|--helpflag. -
If
changesRequiredOutputis given, it will be printed to stderr after the default output when the codemod is run with the--fail-on-changesflag and changes are in fact required.
-
0.1.2 #
-
Fix a typing issue with the
AggregateSuggestor's constructor param. -
Add tests for
AggregateSuggestorandAstVisitingSuggestorMixin
0.1.1 #
- Update
pubspec.yamlfor initial OSS release.
0.1.0 #
- Initial tag.