// 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. import 'dart:io'; import 'package:path/path.dart' as path; /// Scans the dartdoc HTML output in the provided `htmlOutputPath` for /// unresolved dartdoc directives (`{@foo x y}`). /// /// Dartdoc usually replaces those directives with other content. However, /// if the directive is misspelled (or contains other errors) it is placed /// verbatim into the HTML output. That's not desirable and this check verifies /// that no directives appear verbatim in the output by checking that the /// string `{@` does not appear in the HTML output outside of <code> sections. /// /// The string `{@` is allowed in <code> sections, because those may contain /// sample code where the sequence is perfectly legal, e.g. for required named /// parameters of a method: /// /// ``` /// void foo({@required int bar}); /// ``` void checkForUnresolvedDirectives(String htmlOutputPath) { final Directory dartDocDir = Directory(htmlOutputPath); if (!dartDocDir.existsSync()) { throw Exception('Directory with dartdoc output (${dartDocDir.path}) does not exist.'); } // Makes sure that the path we were given contains some of the expected // libraries and HTML files. final List<String> canaryLibraries = <String>[ 'animation', 'cupertino', 'material', 'widgets', 'rendering', 'flutter_driver', ]; final List<String> canaryFiles = <String>[ 'Widget-class.html', 'Material-class.html', 'Canvas-class.html', ]; print('Scanning for unresolved dartdoc directives...'); final List<FileSystemEntity> toScan = dartDocDir.listSync(); int count = 0; while (toScan.isNotEmpty) { final FileSystemEntity entity = toScan.removeLast(); if (entity is File) { if (path.extension(entity.path) != '.html') { continue; } canaryFiles.remove(path.basename(entity.path)); count += _scanFile(entity); } else if (entity is Directory) { canaryLibraries.remove(path.basename(entity.path)); toScan.addAll(entity.listSync()); } else { throw Exception('$entity is neither file nor directory.'); } } if (canaryLibraries.isNotEmpty) { throw Exception('Did not find docs for the following libraries: ${canaryLibraries.join(', ')}.'); } if (canaryFiles.isNotEmpty) { throw Exception('Did not find docs for the following files: ${canaryFiles.join(', ')}.'); } if (count > 0) { throw Exception('Found $count unresolved dartdoc directives (see log above).'); } print('No unresolved dartdoc directives detected.'); } int _scanFile(File file) { assert(path.extension(file.path) == '.html'); final Iterable<String> matches = _pattern.allMatches(file.readAsStringSync()) .map((RegExpMatch m ) => m.group(0)); if (matches.isNotEmpty) { stderr.writeln('Found unresolved dartdoc directives in ${file.path}:'); for (final String match in matches) { stderr.writeln(' $match'); } } return matches.length; } // Matches all `{@` that are not within `<code></code>` sections. // // This regex may lead to false positives if the docs ever contain nested tags // inside <code> sections. Since we currently don't do that, doing the matching // with a regex is a lot faster than using an HTML parser to strip out the // <code> sections. final RegExp _pattern = RegExp(r'({@[^}\n]*}?)(?![^<>]*</code)'); // Usually, the checker is invoked directly from `dartdoc.dart`. Main method // is included for convenient local runs without having to regenerate // the dartdocs every time. // // Provide the path to the dartdoc HTML output as an argument when running the // program. void main(List<String> args) { if (args.length != 1) { throw Exception('Must provide the path to the dartdoc HTML output as argument.'); } if (!Directory(args.single).existsSync()) { throw Exception('The dartdoc HTML output directory ${args.single} does not exist.'); } checkForUnresolvedDirectives(args.single); }