-
Notifications
You must be signed in to change notification settings - Fork 400
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 a component namespace and name parsing issue #5117
base: master
Are you sure you want to change the base?
Conversation
The original approach cannot correctly parse `namespace` and `name` from file path in Windows as the path may contain `\` as delimeters. It doesn't work well for relative paths start with './' either which derives from implicit reference to CSS files.
Thanks for the contribution! Before we can merge this, we need @Mr-VincentW to sign the Salesforce Inc. Contributor License Agreement. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you for looking into this!
In the original issue you mentioned:
Using /[\/]/ here solves the problem for me.
Can you simplify this PR to fix just that and provide a minimal repo that demonstrates this issue?
I gather you're trying to solve what you also mentioned:
It doesn't work well for relative paths start with ./ either which derives from implicit reference to CSS files from HTML templates.
I think this should be done as a separate PR once this one is merged.
--
PS: I'm only a contributor, not a maintainer, so this is just my advice based on a past windows-related issue and how tricky it can be to diagnose, validate the fix and ensure no breaking changes:
cardoso/vite-plugin-lwc#1 -> danjunger/vite-repro#5 -> #4825 (comment) -> #5038
@@ -332,7 +332,7 @@ export default function lwc(pluginOptions: RollupLwcOptions = {}): Plugin { | |||
const [namespace, name] = | |||
// Note we do not need to use path.sep here because this filename contains | |||
// a '/' regardless of Windows vs Unix, since it comes from the Rollup `id` | |||
specifier?.split('/') ?? path.dirname(filename).split('/').slice(-2); | |||
specifier?.split('/') ?? /(?:([^\\/]+?)[\\/])?([^\\/]+?)(?:[\\/]\2)?(?:\.scoped)?\.[^.]+$/.exec(filename)?.slice(1); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think using a regex here is the right approach.
However, in my case (on Windows), I found them cometimes containing \ as well.
Are you able to debug what specifier
and filename
are here in this case you mentioned?
Ideally this line remains unchanged, and the filename
is normalized via simple String.replace previous to it.
The assumption is that rollup already does this normalization, which is why providing a log of the variables as well a minimal repo would be needed to figure out the proper fix.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I've updated my fix to only fix the delimiter issue.
BTW, rollup does normalize the id
s but not the importer
, and the LWC rollup plugin has a custom resolveId
hook in the plugin to resolves id
s based on both importee
(id
) and importer
, which is where the issue derives.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Simplified the fix as only having replaced the path delimiter for parsing `namespace` and `name` from filenames.
Hey, thank you for the feedback. I've simplified my fix and created a demo repo to showcase the issue. Now the fix only solves the path delimiter issue but doesn't handle the relative path problem. Cheers. |
@Mr-VincentW thanks a lot for the demo repo! Does the error still happen if you remove the html plugin and create the index.html file yourself? Example. If the error still happens, I suggest further simplifying the repo by removing the html plugin. If the error goes away, we need to understand why and figure out the best way to deal with it. |
@cardoso Yes, the issue continues. I tend not to remove the plugin in the repo, or you'll need to either copy the Anyway, it has nothing to do with the html. No matter if the demo can run or not in the browser, the issue happens when the package is built. Also, I have already explained the reason why this issue happens here. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sorry for the delayed response, we've had a busy few weeks!
Could you also add a unit test that fails on Windows without this fix, but passes with the fix? The integrations test is a simple one you use as a starting point. Thanks!
// Note we do not need to use path.sep here because this filename contains | ||
// a '/' regardless of Windows vs Unix, since it comes from the Rollup `id` |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// Note we do not need to use path.sep here because this filename contains | |
// a '/' regardless of Windows vs Unix, since it comes from the Rollup `id` | |
// Note we do not need to use path.sep with `specifier` because it contains | |
// a '/' regardless of Windows vs Unix, since it comes from the Rollup `id` | |
// We do use path.sep with `filename` because that is not normalized |
This comment should be updated, because now we do use path.sep
, but not always.
@@ -332,7 +332,7 @@ export default function lwc(pluginOptions: RollupLwcOptions = {}): Plugin { | |||
const [namespace, name] = | |||
// Note we do not need to use path.sep here because this filename contains | |||
// a '/' regardless of Windows vs Unix, since it comes from the Rollup `id` | |||
specifier?.split('/') ?? path.dirname(filename).split('/').slice(-2); | |||
specifier?.split('/') ?? path.dirname(filename).split(/[\\/]/).slice(-2); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
specifier?.split('/') ?? path.dirname(filename).split(/[\\/]/).slice(-2); | |
specifier?.split('/') ?? path.dirname(filename).split(path.sep).slice(-2); |
Are there any cases where we'd want to split on /
on windows or \
on unix?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think that just reverts #5038 🫠 so it would be a regression. path.sep is very unreliable.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Since the filename
is extracted from the parameter id
which could be a Rollup-normalized id
, a hard-coded string (e.g., @lwc/resources/empty_css.css
) or something generated by the custom resolveId
hook, it could use either \
or /
as the path delimiters on Windows. Therefore path.sep
will not work as expected here, I reckon?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, path.sep won't work. I'm not sure what the right fix is. I haven't used Windows since XP.
As per issue #5113 :
The original approach cannot correctly parse
namespace
andname
from file path in Windows as the path may contain\
as delimeters.It doesn't work well for relative paths start with
./
either which derives from implicit reference to CSS files from HTML templates.The fix still cannot extract the
namespace
for relative paths since there's only a./
. However, it does correctly identify thename
for the component.Details
Does this pull request introduce a breaking change?
Does this pull request introduce an observable change?