Unverified Commit f5a98353 authored by Greg Spencer's avatar Greg Spencer Committed by GitHub

Check sample links for malformed links (#137807)

## Description

This checks API doc strings for malformed links to examples. It prevents errors in capitalization, spacing, number of asterisks, etc.  It won't catch all errors, because it needs to have a minimally indicative string to know that it even is trying to be a link to an example.  At a minimum, the line needs to look like (literally, not as a regexp) `///*seecode.*` in order to be seen as a link to an example.

Separately, I'm going to add a check to the snippets tool that checks to make sure that an `{@tool}` block includes either a link to a sample file or a dart code block.

## Tests
 - Added a test to make sure it catches some malformed links.
parent defa4bce
...@@ -97,6 +97,19 @@ void main(List<String> args) { ...@@ -97,6 +97,19 @@ void main(List<String> args) {
reportSuccessAndExit('All examples are linked and have tests.'); reportSuccessAndExit('All examples are linked and have tests.');
} }
class LinkInfo {
const LinkInfo(this.link, this.file, this.line);
final String link;
final File file;
final int line;
@override
String toString() {
return '${file.path}:$line: $link';
}
}
class SampleChecker { class SampleChecker {
SampleChecker({ SampleChecker({
required this.examples, required this.examples,
...@@ -119,10 +132,12 @@ class SampleChecker { ...@@ -119,10 +132,12 @@ class SampleChecker {
final List<File> exampleFilenames = getExampleFilenames(examples); final List<File> exampleFilenames = getExampleFilenames(examples);
// Get a list of all the example link paths that appear in the source files. // Get a list of all the example link paths that appear in the source files.
final Set<String> exampleLinks = getExampleLinks(packages); final (Set<String> exampleLinks, Set<LinkInfo> malformedLinks) = getExampleLinks(packages);
// Also add in any that might be found in the dart:ui directory. // Also add in any that might be found in the dart:ui directory.
exampleLinks.addAll(getExampleLinks(dartUIPath)); final (Set<String> uiExampleLinks, Set<LinkInfo> uiMalformedLinks) = getExampleLinks(dartUIPath);
exampleLinks.addAll(uiExampleLinks);
malformedLinks.addAll(uiMalformedLinks);
// Get a list of the filenames that were not found in the source files. // Get a list of the filenames that were not found in the source files.
final List<String> missingFilenames = checkForMissingLinks(exampleFilenames, exampleLinks); final List<String> missingFilenames = checkForMissingLinks(exampleFilenames, exampleLinks);
...@@ -136,7 +151,7 @@ class SampleChecker { ...@@ -136,7 +151,7 @@ class SampleChecker {
// generate new examples. // generate new examples.
missingFilenames.removeWhere((String file) => _knownUnlinkedExamples.contains(file)); missingFilenames.removeWhere((String file) => _knownUnlinkedExamples.contains(file));
if (missingFilenames.isEmpty && missingTests.isEmpty && noLongerMissing.isEmpty) { if (missingFilenames.isEmpty && missingTests.isEmpty && noLongerMissing.isEmpty && malformedLinks.isEmpty) {
return true; return true;
} }
...@@ -167,6 +182,19 @@ class SampleChecker { ...@@ -167,6 +182,19 @@ class SampleChecker {
buffer.write('Either link them to a source file API doc comment, or remove them.'); buffer.write('Either link them to a source file API doc comment, or remove them.');
foundError(buffer.toString().split('\n')); foundError(buffer.toString().split('\n'));
} }
if (malformedLinks.isNotEmpty) {
final StringBuffer buffer =
StringBuffer('The following malformed links were found in API doc comments:\n');
for (final LinkInfo link in malformedLinks) {
buffer.writeln(' $link');
}
buffer.write(
'Correct the formatting of these links so that they match the exact pattern:\n'
r" r'\*\* See code in (?<path>.+) \*\*'"
);
foundError(buffer.toString().split('\n'));
}
return false; return false;
} }
...@@ -199,21 +227,34 @@ class SampleChecker { ...@@ -199,21 +227,34 @@ class SampleChecker {
); );
} }
Set<String> getExampleLinks(Directory searchDirectory) { (Set<String>, Set<LinkInfo>) getExampleLinks(Directory searchDirectory) {
final List<File> files = getFiles(searchDirectory, RegExp(r'\.dart$')); final List<File> files = getFiles(searchDirectory, RegExp(r'\.dart$'));
final Set<String> searchStrings = <String>{}; final Set<String> searchStrings = <String>{};
final RegExp exampleRe = RegExp(r'\*\* See code in (?<path>.*) \*\*'); final Set<LinkInfo> malformedStrings = <LinkInfo>{};
final RegExp validExampleRe = RegExp(r'\*\* See code in (?<path>.+) \*\*');
// Looks for some common broken versions of example links. This looks for
// something that is at minimum "///*seecode<something>*" to indicate that it
// looks like an example link. It should be narrowed if we start gettting false
// positives.
final RegExp malformedLinkRe = RegExp(r'^(?<malformed>\s*///\s*\*\*?\s*[sS][eE][eE]\s*[Cc][Oo][Dd][Ee].+\*\*?)');
for (final File file in files) { for (final File file in files) {
final String contents = file.readAsStringSync(); final String contents = file.readAsStringSync();
searchStrings.addAll( final List<String> lines = contents.split('\n');
contents.split('\n').where((String s) => s.contains(exampleRe)).map<String>( int count = 0;
(String e) { for (final String line in lines) {
return exampleRe.firstMatch(e)!.namedGroup('path')!; count += 1;
}, final RegExpMatch? validMatch = validExampleRe.firstMatch(line);
), if (validMatch != null) {
); searchStrings.add(validMatch.namedGroup('path')!);
}
final RegExpMatch? malformedMatch = malformedLinkRe.firstMatch(line);
// It's only malformed if it doesn't match the valid RegExp.
if (malformedMatch != null && validMatch == null) {
malformedStrings.add(LinkInfo(malformedMatch.namedGroup('malformed')!, file, count));
}
}
} }
return searchStrings; return (searchStrings, malformedStrings);
} }
List<String> checkForMissingLinks(List<File> exampleFilenames, Set<String> searchStrings) { List<String> checkForMissingLinks(List<File> exampleFilenames, Set<String> searchStrings) {
......
...@@ -25,7 +25,8 @@ void main() { ...@@ -25,7 +25,8 @@ void main() {
return path.relative(file.absolute.path, from: flutterRoot.absolute.path); return path.relative(file.absolute.path, from: flutterRoot.absolute.path);
} }
void writeLink({required File source, required File example}) { void writeLink({required File source, required File example, String? alternateLink}) {
final String link = alternateLink ?? ' ** See code in ${getRelativePath(example)} **';
source source
..createSync(recursive: true) ..createSync(recursive: true)
..writeAsStringSync(''' ..writeAsStringSync('''
...@@ -34,12 +35,12 @@ void main() { ...@@ -34,12 +35,12 @@ void main() {
/// {@tool dartpad} /// {@tool dartpad}
/// Example description /// Example description
/// ///
/// ** See code in ${getRelativePath(example)} ** ///$link
/// {@end-tool} /// {@end-tool}
'''); ''');
} }
void buildTestFiles({bool missingLinks = false, bool missingTests = false}) { void buildTestFiles({bool missingLinks = false, bool missingTests = false, bool malformedLinks = false}) {
final Directory examplesLib = examples.childDirectory('lib').childDirectory('layer')..createSync(recursive: true); final Directory examplesLib = examples.childDirectory('lib').childDirectory('layer')..createSync(recursive: true);
final File fooExample = examplesLib.childFile('foo_example.0.dart') final File fooExample = examplesLib.childFile('foo_example.0.dart')
..createSync(recursive: true) ..createSync(recursive: true)
...@@ -73,9 +74,15 @@ void main() { ...@@ -73,9 +74,15 @@ void main() {
} }
final Directory flutterPackage = packages.childDirectory('flutter').childDirectory('lib').childDirectory('src') final Directory flutterPackage = packages.childDirectory('flutter').childDirectory('lib').childDirectory('src')
..createSync(recursive: true); ..createSync(recursive: true);
writeLink(source: flutterPackage.childDirectory('layer').childFile('foo.dart'), example: fooExample); if (malformedLinks) {
writeLink(source: flutterPackage.childDirectory('layer').childFile('bar.dart'), example: barExample); writeLink(source: flutterPackage.childDirectory('layer').childFile('foo.dart'), example: fooExample, alternateLink: '*See Code *');
writeLink(source: flutterPackage.childDirectory('animation').childFile('curves.dart'), example: curvesExample); writeLink(source: flutterPackage.childDirectory('layer').childFile('bar.dart'), example: barExample, alternateLink: ' ** See code examples/api/lib/layer/bar_example.0.dart **');
writeLink(source: flutterPackage.childDirectory('animation').childFile('curves.dart'), example: curvesExample, alternateLink: '* see code in examples/api/lib/animation/curves/curve2_d.0.dart *');
} else {
writeLink(source: flutterPackage.childDirectory('layer').childFile('foo.dart'), example: fooExample);
writeLink(source: flutterPackage.childDirectory('layer').childFile('bar.dart'), example: barExample);
writeLink(source: flutterPackage.childDirectory('animation').childFile('curves.dart'), example: curvesExample);
}
} }
setUp(() { setUp(() {
...@@ -124,6 +131,43 @@ void main() { ...@@ -124,6 +131,43 @@ void main() {
expect(success, equals(false)); expect(success, equals(false));
}); });
test('check_code_samples.dart - checkCodeSamples catches malformed links', () async {
buildTestFiles(malformedLinks: true);
bool? success;
final String result = await capture(
() async {
success = checker.checkCodeSamples();
},
shouldHaveErrors: true,
);
final bool isWindows = Platform.isWindows;
final String lines = <String>[
'╔═╡ERROR╞═══════════════════════════════════════════════════════════════════════',
'║ The following examples are not linked from any source file API doc comments:',
if (!isWindows) '║ examples/api/lib/animation/curves/curve2_d.0.dart',
if (!isWindows) '║ examples/api/lib/layer/foo_example.0.dart',
if (!isWindows) '║ examples/api/lib/layer/bar_example.0.dart',
if (isWindows) r'║ examples\api\lib\animation\curves\curve2_d.0.dart',
if (isWindows) r'║ examples\api\lib\layer\foo_example.0.dart',
if (isWindows) r'║ examples\api\lib\layer\bar_example.0.dart',
'║ Either link them to a source file API doc comment, or remove them.',
'╚═══════════════════════════════════════════════════════════════════════════════',
'╔═╡ERROR╞═══════════════════════════════════════════════════════════════════════',
'║ The following malformed links were found in API doc comments:',
if (!isWindows) '║ /flutter sdk/packages/flutter/lib/src/animation/curves.dart:6: ///* see code in examples/api/lib/animation/curves/curve2_d.0.dart *',
if (!isWindows) '║ /flutter sdk/packages/flutter/lib/src/layer/foo.dart:6: ///*See Code *',
if (!isWindows) '║ /flutter sdk/packages/flutter/lib/src/layer/bar.dart:6: /// ** See code examples/api/lib/layer/bar_example.0.dart **',
if (isWindows) r'║ C:\flutter sdk\packages\flutter\lib\src\animation\curves.dart:6: ///* see code in examples/api/lib/animation/curves/curve2_d.0.dart *',
if (isWindows) r'║ C:\flutter sdk\packages\flutter\lib\src\layer\foo.dart:6: ///*See Code *',
if (isWindows) r'║ C:\flutter sdk\packages\flutter\lib\src\layer\bar.dart:6: /// ** See code examples/api/lib/layer/bar_example.0.dart **',
'║ Correct the formatting of these links so that they match the exact pattern:',
r"║ r'\*\* See code in (?<path>.+) \*\*'",
'╚═══════════════════════════════════════════════════════════════════════════════',
].join('\n');
expect(result, equals('$lines\n'));
expect(success, equals(false));
});
test('check_code_samples.dart - checkCodeSamples catches missing tests', () async { test('check_code_samples.dart - checkCodeSamples catches missing tests', () async {
buildTestFiles(missingTests: true); buildTestFiles(missingTests: true);
bool? success; bool? success;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment