-
Notifications
You must be signed in to change notification settings - Fork 607
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[api-extractor] Add options to include forgotten exports in API report and doc model files #3552
Conversation
// All internal, exported APIs default to public if no release tag is specified. | ||
options.effectiveReleaseTag = ReleaseTag.Public; | ||
} | ||
} else { | ||
// All external APIs default to public if no release tag is specified. | ||
options.effectiveReleaseTag = ReleaseTag.Public; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This change prevents unexported items included in an API report from being assigned the @public
release tag if none is specified.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
After some discussion, I think this logic is not right. The "effective" release tag must always be something other than ReleaseTag.None
.
The simplest approach would be to handle forgotten exports like any other API item:
- API Extractor complains if it doesn't have a release tag (because for example .d.ts trimming is not going to produce good results if developers don't think carefully about what is/isn't public)
- If you disable the warnings, then everything defaults to
@public
(because in this case, the developer presumably doesn't care very much about trimming or release tags)
Note that when includeForgottenExports=false
then this validation is not useful, and prior to this PR, API Extractor would not complain about problems with forgotten exports. But with includeForgottenExports=true
we probably do want such validation.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Updated this PR to match the behavior you described above. However, I was thinking a bit more about forgotten exports and release tags and sent you a message on Zulip, reproduced below:
We decided yesterday that the missing release tag warning should only be logged if includeForgottenExports is on. I'm wondering now if the missing release tag warning should be logged regardless of whether that config is on. Why? Because .d.ts trimming will produce better results if forgotten exports have release tags.
Maybe the original thinking was that a developer will already get an ae-forgotten-export warning for these items, and so there's already "some warning" indicating that these items should be exported (and thus also should have release tags). But (1) what if the developer turned off that warning and (2) even if the developer didn't, it still might be useful to report both warnings (otherwise a developer might slap on an export, re-run API Extractor, and then just hit another warning right after, as they forgot to also add a release tag).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Agreed, but I think we could implement that extra validation in a separate PR.
build-tests/api-extractor-lib2-test/etc/api-extractor-lib2-test.api.md
Outdated
Show resolved
Hide resolved
// @public (undocumented) | ||
class ForgottenClass { | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We want this change. ForgottenClass
is exported by forgottenNs
but forgottenNs
isn't actually exported itself. Thus, ForgottenClass
isn't exported and thus shouldn't appear in the API report. Previously ForgottenClass
was marked as consumable when it in fact was not (this bug is mentioned in the PR description).
...d-tests/api-extractor-scenarios/etc/includeForgottenExports/api-extractor-scenarios.api.json
Outdated
Show resolved
Hide resolved
build-tests/api-extractor-scenarios/etc/includeForgottenExports/api-extractor-scenarios.api.md
Outdated
Show resolved
Hide resolved
...d-tests/api-extractor-scenarios/etc/includeForgottenExports/api-extractor-scenarios.api.json
Outdated
Show resolved
Hide resolved
"members": [ | ||
{ | ||
"kind": "Class", | ||
"canonicalReference": "api-extractor-scenarios!~ForgottenExport4~ForgottenExport5:class", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This canonical reference should be api-extractor-scenarios!~ForgottenExport4.ForgottenExport5:class
, because ForgottenExport5
is exported from ForgottenExport4
. However, unfortunately ApiExportedMixin.isExported
is false
for ForgottenExport5
, because ForgottenExport5
determines its exported state from ForgottenExport4
's CollectorEntity
(i.e. it doesn't have its own CollectorEntity
).
This is fundamentally an instance of the pre-existing bug 2 mentioned in the PR description.
{ | ||
"kind": "Reference", | ||
"text": "DuplicateName", | ||
"canonicalReference": "api-extractor-scenarios!~DuplicateName:type" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This canonical reference should be api-extractor-scenarios!~DuplicateName_2:type
. This is the pre-existing bug 1 mentioned in the PR description.
{ | ||
"kind": "Reference", | ||
"text": "ForgottenExport6", | ||
"canonicalReference": "api-extractor-scenarios!ForgottenExport6:class" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Observe that the canonical reference for ForgottenExport6
as determined from the .api.json
above is api-extractor-scenarios!internal2.ForgottenExport6:class
. This is an existing bug with AstNamespaceImport
s, discussed a bit in #3584 (comment).
prop?: ForgottenExport2; | ||
} | ||
|
||
// Warning: (ae-unresolved-inheritdoc-reference) The @inheritDoc reference could not be resolved: The package "api-extractor-scenarios" does not have an export "ForgottenExport1" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is bug 3 mentioned in the PR description.
👍 I tried running this on a bunch of monorepo projects with |
# Conflicts: # build-tests/install-test-workspace/workspace/common/pnpm-lock.yaml
🚀 This feature was released with API Extractor 7.30.0 |
Summary
Fixes #3513.
Details
apiReport.includeForgottenExports
anddocModel.includeForgottenExports
configs toapi-extractor.json
that control whether forgotten exports are included in the API report and doc model files, respectively.ApiExportedMixin
mixin to api-extractor-model to allow API items to store whether they are exported or not from their container module (i.e. entry point or namespace). This is necessary to build the proper canonical reference (i.e. with or without a~
navigation step) from the.api.json
. Note that this mixin doesn't actually serialize the newisExported
property to the.api.json
because this info is already present in the canonical reference, thus no.api.json
change needed.CollectionEntity
and the parts of theCollector
that construct them to change the definitions of "consumable" and "exported" as previously defined by api-extractor. More on this below.Consumable vs exported
Before this PR, collector entities could be exported and/or consumable. An exported entity was a top-level export, like:
whereas a consumable entity was either a top-level export OR an entity exported from an
AstNamespaceImport
. An example of the latter might be:This previous definition of consumable is incomplete and misleading for a few reasons:
AstNamespaceImport
n
is exported. This is a bug. This PR fixes that bug.AstNamespaceImport
namespaces, not normalAstSymbol
namespaces. There's fundamentally no reason why the two should be treated differently. This also leads to bugs. This PR doesn't updateAstSymbol
namespaces to be treated similarly toAstNamespaceImport
s, but reworksCollectionEntity
and the parts of theCollector
that construct them in order to pave the way for a future PR to do this.This PR changes the definition of consumable to the following: A consumable entity is either a top-level export or an entity exported from a consumable parent namespace.
This PR also updates the definition of exported. The previous definition is not particularly useful today (almost all code paths use
consumable
instead ofexported
). Instead, we need to know whether an entity is exported with respect to its parent module, for the purposes of writing the proper navigation step in api-extractor-model. Thus, this PR updates the definition of exported to the following: An exported entity is an entity exported from its parent API item.Ambiguous IDs
As discussed in #3513, unexported API items will encounter an instance of the "ae-ambiguous-ids" problem. In short, it's possible for there to be multiple unexported API items with the same name that all need to be written to the
.api.json
. Consider the following:foo.ts:SomeType
andbar.ts:SomeType
are both forgotten exports and will be written to the doc model. ifdocModel.includeForgottenExports
is enabled. What names and canonical references do we give these items?This PR reuses the existing logic in
Collector._makeUniqueNames
to assign a uniquenameForEmit
for each item. In practice, oneSomeType
will keep it's name, and the other will be de-duped toSomeType_2
. These unique names will be used when writing both the items' names and canonical references to the doc model. TheincludeForgottenExports
test case tests this scenario.There are a few important things to note about this:
ambientNameConflict2
test case,Date
andDate_2
). This PR uses this same solution.@label
tag as discussed in [api-extractor] Option to include un-exported types in doc model output #3513.Known bugs/missing features
This is a list of some known bugs and missing features associated with this PR. If anything in this list should block this PR being submitted, please let me know!
DeclarationReferenceGenerator
) use the item's original name, not it's uniquenameForEmit
name. I didn't include this fix in this PR because this is an existing bug..api.json
and.api.md
even if theincludeForgottenExports
setting is off ([api-extractor] Add options to include forgotten exports in API report and doc model files #3552 (comment)). I didn't include this fix in this PR because this is an existing bug.@link
and@inheritDoc
references to unexported API items cannot be resolved. I didn't include this fix in this PR because it seemed like a a relatively minor missing feature and easy to fix as a follow-on task.How it was tested
I added a new test to api-extractor-scenarios and manually validated that stuff looked correct.