Skip to content

Commit

Permalink
Parse and remove doc-imports from comment text (#3803)
Browse files Browse the repository at this point in the history
  • Loading branch information
srawlins authored Jun 28, 2024
1 parent 31833c3 commit 4259b0b
Show file tree
Hide file tree
Showing 8 changed files with 413 additions and 228 deletions.
2 changes: 1 addition & 1 deletion lib/src/model/accessor.dart
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@ class Accessor extends ModelElement {
}
// TODO(srawlins): This doesn't seem right... the super value has delimiters
// (like `///`), but this one doesn't?
return stripComments(super.documentationComment);
return stripCommentDelimiters(super.documentationComment);
}();

/// If this is a getter, assume we want synthetic documentation.
Expand Down
38 changes: 33 additions & 5 deletions lib/src/model/documentation_comment.dart
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,10 @@ mixin DocumentationComment
/// `{@}`-style directives (except tool directives), returning the processed
/// result.
String _processCommentWithoutTools(String documentationComment) {
var docs = stripComments(documentationComment);
// We must first strip the comment of directives like `@docImport`, since
// the offsets are for the source text.
var docs = _stripDocImports(documentationComment);
docs = stripCommentDelimiters(docs);
if (docs.contains('{@')) {
docs = _injectYouTube(docs);
docs = _injectAnimations(docs);
Expand All @@ -111,9 +114,13 @@ mixin DocumentationComment
/// Process [documentationComment], performing various actions based on
/// `{@}`-style directives, returning the processed result.
@visibleForTesting
Future<String> processComment(String documentationComment) async {
var docs = stripComments(documentationComment);
// Must evaluate tools first, in case they insert any other directives.
Future<String> processComment() async {
// We must first strip the comment of directives like `@docImport`, since
// the offsets are for the source text.
var docs = _stripDocImports(documentationComment);
docs = stripCommentDelimiters(docs);
// Then we evaluate tools, in case they insert any other directives that
// would need to be processed by `processCommentDirectives`.
docs = await _evaluateTools(docs);
docs = processCommentDirectives(docs);
_analyzeCodeBlocks(docs);
Expand Down Expand Up @@ -557,6 +564,27 @@ mixin DocumentationComment
});
}

String _stripDocImports(String content) {
if (modelNode?.commentData case var commentData?) {
var commentOffset = commentData.offset;
var buffer = StringBuffer();
if (commentData.docImports.isEmpty) return content;
var firstDocImport = commentData.docImports.first;
buffer.write(content.substring(0, firstDocImport.offset - commentOffset));
var offset = firstDocImport.end - commentOffset;
for (var docImport in commentData.docImports.skip(1)) {
buffer
.write(content.substring(offset, docImport.offset - commentOffset));
offset = docImport.end - commentOffset;
}
// Write from the end of the last doc-import to the end of the comment.
buffer.write(content.substring(offset));
return buffer.toString();
} else {
return content;
}
}

/// Parse and remove &#123;@inject-html ...&#125; in API comments and store
/// them in the index on the package, replacing them with a SHA1 hash of the
/// contents, where the HTML will be re-injected after Markdown processing of
Expand Down Expand Up @@ -793,7 +821,7 @@ mixin DocumentationComment
assert(_rawDocs == null,
'reentrant calls to _buildDocumentation* not allowed');
// Do not use the sync method if we need to evaluate tools or templates.
var rawDocs = await processComment(documentationComment);
var rawDocs = await processComment();
return _rawDocs = buildDocumentationAddition(rawDocs);
}

Expand Down
43 changes: 37 additions & 6 deletions lib/src/model/model_node.dart
Original file line number Diff line number Diff line change
Expand Up @@ -18,14 +18,14 @@ class ModelNode {
final int _sourceEnd;
final int _sourceOffset;

/// Data about each comment reference found in the doc comment of this node.
final Map<String, CommentReferenceData>? commentReferenceData;
/// Data about the doc comment of this node.
final CommentData? commentData;

factory ModelNode(
AstNode? sourceNode,
Element element,
AnalysisContext analysisContext, {
required Map<String, CommentReferenceData>? commentReferenceData,
CommentData? commentData,
}) {
if (sourceNode == null) {
return ModelNode._(element, analysisContext,
Expand All @@ -44,7 +44,7 @@ class ModelNode {
analysisContext,
sourceEnd: sourceNode.end,
sourceOffset: sourceNode.offset,
commentReferenceData: commentReferenceData,
commentData: commentData,
);
}
}
Expand All @@ -54,7 +54,7 @@ class ModelNode {
this._analysisContext, {
required int sourceEnd,
required int sourceOffset,
this.commentReferenceData = const {},
this.commentData,
}) : _sourceEnd = sourceEnd,
_sourceOffset = sourceOffset;

Expand All @@ -79,10 +79,41 @@ class ModelNode {
}();
}

/// Comment data from the syntax tree.
///
/// Various comment data is not available on the analyzer's Element model, so we
/// store it in instances of this class after resolving libraries.
class CommentData {
/// The offset of this comment in the source text.
final int offset;
final List<CommentDocImportData> docImports;
final Map<String, CommentReferenceData> references;

CommentData({
required this.offset,
required this.docImports,
required this.references,
});
}

/// doc-import data from the syntax tree.
///
/// Comment doc-import data is not available on the analyzer's Element model, so
/// we store it in instances of this class after resolving libraries.
class CommentDocImportData {
/// The offset of the doc import in the source text.
final int offset;

/// The offset of the end of the doc import in the source text.
final int end;

CommentDocImportData({required this.offset, required this.end});
}

/// Comment reference data from the syntax tree.
///
/// Comment reference data is not available on the analyzer's Element model, so
/// we store it after resolving libraries in instances of this class.
/// we store it in instances of this class after resolving libraries.
class CommentReferenceData {
final Element element;
final String name;
Expand Down
74 changes: 35 additions & 39 deletions lib/src/model/package_graph.dart
Original file line number Diff line number Diff line change
Expand Up @@ -238,14 +238,14 @@ class PackageGraph with CommentReferable, Nameable {
for (var directive in unit.directives.whereType<LibraryDirective>()) {
// There should be only one library directive. If there are more, there
// is no harm in grabbing ModelNode for each.
var commentReferenceData = directive.documentationComment?.data;
var commentData = directive.documentationComment?.data;
_modelNodes.putIfAbsent(
resolvedLibrary.element,
() => ModelNode(
directive,
resolvedLibrary.element,
analysisContext,
commentReferenceData: commentReferenceData,
commentData: commentData,
));
}

Expand Down Expand Up @@ -287,50 +287,39 @@ class PackageGraph with CommentReferable, Nameable {
}

void _populateModelNodeFor(Declaration declaration) {
var commentReferenceData = declaration.documentationComment?.data;
var commentData = declaration.documentationComment?.data;

void addModelNode(Declaration declaration) {
var element = declaration.declaredElement;
if (element == null) {
throw StateError("Expected '$declaration' to declare an element");
}
_modelNodes.putIfAbsent(
element,
() => ModelNode(
declaration,
element,
analysisContext,
commentData: commentData,
),
);
}

if (declaration is FieldDeclaration) {
var fields = declaration.fields.variables;
for (var field in fields) {
var element = field.declaredElement!;
_modelNodes.putIfAbsent(
element,
() => ModelNode(
field,
element,
analysisContext,
commentReferenceData: commentReferenceData,
),
);
addModelNode(field);
}
return;
}
if (declaration is TopLevelVariableDeclaration) {
var fields = declaration.variables.variables;
for (var field in fields) {
var element = field.declaredElement!;
_modelNodes.putIfAbsent(
element,
() => ModelNode(
field,
element,
analysisContext,
commentReferenceData: commentReferenceData,
),
);
addModelNode(field);
}
return;
}
var element = declaration.declaredElement!;
_modelNodes.putIfAbsent(
element,
() => ModelNode(
declaration,
element,
analysisContext,
commentReferenceData: commentReferenceData,
),
);
addModelNode(declaration);
}

ModelNode? getModelNodeFor(Element element) => _modelNodes[element];
Expand Down Expand Up @@ -1073,10 +1062,16 @@ class InheritableElementsKey {

extension on Comment {
/// A mapping of all comment references to their various data.
Map<String, CommentReferenceData> get data {
if (references.isEmpty) return const {};
CommentData get data {
var docImportsData = <CommentDocImportData>[];
for (var docImport in docImports) {
docImportsData.add(
CommentDocImportData(
offset: docImport.offset, end: docImport.import.end),
);
}

var data = <String, CommentReferenceData>{};
var referencesData = <String, CommentReferenceData>{};
for (var reference in references) {
var commentReferable = reference.expression;
String name;
Expand All @@ -1096,15 +1091,16 @@ extension on Comment {
continue;
}

if (staticElement != null && !data.containsKey(name)) {
data[name] = CommentReferenceData(
if (staticElement != null && !referencesData.containsKey(name)) {
referencesData[name] = CommentReferenceData(
staticElement,
name,
commentReferable.offset,
commentReferable.length,
);
}
}
return data;
return CommentData(
offset: offset, docImports: docImportsData, references: referencesData);
}
}
3 changes: 2 additions & 1 deletion lib/src/utils.dart
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,8 @@ Iterable<String> stripCommonWhitespace(String str) sync* {
}
}

String stripComments(String str) {
/// Strips [str] of all comment delimiters.
String stripCommentDelimiters(String str) {
if (str.isEmpty) return '';
final buf = StringBuffer();

Expand Down
Loading

0 comments on commit 4259b0b

Please sign in to comment.