| Index: pkg/analyzer/lib/src/lint/util.dart
|
| diff --git a/pkg/analyzer/lib/src/lint/util.dart b/pkg/analyzer/lib/src/lint/util.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..e1c63af6a62eb1ec5fdcb3363a293cc91afe8e01
|
| --- /dev/null
|
| +++ b/pkg/analyzer/lib/src/lint/util.dart
|
| @@ -0,0 +1,172 @@
|
| +// Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file
|
| +// for details. 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:analyzer/analyzer.dart';
|
| +import 'package:analyzer/dart/ast/token.dart';
|
| +import 'package:analyzer/src/dart/ast/token.dart';
|
| +import 'package:analyzer/src/dart/scanner/reader.dart';
|
| +import 'package:analyzer/src/dart/scanner/scanner.dart';
|
| +import 'package:analyzer/src/generated/parser.dart' show Parser;
|
| +import 'package:analyzer/src/string_source.dart' show StringSource;
|
| +import 'package:path/path.dart' as p;
|
| +
|
| +final _identifier = new RegExp(r'^([(_|$)a-zA-Z]+([_a-zA-Z0-9])*)$');
|
| +
|
| +final _lowerCamelCase =
|
| + new RegExp(r'^(_)*[?$a-z][a-z0-9?$]*([A-Z][a-z0-9?$]*)*$');
|
| +
|
| +final _lowerCaseUnderScore = new RegExp(r'^([a-z]+([_]?[a-z0-9]+)*)+$');
|
| +
|
| +final _lowerCaseUnderScoreWithDots =
|
| + new RegExp(r'^[a-z][_a-z0-9]*(\.[a-z][_a-z0-9]*)*$');
|
| +
|
| +final _pubspec = new RegExp(r'^[_]?pubspec\.yaml$');
|
| +
|
| +final _underscores = new RegExp(r'^[_]+$');
|
| +
|
| +/// Create a library name prefix based on [libraryPath], [projectRoot] and
|
| +/// current [packageName].
|
| +String createLibraryNamePrefix(
|
| + {String libraryPath, String projectRoot, String packageName}) {
|
| + // Use the posix context to canonicalize separators (`\`).
|
| + var libraryDirectory = p.posix.dirname(libraryPath);
|
| + var path = p.posix.relative(libraryDirectory, from: projectRoot);
|
| + // Drop 'lib/'.
|
| + var segments = p.split(path);
|
| + if (segments[0] == 'lib') {
|
| + path = p.posix.joinAll(segments.sublist(1));
|
| + }
|
| + // Replace separators.
|
| + path = path.replaceAll('/', '.');
|
| + // Add separator if needed.
|
| + if (path.isNotEmpty) {
|
| + path = '.$path';
|
| + }
|
| +
|
| + return '$packageName$path';
|
| +}
|
| +
|
| +/// Returns `true` if this [fileName] is a Dart file.
|
| +bool isDartFileName(String fileName) => fileName.endsWith('.dart');
|
| +
|
| +/// Returns `true` if this [name] is a legal Dart identifier.
|
| +bool isIdentifier(String name) => _identifier.hasMatch(name);
|
| +
|
| +/// Returns `true` of the given [name] is composed only of `_`s.
|
| +bool isJustUnderscores(String name) => _underscores.hasMatch(name);
|
| +
|
| +/// Returns `true` if this [id] is `lowerCamelCase`.
|
| +bool isLowerCamelCase(String id) =>
|
| + id.length == 1 && isUpperCase(id.codeUnitAt(0)) ||
|
| + id == '_' ||
|
| + _lowerCamelCase.hasMatch(id);
|
| +
|
| +/// Returns `true` if this [id] is `lower_camel_case_with_underscores`.
|
| +bool isLowerCaseUnderScore(String id) => _lowerCaseUnderScore.hasMatch(id);
|
| +
|
| +/// Returns `true` if this [id] is `lower_camel_case_with_underscores_or.dots`.
|
| +bool isLowerCaseUnderScoreWithDots(String id) =>
|
| + _lowerCaseUnderScoreWithDots.hasMatch(id);
|
| +
|
| +/// Returns `true` if this [fileName] is a Pubspec file.
|
| +bool isPubspecFileName(String fileName) => _pubspec.hasMatch(fileName);
|
| +
|
| +/// Returns `true` if the given code unit [c] is upper case.
|
| +bool isUpperCase(int c) => c >= 0x40 && c <= 0x5A;
|
| +
|
| +class Spelunker {
|
| + final String path;
|
| + final IOSink sink;
|
| + Spelunker(this.path, {IOSink sink}) : this.sink = sink ?? stdout;
|
| +
|
| + void spelunk() {
|
| + var contents = new File(path).readAsStringSync();
|
| +
|
| + var errorListener = new _ErrorListener();
|
| +
|
| + var reader = new CharSequenceReader(contents);
|
| + var stringSource = new StringSource(contents, path);
|
| + var scanner = new Scanner(stringSource, reader, errorListener);
|
| + var startToken = scanner.tokenize();
|
| +
|
| + errorListener.throwIfErrors();
|
| +
|
| + var parser = new Parser(stringSource, errorListener);
|
| + var node = parser.parseCompilationUnit(startToken);
|
| +
|
| + errorListener.throwIfErrors();
|
| +
|
| + var visitor = new _SourceVisitor(sink);
|
| + node.accept(visitor);
|
| + }
|
| +}
|
| +
|
| +class _ErrorListener implements AnalysisErrorListener {
|
| + final errors = <AnalysisError>[];
|
| +
|
| + @override
|
| + void onError(AnalysisError error) {
|
| + errors.add(error);
|
| + }
|
| +
|
| + void throwIfErrors() {
|
| + if (errors.isNotEmpty) {
|
| + throw new Exception(errors);
|
| + }
|
| + }
|
| +}
|
| +
|
| +class _SourceVisitor extends GeneralizingAstVisitor {
|
| + int indent = 0;
|
| +
|
| + final IOSink sink;
|
| + _SourceVisitor(this.sink);
|
| +
|
| + String asString(AstNode node) =>
|
| + typeInfo(node.runtimeType) + ' [${node.toString()}]';
|
| +
|
| + List<CommentToken> getPrecedingComments(Token token) {
|
| + var comments = <CommentToken>[];
|
| + var comment = token.precedingComments;
|
| + while (comment is CommentToken) {
|
| + comments.add(comment);
|
| + comment = comment.next;
|
| + }
|
| + return comments;
|
| + }
|
| +
|
| + String getTrailingComment(AstNode node) {
|
| + var successor = node.endToken.next;
|
| + if (successor != null) {
|
| + var precedingComments = successor.precedingComments;
|
| + if (precedingComments != null) {
|
| + return precedingComments.toString();
|
| + }
|
| + }
|
| + return '';
|
| + }
|
| +
|
| + String typeInfo(Type type) => type.toString();
|
| +
|
| + @override
|
| + visitNode(AstNode node) {
|
| + write(node);
|
| +
|
| + ++indent;
|
| + node.visitChildren(this);
|
| + --indent;
|
| + return null;
|
| + }
|
| +
|
| + write(AstNode node) {
|
| + //EOL comments
|
| + var comments = getPrecedingComments(node.beginToken);
|
| + comments.forEach((c) => sink.writeln('${" " * indent}$c'));
|
| +
|
| + sink.writeln(
|
| + '${" " * indent}${asString(node)} ${getTrailingComment(node)}');
|
| + }
|
| +}
|
|
|