Skip to content
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

Feature: Use subpath exports to declare what is and isn't public #5345

Closed
mxxk opened this issue Dec 6, 2023 · 6 comments
Closed

Feature: Use subpath exports to declare what is and isn't public #5345

mxxk opened this issue Dec 6, 2023 · 6 comments
Labels
enhancement Improvement over existing feature

Comments

@mxxk
Copy link

mxxk commented Dec 6, 2023

I'm working on migrating from DraftJS, and am looking to use various plugins from @lexical/react. However, I've noticed that most plugins there are not actually documented, which makes me wonder whether such undocumented plugins (e.g., LexicalContextMenuPlugin) and hooks (e.g., useLexicalTextEntity) are okay to use. Presently, it's not very clear what is part of the package's public interface (and therefore can be safely imported and depended on) and what is part of the package's internals (and therefore should not be imported). The risk of consuming a private part of any package is that it is unsupported and there are no stability guarantees around it.

I imagine that whatever Lexical has included in the documentation is part of the public interface. However, documentation doesn't always keep pace with the pace of rapid development, so I wonder if the Lexical team has considered one of the following approaches to declare the interface of Lexical packages?

  1. Using package.json subpath exports. Subpath exports were introduced in Node.js v12.7.0 (released July 23, 2019), and have gained JS build tool support over the years (Webpack, Rollup, Vite).

  2. Establishing a convention which distinguishes public and private import paths of Lexical packages. An example of such a convention can be seen at Material UI:

    Be aware that we only support first and second-level imports. Anything deeper is considered private and can cause issues, such as module duplication in your bundle.

    // ✅ OK
    import { Add as AddIcon } from '@mui/icons-material';
    import { Tabs } from '@mui/material';
    //                         ^^^^^^^^ 1st or top-level
    
    // ✅ OK
    import AddIcon from '@mui/icons-material/Add';
    import Tabs from '@mui/material/Tabs';
    //                              ^^^^ 2nd level
    
    // ❌ NOT OK
    import TabIndicator from '@mui/material/Tabs/TabIndicator';
    //                                           ^^^^^^^^^^^^ 3rd level 

    (Source: https://mui.com/material-ui/guides/minimizing-bundle-size/#option-one-use-path-imports)

Between the two options above, subpath exports would probably be preferable, since there is nothing to stop users from violating a convention (besides their own conscience, perhaps 😄), but subpath exports actually enforce their contract:

Now only the defined subpath in "exports" can be imported by a consumer:

import submodule from 'es-module-package/submodule.js';
// Loads ./node_modules/es-module-package/src/submodule.js COPY

While other subpaths will error:

import submodule from 'es-module-package/private-module.js';
// Throws ERR_PACKAGE_PATH_NOT_EXPORTED

Since the changes proposed here apply to multiple packages and possibly considered breaking, I was also wondering if the team can shed some light on a rule/heuristic/convention, if any, which users of Lexical can rely on to know whether an import path is public or private? It would really help to prevent inadvertently building a dependency on something package-private.

Thank you in advance!

@mxxk mxxk added the enhancement Improvement over existing feature label Dec 6, 2023
@acywatson
Copy link
Contributor

Thanks for your thoughts here! I like the subpath idea. I have seen this before, but wasn't aware that there was a convention around it. Seems like something we could consider adopting.

To help out out practically here - it has been our practice to consider anything exported from a public package to be part of our public API for purposes of versioning/support. In other words, we stick to the rules of semantic versioning for any of that.

I still think it would be valuable to communciate it implicitly in one of the ways you describe, but hopefully this helps you feel comfortable moving forward for now.

@mxxk
Copy link
Author

mxxk commented Dec 7, 2023

Thanks for your thoughts here! I like the subpath idea. I have seen this before, but wasn't aware that there was a convention around it. Seems like something we could consider adopting.

To help out out practically here - it has been our practice to consider anything exported from a public package to be part of our public API for purposes of versioning/support. In other words, we stick to the rules of semantic versioning for any of that.

I still think it would be valuable to communicate it implicitly in one of the ways you describe, but hopefully this helps you feel comfortable moving forward for now.

Thank you for your clarification @acywatson! If I understood correctly, Lexical has some packages which are considered "public", and other packages which are considered "private" (and public packages call private packages as needed). Nifty!

As far as figuring out which packages are public, would that be the list explicitly documented on the website?

{
items: [
'packages/lexical',
'packages/lexical-clipboard',
'packages/lexical-code',
'packages/lexical-dragon',
'packages/lexical-file',
'packages/lexical-hashtag',
'packages/lexical-headless',
'packages/lexical-history',
'packages/lexical-link',
'packages/lexical-list',
'packages/lexical-markdown',
'packages/lexical-offset',
'packages/lexical-plain-text',
'packages/lexical-rich-text',
'packages/lexical-selection',
'packages/lexical-table',
'packages/lexical-text',
'packages/lexical-utils',
],
label: 'Packages',
type: 'category',
},

If this is so, then other packages (e.g., @lexical/overflow) are private. Is this accurate?

@acywatson
Copy link
Contributor

We enforce private methods via module exports. No package is entirely private. If you can import a function from that package through conventional means (i.e., if it's exported from the publicly vended package artifacts on NPM), then we treat it as part of our public API.

An example of something that's NOT part of the public API would be ImageNode - anything in the playground, basically.

@mxxk
Copy link
Author

mxxk commented Dec 8, 2023

Understood, thanks again. My original question around public/private package API has been answered, so this issue can be closed, or if it helps to keep it open in case you wanted to pursue subpath exports, however would be best!

@etrepum
Copy link
Collaborator

etrepum commented May 2, 2024

I think all of the issues around this have been resolved in the process of supporting ESM. The public APIs should all show up in the API documentation as well since we have consolidated all of the logic for what's public and what's private and the build and docs related tools all use it now.

It's not using subpath exports because of some implementation details in the way we did it, all of the modules that are exported from each package are explicit.

@mxxk
Copy link
Author

mxxk commented May 2, 2024

Thanks @etrepum! Closing as completed.

@mxxk mxxk closed this as completed May 2, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement Improvement over existing feature
Projects
None yet
Development

No branches or pull requests

3 participants