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

Fix virtual module type to properly handle optional properties #13327

Open
wants to merge 4 commits into
base: dev
Choose a base branch
from

Conversation

smorimoto
Copy link

@smorimoto smorimoto commented Mar 29, 2025

Overview

This PR fixes the TypeScript type definition for the virtual module virtual:react-router/server-build to properly handle optional properties in the ServerBuild interface. The change moves from named exports to a CommonJS-style export = syntax to ensure better TypeScript compatibility, especially with exactOptionalPropertyTypes enabled.

Potential Issues or Risks

  • Migration impact: The change from named exports to a default export might require modifications in code that specifically imports individual properties from the virtual module
  • ESM compatibility: As noted in the comments, this approach isn't strictly ESM-compliant, which is a conscious tradeoff made for better typing support

Conclusion

This PR provides a solid solution to type compatibility issues with optional properties in the ServerBuild interface. The approach is well-reasoned and properly tested. The change should improve type-checking accuracy, especially in projects using strict TypeScript configurations.

Copy link

changeset-bot bot commented Mar 29, 2025

🦋 Changeset detected

Latest commit: 4d600f5

The changes in this PR will be included in the next version bump.

This PR includes changesets to release 11 packages
Name Type
react-router Patch
@react-router/architect Patch
@react-router/cloudflare Patch
@react-router/dev Patch
react-router-dom Patch
@react-router/express Patch
@react-router/node Patch
@react-router/serve Patch
@react-router/fs-routes Patch
@react-router/remix-routes-option-adapter Patch
create-react-router Patch

Not sure what this means? Click here to learn what changesets are.

Click here if you're a maintainer who wants to add another changeset to this PR

@remix-cla-bot
Copy link
Contributor

remix-cla-bot bot commented Mar 29, 2025

Hi @smorimoto,

Welcome, and thank you for contributing to React Router!

Before we consider your pull request, we ask that you sign our Contributor License Agreement (CLA). We require this only once.

You may review the CLA and sign it by adding your name to contributors.yml.

Once the CLA is signed, the CLA Signed label will be added to the pull request.

If you have already signed the CLA and received this response in error, or if you have any questions, please contact us at [email protected].

Thanks!

- The Remix team

@remix-cla-bot
Copy link
Contributor

remix-cla-bot bot commented Mar 29, 2025

Thank you for signing the Contributor License Agreement. Let's get this merged! 🥳

@smorimoto
Copy link
Author

This PR will also fix this issue: #13152 (comment)

@smorimoto smorimoto force-pushed the fix-virtual-module-type branch from 8648133 to 8b693bf Compare March 30, 2025 09:37
@smorimoto smorimoto changed the title Simplify the type declaration for the virtual module virtual:react-router/server-build Fix virtual module type to properly handle optional properties Mar 30, 2025
@smorimoto smorimoto force-pushed the fix-virtual-module-type branch from 8b693bf to a5ca825 Compare March 30, 2025 09:39
@smorimoto
Copy link
Author

@timdorr Thanks a lot for clicking accept for the GitHub Actions workflow, but the previous commit was not a complete fix. Sorry!

@markdalgleish I think you will be interested in this, but maybe this change is not to your liking... (I hope you don't hate it!)

@smorimoto
Copy link
Author

Off-topic: I believe both contributors and you all could benefit from slightly relaxing the rules around GitHub Actions :)

https://github.com/organizations/remix-run/settings/actions

image

@smorimoto
Copy link
Author

The hosted runner lost communication with the server. Anything in your workflow that terminates the runner process, starves it for CPU/Memory, or blocks its network access can cause this error.

🥲

@smorimoto
Copy link
Author

Ready to go!

@smorimoto smorimoto force-pushed the fix-virtual-module-type branch from a5ca825 to 745a4c0 Compare April 2, 2025 21:10
@smorimoto
Copy link
Author

Can someone review this?

The virtual module `virtual:react-router/server-build` now deliberately
implements CommonJS `export =` syntax to correctly support optional
properties in the `ServerBuild` type, ensuring proper TypeScript
compatibility with `exactOptionalPropertyTypes`.

Signed-off-by: Sora Morimoto <[email protected]>
Signed-off-by: Sora Morimoto <[email protected]>
@smorimoto smorimoto force-pushed the fix-virtual-module-type branch from 745a4c0 to 017b8cb Compare April 2, 2025 21:12
Copy link
Member

@markdalgleish markdalgleish left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@smorimoto Thanks for the PR!

After pulling down your PR and trying it out, I noticed that this introduces a different issue which is that the virtual module type now has a default property that doesn't exist in the runtime code.

However, I discovered another solution to the problem that avoids this. By adding | undefined to the types for the optional ServerBuild properties, I'm able to keep your new test passing.

I've pushed an update to your PR with the new implementation but keeping your test case so you still get the credit for helping out here. Hope that's ok! Let me know if this solution looks good from your side and we can get this merged in.

@@ -152,7 +152,7 @@ function register(ctx: Context) {

const virtual = ts`
declare module "virtual:react-router/server-build" {
import { ServerBuild } from "react-router";
import type { ServerBuild } from "react-router"; import { ServerBuild } from "react-router";
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
import type { ServerBuild } from "react-router"; import { ServerBuild } from "react-router";
import type { ServerBuild } from "react-router";

@smorimoto
Copy link
Author

smorimoto commented Apr 4, 2025

@markdalgleish I thought exporting never might be the solution, but that does not seem to work...

declare module '...' {
  ...
  export default never; // <- does not work...
  export default undefined // <- same...
}

Copy link
Author

@smorimoto smorimoto left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This change is technically required for compatibility with exactOptionalPropertyTypes. That said, I don't know why the current test does not catch this issue correctly...

basename?: string;
// `| undefined` is required to ensure compatibility with TypeScript's
// `exactOptionalPropertyTypes` option
basename?: string | undefined;
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
basename?: string | undefined;
basename: string | undefined;

unstable_getCriticalCss?: (args: {
pathname: string;
}) => OptionalCriticalCss | Promise<OptionalCriticalCss>;
unstable_getCriticalCss?:
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
unstable_getCriticalCss?:
unstable_getCriticalCss:

@smorimoto
Copy link
Author

@markdalgleish I'm not familiar with Vite's virtual module convention, but on the contrary, is it difficult to publish the default export instead?

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

Successfully merging this pull request may close these issues.

2 participants