create-react-app-typescript
create-react-app-typescript copied to clipboard
Sharing code between multiple projects
Hello. I'm trying to prototype sharing TS code between a web and a react-native application. At the moment my directory structure looks like this:
/ <-- project root
web <-- folder created by create-react-app-typescript
src
shared <-- symlink to ../../shared
App.tsx
native <-- folder created by create-react-native-app
shared <-- folder with .ts and .tsx files I'd like to share between both directories
x.js
x.tsx
Working inside of App.tsx, if I try to import { foo } from './shared/x.js' I can do so without any trouble. However if I try to import { foo } from './shared/x.tsx' I get errors that make it sound like x.tsx is not getting compiled.
Any advice on how to solve this issue? Ideally by teaching ts to properly resolve symlinks.
[...] I get errors that make it sound like [...]
Can you be a bit more precise on this, i.e. provide the error output?
I don't actually have them on hand, unfortunately - I've kept iterating as the day went on. When importing a named export (import {foo} from '../shared/file) foo was undefined, and when importing a default export (import foo from '../shared/file) the shared file was erroring with an illegal character. Same file otherwise, only difference was export vs export default
if that isn't enough info, feel free to close the issue. I'd understand
I'd be happy to help, but without any reproduction project or at least the error message(s), it's hard to figure something out.
Lemme see if I can reproduce it - I have more time today than yesterday
https://github.com/JNaftali/codeshare-demo
Hm... are you sure this repo illustrates the error you're receiving? I don't get any, though.
- Cloned the repo
- Switched to the
webfolder yarnyarn start
=> No errors in terminal, and app works fine.
the 'shared' folder contains two files (jstest.js and tstest.ts) each with identical code. Each file exports two strings, one as a named export and one as a default export. I try to reference all 4 of the strings in App.tsx but only the strings exported from jstest.js make it through. The named TS export is replaced by what appears to be a partial file path, and the default export is replaced by undefined
Hm, I see.
This is what I get when logging these entries (same order as in your code):
App.tsx:8 This is a js named export
App.tsx:9 This is a js default export
App.tsx:10 undefined
App.tsx:11 /static/media/tstest.e6792449.ts
The last entry is a file path generated by file-loader, which serves as a fallback for all files that have not been processed otherwise (the "oneOf" rule).
Most of the rules in there have a limited inclusion scope, paths.appSrc in most cases. While this path is forcefully resolved via the paths configuration, this does not seem to affect any paths inside of it.
There is a known (huge, unlikely) glitch regarding symlink resolution in webpack (quite old, but still active: https://github.com/webpack/webpack/issues/1643).
Setting symlinks: false in the resolve part of your webpack config should solve this - at least the result in my case was:
App.tsx:8 This is a js named export
App.tsx:9 This is a js default export
App.tsx:10 this is a ts named export
App.tsx:11 this is a ts default export
But beware of the potential side-effects this might have - see https://webpack.js.org/configuration/resolve/#resolve-symlinks.
Suppose that's why this option is true by default. Thus, you won't be able to change it without ejecting or using unofficial overwriting utilities.
I suppose I could also fork your version of react-scripts... but whatever I end up doing, thank you so much for your help with this and all the work you've done on this project!
@JNaftali I just ran into this problem and was about to create an issue for it. How did you resolve the issue?
It seems like webpack doesn't like symlinks: webpack/webpack#1643 So CRA has this same problem: facebook/create-react-app#1333
They recommend we share code with yarn/lerna workspace. CRA support for workspace already landed on 2.0: facebook/create-react-app#3815
So we should get it when CRA 2.0 is merged?, IIUC.
My work hasn't started on the project for which we require this, so we haven't settled on an answer yet. We're not too enthused about workspaces so we're considering using git submodules.
Another alternative is making sure TS transpiles dependencies and declaring that folder as a dependency (stackoverflow link)
I've implemented a workaround using tsc's baseUrl which CRA-ts already supported.
https://github.com/untsamphan/cra-ts-monorepo-example
Need to eject (only) if you want tsc/webpack to grab .ts from the local packages.
Steps
-
Initial boilerplate from create-react-app-typescript in root/webapp commit.
-
Create yarn workspace commit.
-
Move CRA tsconfig.json to root to be shared with other local packages commit 1, commit 2.
-
Implement a trivial local packages
root/packages/mymaincommit. Run yarn now so it'll create symlink to mymain inroot/node_modules. Then we canimport from 'mymain'from anywhere. -
Make the CRA app use the new local packages commit. Now the CRA app will tsc compile correctly (because we have index.ts at mymain root). But when
yarn startit'll fail in browser because mymain isn't built. To fix this we can tsc in mymain to build the package and the app will run successfully. However, when wego to definitionto a symbol in mymain, it'll goto a .d.ts. -
To achieve goal 3 (go to definition -> .ts), we configure tsconfig.json
baseUrlto directly reference local packages. Since webpack won't bundle code outsidewebapp/src, and jest won't find the packages, we have to eject to configure them. commit -
Simple webpack config hack to allow it to bundle code outside
webpack/src. This to achieve goal 3. commit. Don't forget to deletebuildin local packages, because otherwise everyone will usebuild/index.*(per NPM spec) instead ofindex.tsat the local package root (a TS-specific behavior). -
Simple jest config hack to make jest inside webapp also run all tests in local packages. commit
Now that TypeScript 3.0 introduced project references, it looks like we just need to wait for ts-loader to support project references and we'll then be able to use that?
See https://github.com/TypeStrong/ts-loader/issues/815 for reference.
Looks like ts-loader has support for this now (815 is closed)
when one issue closes, another one opens - https://github.com/TypeStrong/ts-loader/issues/851