-
Notifications
You must be signed in to change notification settings - Fork 12.6k
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
Allow circular references of project references #33685
Comments
In general, I like that project references are acyclic, honestly. It enforces a proper separation of concerns. Like how F# doesn't like circular references, However, being able to explicitly opt-in to circular project references safely would be fantastic! At the moment, when one has multiple directories that circularly depend on each other, you have to make these directories part of the same subproject. Otherwise, you get a circular reference error. In general, this is great. But... If these directories are huge, then you lose the benefit of chunked incremental builds. Changing code in one directory causes all directories to be re-checked. You lose all the benefits of project references just because your directories are circular. Shame =( I have a use case that would benefit from circular project references, actually. I have 1060 functions that build SQL expression objects in my data access subproject. Build times are slow at the moment and changing one function requires TS to re-check all 1060 functions (18s to check, 58s to emit). I can't make each directory a separate subproject at the moment because,
So, explicit circular project references for my data access layer would be super awesome. TL;DR Complex code modeling relational databases would benefit greatly from explicit circular project references [Edit] |
Anyone have ideas for how to model project references as if they were circular until this is implemented? Maybe have two typescript processes in parallel, where one process uses graph A -> B and other process reads B -> A |
@alexleung I'm not sure if this helps, but my team has been using pnpm workspaces to have our monorepo projects symlinked as node_module dependencies. I don't know if this keeps any of the benefits of TypeScript project references, but it allows us to keep our code separated into distinct modules. The TypeScript compiler works great with this project structure, and the TypeScript Language Server for VSCode works too, although it can be a little slow at times in our project. I'm not sure if it doesn't understand our symlinked structure and sometimes loads the same file multiple times, or if our project is just huge and abusive of TypeScript's powers... Overall it works great for us though, and we've got 30+ modules |
@alexleung In most cases it's possible to break circularities using additional package that will contain this things B needs from A, and both A & B will import from it. |
@dqsully Thanks for that mention of pnpm! I like the idea a lot and have actually been working on a similar project (only 1 installation in entire workspace) which is specific to typescript workspaces (creates project references automatically). Not sure that I can use pnpm unfortunately because it's not flattening out the node_modules folder, which means that the behavior of a package maintained in a pnpm workspace would not be consistent with the package after being published to npm. For example, graphql-js would probably break in pnpm if multiple modules in a pnpm workspace depend on it then depend on each other, since graphql-js has a check where it enforces that schemas exported from a module cannot be imported into different instance of the same module. Npm package flattening solves that problem, but pnpm would not. @Bnaya that isn't an option because I am actually creating a tool similar to lerna, yarn, and pnpm which allows users to define project dependencies as they want with the same restrictions as npm. Npm allows for circular dependencies so I need to find a way to get circular project references auto-generated. |
This is still an issue even when using typescript@next (
So two npm packages within a monorepo that are using each other. The problem is that "module-A" compiles first and it expects "module-B" to be already compiled, but it isn't. The error looks like so:
|
We are moving away from |
I am trying to adopt |
@RyanCavanaugh Do you have any updates on the issue? We're in the process of migrating a project to use project reference, but we've hit a roadblock due to circular dependencies. |
Friendly ping. This would be helpful for our monorepo. |
@RyanCavanaugh I work on elastic/kibana - one of the biggest public TS repos on Github. We've split up the repo in TS projects to improve IDE responsiveness (I'm suspicious whether this was achieved) and CLI type checks. Unfortunately, this creates a problem of circular references between these different projects. It's a ton of work for us to break everything down into smaller |
@dgieselaar nit: you don't need to break into packages (assuming you mean in the NodeJS sense of the word). you need to break into TS projects. |
@david-fong thanks, slip of the tongue, edited the message 🙂 |
Hey, it would be also helpful for us! |
Search Terms
circular reference project references graph
Suggestion
Currently, project reference graphs are required to be acyclical because
.d.ts
file is loaded, and.d.ts
files can't exist before the build occursBoth of these problems are solvable without too much work.
With the work done in #32028, we can effectively toggle the redirecting behavior, fixing the first issue. This would be done during the initial build phase only when an upstream project dependency isn't built yet.
The other problem is that project build order might not be predictable if we arbitrarily pick some starting point in the unsortable graph. This is fixable if we force solution authors to indicate which edges in the graph should be treated as ignored for the purposes of the topological sort:
The benefits of this are:
.d.ts
generation or memory pressure, developers can control where the "weak" link occurs in the build processUse Cases
npm
and other package managers do allow circularities in dependency graphs, so this is apparently a thing.We've also gotten feedback from some industry partners that they want to move to project references, but their dependency graph is circular in a way that would require a lot of work to "fix".
Examples
Given the graph:
The build order is deterministically
C
,B
,A
. DuringC
's compilation, source file redirects fromC
toA
are not active.Checklist
My suggestion meets these guidelines:
The text was updated successfully, but these errors were encountered: