- Add
PackageContextForTest
topackage:codemod/test.dart
to help test suggestors that require a fully resolved AST from the analyzer (for example: suggestors using theAstVisitingSuggestor
mixin withshouldResolveAst
enabled).
- Compatibility with Dart 3 and analyzer 6.
- Widen analyzer dependency range to include v3, v4, and v5.
- Update analyzer dependency to v2
- Switch to replacements for deprecated Dart CLIs
- Fix wildcard in GitHub CI
- Raise Dart SDK minimum to 2.11.0
- Include file path in error message when parsing a Dart file fails.
- Null-safety release.
AstVisitingSuggestor.context
will throw aStateError
if accessed outside one of the visitor methods.
- Fix invalid file path error on windows when applying patches.
-
Breaking:
runInteractiveCodemod
andrunInteractiveCodemodSequence
are both now async. -
Breaking:
Suggestor
is now a function typedef instead of a class. Addtionally, it takes the newFileContext
type as its only parameter (previously thegeneratePatches
method 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
Patch
class 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
AstVisitingSuggestorMixin
toAstVisitingSuggestor
since theMixin
suffix was redundant. -
Breaking: Remove
AggregateSuggestor
class in favor of anaggregate(Iterable<Suggestor> suggestors)
function. -
Breaking: Move
applyPatches
from the mainpackage:codemod/codemod.dart
entrypoint to the newpackage:codemod/test.dart
entrypoint 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.dart
entrypoint 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).
-
Breaking Change: remove the
FileQuery
class. TherunInteractiveCodemod()
function now expects anIterable<File>
instead of aFileQuery
. This is intended to simplify consumption;package:glob
can be used to easily query for the desired files. -
Breaking Change: remove
createPathFilter()
- this was used withFileQuery
and can be replaced by custom logic now thatrunInteractiveCodemod()
accepts anIterable<File>
. -
Breaking Change: remove
isDartFile()
- usepackage:glob
instead to target files with the.dart
extension (e.g.Glob('**.dart')
). -
Add a
filePathsFromGlob()
utility function that takes aGlob
instance 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
Patch
overrideoperator ==
andhashCode
so that instances can be compared for equality. If twoPatch
instances target the same span in the same file and have the sameupdatedText
, they are considered equal. -
AstVisitingSuggestorMixin
now 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 multiplevisit
methods. -
Improve the error/help output when overlapping patches are found.
-
Widen Analyzer dependency range from
^0.37.0
to>=0.37.0 <0.39.0
. -
Exclude
build/
folder when a codemod gets run.
- Prompts the user to either skip overlapping patches or quit when they are found.
-
Codemod authors can now augment the help output and the changes required output via
runInteractiveCodemod()
andrunInteractiveCodemodSequence()
using the optionaladditionalHelpOutput
andchangesRequiredOutput
params.-
If
additionalHelpOutput
is given, it will be printed to stderr after the default help output when the codemod is run with the-h|--help
flag. -
If
changesRequiredOutput
is given, it will be printed to stderr after the default output when the codemod is run with the--fail-on-changes
flag and changes are in fact required.
-
-
Fix a typing issue with the
AggregateSuggestor
's constructor param. -
Add tests for
AggregateSuggestor
andAstVisitingSuggestorMixin
- Update
pubspec.yaml
for initial OSS release.
- Initial tag.