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

Refine how TypeScript types are handled #5593

Merged
merged 18 commits into from
Oct 28, 2018

Conversation

Timer
Copy link
Contributor

@Timer Timer commented Oct 26, 2018

Right now, we copy our react-app.d.ts file into the user's source folder to register types for our webpack loaders, e.g.:

declare module '*.svg' {
  import * as React from 'react';

  export const ReactComponent: React.SFC<React.SVGProps<SVGSVGElement>>;

  const src: string;
  export default src;
}

declare module '*.module.css' {
  const classes: { [key: string]: string };
  export default classes;
}

I feel like this isn't very optimal, and I'd rather this be "encapsulated" somehow.

On first thought, it looks like we could publish @types/react-scripts and have the types automatically registered -- however, I'm not sure we want that maintenance overhead when the types field exists in package.json.

This pull request is exploring how we could rely on the types key so our types are always specific to the react-scripts version without relying on package manager hoisting metrics/external packages.


It looks like we can trigger TypeScript to import these types by specifiying "types": ["react-scripts"] in tsconfig.json, but this turns off the default behavior of absorbing the @types/ folder.

Also, using include in tsconfig.json seems suboptimal because we can't specify node_modules/react-scripts/config/react-app.d.ts since it might be hoisted.

I suppose the crux of the problem is there's never a time where someone would write import 'react-scripts';.

@Timer Timer added this to the 2.1 milestone Oct 26, 2018
@DanielRosenwasser
Copy link

@weswigham I think you brought this up when we discussed the changes PR earlier. I feel like a global .d.ts is pretty reasonable given that it's app-specific behavior from a loader.

@Timer
Copy link
Contributor Author

Timer commented Oct 27, 2018

By global .d.ts are you referring to us copying the file? Or a new behavior to allow something like this for toolkits/toolboxes to leverage?

p.s. thanks for the quick response!

@weswigham
Copy link

Yeah, I mentioned you could publish the file to DefinitelyTyped and then list it as a dependency, just like @types/node or @types/webpack-env. Type definitions on DT for specificing certain loader/runtime environments are fine. In theory you shouldn't need to specify the explicit types dependency then, since in the absence of an explicitly specified types option, all @types/ definitions locatable within a project are loaded. So yeah, it's a bit more to maintain, but way more automatic, in terms of configuration.

@Timer
Copy link
Contributor Author

Timer commented Oct 27, 2018

I'm worried about versioning and keeping them in sync, would we be able to conceal the dependency from the user by making react-scripts depend on a specific version of the types? I guess we'd be relying on hoisting metrics a bit then, and users could unexpectedly install an improper version.

@weswigham
Copy link

You could list it as a... Peer dependency, I think? So npm will warn if the versions in their project and what the scripts think is right don't match sufficiently.

@Timer
Copy link
Contributor Author

Timer commented Oct 27, 2018

Since TypeScript is optional we'd need to wait for the optional peer dependencies specification to be finalized between Yarn and npm 😅. Maybe copying the file is the best behavior for now...

@Timer
Copy link
Contributor Author

Timer commented Oct 27, 2018

Note: I don't want to go against TypeScript community convention, but I'm not sure if @types/react-scripts is the best route. Users who eject will need to drop this dependency and get a copy of the types file so they can add their modifications.

Because of the above constraint, I feel like it might be best to be consistent and copy the file so there's a better eject story.

I'm more than open to an alternate way to do this than copying a type declaration file into the user's directory. For the meantime, I've altered this PR to restore the copy behavior with additions of type references.
Do you think this is permissible or the best way to do this given the constraints?

@weswigham
Copy link

weswigham commented Oct 27, 2018

You could try making @types/react-scripts-env just be ///<reference types="react-scripts" /> and have people install @types/react-scripts-env as a kind of forwarding service to autoload the environment types.

@Timer
Copy link
Contributor Author

Timer commented Oct 27, 2018

Wow, that's actually an excellent idea! 🤯

Thanks for the suggestion, I think that'll probably work best.

@ianschmitz
Copy link
Contributor

I'm worried about versioning and keeping them in sync, would we be able to conceal the dependency from the user by making react-scripts depend on a specific version of the types? I guess we'd be relying on hoisting metrics a bit then, and users could unexpectedly install an improper version.

Yes this should work @Timer. As we were chatting earlier, this would be picked up automatically by TypeScript (as well as i was mentioning VSCode should pick this up for javascript projects as well in my experience). Are you mainly concerned with @types/react-scripts not ending up in the root of node_modules or something similar?

