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

Resolve symlinks using Node resolution mechanism #3883

Open
gaearon opened this issue Jan 21, 2018 · 16 comments
Open

Resolve symlinks using Node resolution mechanism #3883

gaearon opened this issue Jan 21, 2018 · 16 comments

Comments

@gaearon
Copy link
Contributor

gaearon commented Jan 21, 2018

Copying my comment from #1107 since that issue is pretty crowded and I want it to be here instead.


Having read some discussions in nodejs/node#6537 and https://github.com/isaacs/node6-module-system-change I think we're actually already being inconsistent in how we handle symlinks.

Let's forget about the issue about compiling source code for a second. I feel like that can be solved by #1333 and thus is not the most important one here.

The important part here is that the resolution should match Node algorithm so that our webpack setup matches our test setup. I thought that was the case, but I was wrong.

Consider this structure:

my-comp
  index.js // already compiled
  package.json // react is a peer dependency

my-app
  node_modules
    my-comp // symlink to my-comp folder

It turns out that if my-comp doesn't declare a dependency on react, it can find React in the browser builds but not in tests.

screen shot 2018-01-21 at 12 25 54

screen shot 2018-01-21 at 12 26 06

We introduced this regression in #1359.

We probably missed it because if my-comp does explicitly depend on React (e.g. as a devDependency), the test passes (and in the browser we just get two Reacts).

@gaearon gaearon added this to the 2.0.0 milestone Jan 21, 2018
@gaearon
Copy link
Contributor Author

gaearon commented Jan 21, 2018

I think the correct fix would be to:

  • Have the webpack build respect Node resolution and fail in this case
  • Add an opt-in to Node "preserve symlinks" behavior which seems to be what people using npm link really want

@gaearon
Copy link
Contributor Author

gaearon commented Jan 21, 2018

Even more interesting. Apparently Jest itself doesn't quite use Node module resolution.

In fact I was the one who changed Jest from --preserve-symlinks behavior to normal behavior 😅 jestjs/jest#4761

It seems like maybe we should support both? I believe webpack has a flag for it. Maybe Jest could have a flag too.

@gaearon
Copy link
Contributor Author

gaearon commented Jan 21, 2018

While this is fresh in my mind, I immortalized this dilemma in a tweet.
https://mobile.twitter.com/dan_abramov/status/955061849648779264

Hope this is helpful to whoever ends up reading this to consider fixes (me?)

@gaearon
Copy link
Contributor Author

gaearon commented Jan 21, 2018

Filed an issue in Jest.
jestjs/jest#5356

@gaearon
Copy link
Contributor Author

gaearon commented Jan 21, 2018

@bradfordlemley Do you have thoughts on this? You probably spent more thinking about symlinks than most of us :-)

@gaearon
Copy link
Contributor Author

gaearon commented Jan 21, 2018

After more digging, the reason "jest already worked correctly" in #1359 per #1359 (comment) was because Jest was not respecting Node resolution. Since I fixed Jest in jestjs/jest#4761, that integration has regressed. But again, it was broken in the first place because it didn't match how Node resolves modules.

@1st
Copy link

1st commented Mar 27, 2018

Is it somehow related to #3547?

@1st
Copy link

1st commented Mar 27, 2018

I found a workaround that I use in my projects. You can run a watch process to compile your shared library and use it as symlink in your project. Here is my gist.

@conartist6
Copy link
Contributor

Any progress on this? I'm game to help if there's work to be done.

@omerdn1
Copy link

omerdn1 commented Sep 11, 2018

Any progress or a workaround for this that doesn't require running a watcher compiler?

@Timer Timer modified the milestones: 2.0.x, 2.x Sep 26, 2018
@Timer Timer removed this from the 2.x milestone Nov 2, 2018
@ianqueue
Copy link

ianqueue commented Mar 1, 2019

Any progress on this?

@pietmichal
Copy link

pietmichal commented Mar 7, 2019

Is there a scenario where someone would prefer having dependencies read from linked node_modules instead from CRA-based project's node_modules? Modifying webpack.config.js to have { resolve: { symlinks: false } } fixes the "two react" problem.

If there is no scenario, then maybe CRA should have the value, from above, hardcoded?

@tannerlinsley
Copy link

tannerlinsley commented Jul 3, 2019

I just encountered this and after many many attempts to get around it, I ended up installing rescripts and doing this:

// .rescriptsrc.js

const path = require('path')
const resolveFrom = require('resolve-from')

module.exports = [
  config => {
    config.resolve = {
      ...config.resolve,
      alias: {
        ...config.resolve.alias,
        react$: resolveFrom(path.resolve('node_modules'), 'react'),
        'react-dom$': resolveFrom(path.resolve('node_modules'), 'react-dom')
      }
    }
    return config
  }
]

Leaving here for posterity.

@temaivanoff
Copy link

Up

@ianschmitz
Copy link
Contributor

I believe #7993 will fix this.

@PeledYuval
Copy link

Over at Vim we've arrived at a decent working solution using craco.
I've elaborated here: #3547 (comment)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging a pull request may close this issue.