| // Copyright 2014 The Flutter Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // To run this, from the root of the Flutter repository: |
| // bin/cache/dart-sdk/bin/dart --enable-asserts dev/bots/check_code_samples.dart |
| |
| import 'dart:io'; |
| |
| import 'package:args/args.dart'; |
| import 'package:file/file.dart'; |
| import 'package:file/local.dart'; |
| import 'package:path/path.dart' as path; |
| |
| import 'utils.dart'; |
| |
| final String _scriptLocation = path.fromUri(Platform.script); |
| final String _flutterRoot = path.dirname(path.dirname(path.dirname(_scriptLocation))); |
| final String _exampleDirectoryPath = path.join(_flutterRoot, 'examples', 'api'); |
| final String _packageDirectoryPath = path.join(_flutterRoot, 'packages'); |
| final String _dartUIDirectoryPath = path.join(_flutterRoot, 'bin', 'cache', 'pkg', 'sky_engine', 'lib'); |
| |
| final List<String> _knownUnlinkedExamples = <String>[ |
| // These are template files that aren't expected to be linked. |
| 'examples/api/lib/sample_templates/cupertino.0.dart', |
| 'examples/api/lib/sample_templates/widgets.0.dart', |
| 'examples/api/lib/sample_templates/material.0.dart', |
| ]; |
| |
| void main(List<String> args) { |
| final ArgParser argParser = ArgParser(); |
| argParser.addFlag( |
| 'help', |
| negatable: false, |
| help: 'Print help for this command.', |
| ); |
| argParser.addOption( |
| 'examples', |
| valueHelp: 'path', |
| defaultsTo: _exampleDirectoryPath, |
| help: 'A location where the API doc examples are found.', |
| ); |
| argParser.addOption( |
| 'packages', |
| valueHelp: 'path', |
| defaultsTo: _packageDirectoryPath, |
| help: 'A location where the source code that should link the API doc examples is found.', |
| ); |
| argParser.addOption( |
| 'dart-ui', |
| valueHelp: 'path', |
| defaultsTo: _dartUIDirectoryPath, |
| help: 'A location where the source code that should link the API doc examples is found.', |
| ); |
| argParser.addOption( |
| 'flutter-root', |
| valueHelp: 'path', |
| defaultsTo: _flutterRoot, |
| help: 'The path to the root of the Flutter repo.', |
| ); |
| final ArgResults parsedArgs; |
| |
| void usage() { |
| print('dart --enable-asserts ${path.basename(_scriptLocation)} [options]'); |
| print(argParser.usage); |
| } |
| |
| try { |
| parsedArgs = argParser.parse(args); |
| } on FormatException catch (e) { |
| print(e.message); |
| usage(); |
| exit(1); |
| } |
| |
| if (parsedArgs['help'] as bool) { |
| usage(); |
| exit(0); |
| } |
| |
| const FileSystem filesystem = LocalFileSystem(); |
| final Directory examples = filesystem.directory(parsedArgs['examples']! as String); |
| final Directory packages = filesystem.directory(parsedArgs['packages']! as String); |
| final Directory dartUIPath = filesystem.directory(parsedArgs['dart-ui']! as String); |
| final Directory flutterRoot = filesystem.directory(parsedArgs['flutter-root']! as String); |
| |
| final SampleChecker checker = SampleChecker( |
| examples: examples, |
| packages: packages, |
| dartUIPath: dartUIPath, |
| flutterRoot: flutterRoot, |
| ); |
| |
| if (!checker.checkCodeSamples()) { |
| reportErrorsAndExit('Some errors were found in the API docs code samples.'); |
| } |
| 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 { |
| SampleChecker({ |
| required this.examples, |
| required this.packages, |
| required this.dartUIPath, |
| required this.flutterRoot, |
| this.filesystem = const LocalFileSystem(), |
| }); |
| |
| final Directory examples; |
| final Directory packages; |
| final Directory dartUIPath; |
| final Directory flutterRoot; |
| final FileSystem filesystem; |
| |
| bool checkCodeSamples() { |
| filesystem.currentDirectory = flutterRoot; |
| |
| // Get a list of all the filenames in the source directory that end in "[0-9]+.dart". |
| final List<File> exampleFilenames = getExampleFilenames(examples); |
| |
| // Get a list of all the example link paths that appear in the source files. |
| final (Set<String> exampleLinks, Set<LinkInfo> malformedLinks) = getExampleLinks(packages); |
| // Also add in any that might be found in the dart:ui directory. |
| 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. |
| final List<String> missingFilenames = checkForMissingLinks(exampleFilenames, exampleLinks); |
| |
| // Get a list of any tests that are missing, as well as any that used to be |
| // missing, but have been implemented. |
| final (List<File> missingTests, List<File> noLongerMissing) = checkForMissingTests(exampleFilenames); |
| |
| // Remove any that we know are exceptions (examples that aren't expected to be |
| // linked into any source files). These are typically template files used to |
| // generate new examples. |
| missingFilenames.removeWhere((String file) => _knownUnlinkedExamples.contains(file)); |
| |
| if (missingFilenames.isEmpty && missingTests.isEmpty && noLongerMissing.isEmpty && malformedLinks.isEmpty) { |
| return true; |
| } |
| |
| if (noLongerMissing.isNotEmpty) { |
| final StringBuffer buffer = StringBuffer('The following tests have been implemented! Huzzah!:\n'); |
| for (final File name in noLongerMissing) { |
| buffer.writeln(' ${getRelativePath(name)}'); |
| } |
| buffer.writeln('However, they now need to be removed from the _knownMissingTests'); |
| buffer.write('list in the script $_scriptLocation.'); |
| foundError(buffer.toString().split('\n')); |
| } |
| |
| if (missingTests.isNotEmpty) { |
| final StringBuffer buffer = StringBuffer('The following example test files are missing:\n'); |
| for (final File name in missingTests) { |
| buffer.writeln(' ${getRelativePath(name)}'); |
| } |
| foundError(buffer.toString().trimRight().split('\n')); |
| } |
| |
| if (missingFilenames.isNotEmpty) { |
| final StringBuffer buffer = |
| StringBuffer('The following examples are not linked from any source file API doc comments:\n'); |
| for (final String name in missingFilenames) { |
| buffer.writeln(' $name'); |
| } |
| buffer.write('Either link them to a source file API doc comment, or remove them.'); |
| 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; |
| } |
| |
| String getRelativePath(File file, [Directory? root]) { |
| root ??= flutterRoot; |
| return path.relative(file.absolute.path, from: root.absolute.path); |
| } |
| |
| List<File> getFiles(Directory directory, [Pattern? filenamePattern]) { |
| final List<File> filenames = directory |
| .listSync(recursive: true) |
| .map((FileSystemEntity entity) { |
| if (entity is File) { |
| return entity; |
| } else { |
| return null; |
| } |
| }) |
| .where((File? filename) => |
| filename != null && (filenamePattern == null || filename.absolute.path.contains(filenamePattern))) |
| .map<File>((File? s) => s!) |
| .toList(); |
| return filenames; |
| } |
| |
| List<File> getExampleFilenames(Directory directory) { |
| return getFiles( |
| directory.childDirectory('lib'), |
| RegExp(r'\d+\.dart$'), |
| ); |
| } |
| |
| (Set<String>, Set<LinkInfo>) getExampleLinks(Directory searchDirectory) { |
| final List<File> files = getFiles(searchDirectory, RegExp(r'\.dart$')); |
| final Set<String> searchStrings = <String>{}; |
| 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 getting 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) { |
| final String contents = file.readAsStringSync(); |
| final List<String> lines = contents.split('\n'); |
| int count = 0; |
| for (final String line in lines) { |
| 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, malformedStrings); |
| } |
| |
| List<String> checkForMissingLinks(List<File> exampleFilenames, Set<String> searchStrings) { |
| final List<String> missingFilenames = <String>[]; |
| for (final File example in exampleFilenames) { |
| final String relativePath = getRelativePath(example); |
| if (!searchStrings.contains(relativePath)) { |
| missingFilenames.add(relativePath); |
| } |
| } |
| return missingFilenames; |
| } |
| |
| String getTestNameForExample(File example, Directory examples) { |
| final String testPath = path.dirname( |
| path.join( |
| examples.absolute.path, |
| 'test', |
| getRelativePath(example, examples.childDirectory('lib')), |
| ), |
| ); |
| return '${path.join(testPath, path.basenameWithoutExtension(example.path))}_test.dart'; |
| } |
| |
| (List<File>, List<File>) checkForMissingTests(List<File> exampleFilenames) { |
| final List<File> missingTests = <File>[]; |
| final List<File> noLongerMissingTests = <File>[]; |
| for (final File example in exampleFilenames) { |
| final File testFile = filesystem.file(getTestNameForExample(example, examples)); |
| final String name = path.relative(testFile.absolute.path, from: flutterRoot.absolute.path); |
| if (!testFile.existsSync()) { |
| missingTests.add(testFile); |
| } else if (_knownMissingTests.contains(name.replaceAll(r'\', '/'))) { |
| noLongerMissingTests.add(testFile); |
| } |
| } |
| // Skip any that we know are missing. |
| missingTests.removeWhere( |
| (File test) { |
| final String name = path.relative(test.absolute.path, from: flutterRoot.absolute.path).replaceAll(r'\', '/'); |
| return _knownMissingTests.contains(name); |
| }, |
| ); |
| return (missingTests, noLongerMissingTests); |
| } |
| } |
| |
| // These tests are known to be missing. They should all eventually be |
| // implemented, but until they are we allow them, so that we can catch any new |
| // examples that are added without tests. |
| // |
| // TODO(gspencergoog): implement the missing tests. |
| // See https://github.com/flutter/flutter/issues/130459 |
| final Set<String> _knownMissingTests = <String>{ |
| 'examples/api/test/material/color_scheme/dynamic_content_color.0_test.dart', |
| 'examples/api/test/painting/star_border/star_border.0_test.dart', |
| 'examples/api/test/widgets/navigator/navigator.restorable_push_and_remove_until.0_test.dart', |
| 'examples/api/test/widgets/navigator/navigator.restorable_push.0_test.dart', |
| 'examples/api/test/widgets/navigator/navigator_state.restorable_push_replacement.0_test.dart', |
| 'examples/api/test/widgets/navigator/navigator_state.restorable_push_and_remove_until.0_test.dart', |
| 'examples/api/test/widgets/navigator/navigator.restorable_push_replacement.0_test.dart', |
| 'examples/api/test/widgets/navigator/restorable_route_future.0_test.dart', |
| 'examples/api/test/widgets/navigator/navigator_state.restorable_push.0_test.dart', |
| 'examples/api/test/widgets/focus_manager/focus_node.unfocus.0_test.dart', |
| 'examples/api/test/widgets/framework/build_owner.0_test.dart', |
| 'examples/api/test/widgets/nested_scroll_view/nested_scroll_view_state.0_test.dart', |
| 'examples/api/test/widgets/scroll_position/scroll_metrics_notification.0_test.dart', |
| 'examples/api/test/widgets/media_query/media_query_data.system_gesture_insets.0_test.dart', |
| 'examples/api/test/widgets/image/image.frame_builder.0_test.dart', |
| 'examples/api/test/widgets/image/image.loading_builder.0_test.dart', |
| 'examples/api/test/widgets/page_storage/page_storage.0_test.dart', |
| 'examples/api/test/widgets/scrollbar/raw_scrollbar.1_test.dart', |
| 'examples/api/test/widgets/scrollbar/raw_scrollbar.2_test.dart', |
| 'examples/api/test/widgets/scrollbar/raw_scrollbar.desktop.0_test.dart', |
| 'examples/api/test/widgets/scrollbar/raw_scrollbar.shape.0_test.dart', |
| 'examples/api/test/widgets/scrollbar/raw_scrollbar.0_test.dart', |
| 'examples/api/test/widgets/interactive_viewer/interactive_viewer.constrained.0_test.dart', |
| 'examples/api/test/widgets/interactive_viewer/interactive_viewer.transformation_controller.0_test.dart', |
| 'examples/api/test/widgets/interactive_viewer/interactive_viewer.0_test.dart', |
| 'examples/api/test/widgets/notification_listener/notification.0_test.dart', |
| 'examples/api/test/widgets/overscroll_indicator/glowing_overscroll_indicator.1_test.dart', |
| 'examples/api/test/widgets/overscroll_indicator/glowing_overscroll_indicator.0_test.dart', |
| 'examples/api/test/widgets/restoration/restoration_mixin.0_test.dart', |
| 'examples/api/test/widgets/actions/focusable_action_detector.0_test.dart', |
| 'examples/api/test/widgets/focus_scope/focus_scope.0_test.dart', |
| }; |