@Timer
Copy link
Contributor Author

Timer commented Oct 27, 2018

What's best practice to specify our type dependencies (@types/react, @types/react-dom, @types/jest, and @types/node)?

  1. Should they be dependencies of react-scripts?
  2. Dependencies of @types/react-script-env?
  3. Peer dependencies of @types/react-script-env?
  4. A combination of these?

@Timer
Copy link
Contributor Author

Timer commented Oct 27, 2018

Also, is this best practice since we need the types to be installed?
https://github.com/facebook/create-react-app/pull/5593/files#diff-c2f8d8ef1abb763b0555188117513867R5


Are you mainly concerned with @types/react-scripts not ending up in the root of node_modules or something similar?
@ianschmitz

Yes, that's my concern.

e.g. node_modules/react-scripts/node_modules/@types/react-scripts

@ianschmitz
Copy link
Contributor

ianschmitz commented Oct 27, 2018

What's best practice to specify our type dependencies (@types/react, @types/react-dom, @types/jest, and @types/node)?

  1. Should they be dependencies of react-scripts?
  2. Dependencies of @types/react-script-env?
  3. Peer dependencies of @types/react-script-env?
  4. A combination of these?

IMO:

  • @types/react and @types/react-dom should be controlled by the user as they correlate with the react and react-dom dependencies which are user controlled
  • @types/jest i feel should be controlled by CRA. This was an easy source of confusion in CRA-TS as jest was an internally controlled dependency (which was 20.x or 21.x at the time, i forget) but it was easy for a user to accidentally install @types/jest@23.x which would give typings to the wrong APIs
  • @types/node is an interesting one.. not sure what do to do there. as you mentioned it seems specific to the node version that the user has installed

Another thought if you didn't want to manage an @types package, is you could add an entry to the include of tsconfig.json to point at node_modules/react-scripts/declarations/* or a similar path. Typescript will pick up any *.d.ts files in that path and include the typings in the project.

@Timer
Copy link
Contributor Author

Timer commented Oct 28, 2018

node_modules/react-scripts/declarations/* this has the same problem with the root location, no?

@Timer Timer changed the title Add types entry to react-scripts Refine how TypeScript types are handled Oct 28, 2018
@ianschmitz
Copy link
Contributor

node_modules/react-scripts/declarations/* this has the same problem with the root location, no?

I think since react-scripts is a dependency of the user's project, we can guarantee that there will be a node_modules/react-scripts folder?

@Timer
Copy link
Contributor Author

Timer commented Oct 28, 2018

For a while until Yarn PnP is more widespread. Can you DM me on Twitter @timer150?

@Timer
Copy link
Contributor Author

Timer commented Oct 28, 2018

We decided to not publish a types package, but instead use a derivative of the suggested env package.
We'll now create a file forwarding to the react-scripts package. This solves original concerns because it's a one time operation and doesn't cause git diff noise between versions or lints.

Please review this PR and let me know if this seems reasonable! I'm going to merge for now because it's an improvement.

@Timer Timer merged commit eca0ec0 into facebook:master Oct 28, 2018
@Timer Timer deleted the refactor/do-not-expose-types branch October 28, 2018 05:11
zmitry pushed a commit to zmitry/create-react-app that referenced this pull request Oct 29, 2018
* Specify types in package

* Do not remove types file on eject

* Stop copying types into generated project

* Reference react and react-dom

* Reference node types

* Install node types as well

* Restore copying

* Add Node to the list of installed types

* Reference Jest types

* Remove jest types from install

* Remove jest from CRA install

* Remove Jest reference and let user do this themselves

* Stop copying types file

* Add types key to package.json

* Add appTypeDeclarations and create when missing

* Rename declarations file

* Add Jest back to install instructions

* Minimize diff
nate770 pushed a commit to ONTW/create-react-app that referenced this pull request Oct 30, 2018
* Specify types in package

* Do not remove types file on eject

* Stop copying types into generated project

* Reference react and react-dom

* Reference node types

* Install node types as well

* Restore copying

* Add Node to the list of installed types

* Reference Jest types

* Remove jest types from install

* Remove jest from CRA install

* Remove Jest reference and let user do this themselves

* Stop copying types file

* Add types key to package.json

* Add appTypeDeclarations and create when missing

* Rename declarations file

* Add Jest back to install instructions

* Minimize diff
@lock lock bot locked and limited conversation to collaborators Jan 18, 2019
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants