Skip to content
This repository has been archived by the owner on Apr 22, 2023. It is now read-only.

path: resolve normalize drive letter to lower case #8385

Closed
wants to merge 1 commit into from

Conversation

dead-horse
Copy link

make path.resolve work the same as path.normalize.

see https://github.com/joyent/node/blob/v0.12/lib/path.js#L180

@dead-horse
Copy link
Author

here make things inconsistency because process.pwd() and __dirname do not change to lower case. and actually i don't use windows, and dunno why we need to change device to lower case in path.normalize here,

also i think this will break somebody's code like:

var root = path.dirname(__dirname); // root = 'C:\\\\path'

var filepath = 'to/where';
var resolvedPath = path.join(root, filepath); // filepath = 'c:\\\\path\\to\\where';
if (resolvedPath.indexOf(root) !== 0) throw new Error('permission denied');

if we made this change, we should make all these things change together, and notify people in the document or at least in the changelog.

@dead-horse
Copy link
Author

ping @trevnorris, @refack added this one dead-horse#1

@refack
Copy link
Contributor

refack commented Sep 21, 2014

♬Than merge it ohh @dead-horse ♬ ohh @dead-horse ♬ ohh @dead-horse, ♯ merge it into your's ♭ . it'll propagate to here...

@@ -163,6 +163,11 @@ if (isWindows) {
resolvedTail = normalizeArray(resolvedTail.split(/[\\\/]+/).filter(f),
!resolvedAbsolute).join('\\');

// If device is a drive letter, we'll normalize to lower case.
if (resolvedDevice && resolvedDevice.charAt(1) === ':') {
resolvedDevice = resolvedDevice[0].toLowerCase() + resolvedDevice.substr(1);

Choose a reason for hiding this comment

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

I'm pretty sure the exceeds the 80 char limit. Also, drop the { }.

@trevnorris
Copy link

@tjfontaine @orangemocha Either one of you mind taking a look? I'm not the one to give a +1 on this type of Windows change.

@dead-horse dead-horse force-pushed the resolve branch 2 times, most recently from dc1293b to e9498e3 Compare September 23, 2014 01:22
@dead-horse
Copy link
Author

@refack merged

@refack
Copy link
Contributor

refack commented Sep 23, 2014

👍

#ifdef _WIN32
buf[0] = tolower(buf[0]);
#endif

Copy link
Contributor

Choose a reason for hiding this comment

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

What is the point of this change? process.cwd() makes no promise of normalizing the result, and I think this change might not be necessary. I'd rather keep it case-preserving unless there is a good reason to change that.

Copy link
Contributor

Choose a reason for hiding this comment

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

If we're normalizing paths, then as mentioned before, we need to normalize cwd() output, and thus reach consistency.
Anyway drive letter case is arbitrary, and depends only on the process's environment.

C:\>node -e console.log(process.cwd())
C:\

C:\>cmd /k cd "c:\temp"

c:\temp>node -e console.log(process.cwd())
c:\temp


It is semantically equivalent, but breaks allot of path comparisons as they are string based and case sensitive.

@orangemocha
Copy link
Contributor

The part about making path.resolve normalize the drive letter looks good to me. I an not convinced we should be normalizing the drive letter in process.cwd().

@@ -315,13 +315,15 @@ if (isWindows) {
// path.resolve tests
if (isWindows) {
// windows
var cwd = process.cwd();
cwd = cwd[0].toLowerCase() + cwd.substr(1);
Copy link
Contributor

Choose a reason for hiding this comment

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

change this to
cwd = path.normalize(cwd);
?

@dead-horse
Copy link
Author

@refack sorry i revert your commit. maybe better to create another PR with that commit.

@orangemocha done

@orangemocha
Copy link
Contributor

LGTM

@refack
Copy link
Contributor

refack commented Sep 27, 2014

@orangemocha RE: my commit comments. Don't you agree we should normalize across the board?

@orangemocha
Copy link
Contributor

@refack, your change addresses the scenario process.cwd() === path.resolve('.') but it doesn't address all the scenarios in which case-sensitive comparisons could be broken on Windows. You can set arbitrary casing in the process current directory by passing it to CreateProcess, and it will be reflected verbatim in process.cwd(). Doing a case sensitive comparison against anything coming from readdir will fail.

The bottom line is that case-sensitive path comparison is not the correct way of comparing paths on windows. And this arbitrary conversion to lower-case of the drive letter mitigates the problem only partially.

The first part of the pull request is a no brainer, because the docs state that path.resolve will normalize the result, and that is currently broken. But this other part raises all sorts of questions about how to do path comparisons correctly on Windows, for which I would like us to think of a more robust long-term solution.

@tjfontaine : thoughts? In terms of a more robust solution, I can think of the following:
a) Introducing an api for path comparison, eg path.equals() and path.startsWith(), that would do the right thing on all platforms. This is the truly portable solution, but it needs a new API.
b) Making path.normalize convert everything to lower-case (or upper-case). Not as robust as b), but it doesn't need a new API.

@refack
Copy link
Contributor

refack commented Sep 30, 2014

Agreed.
I thought about suggesting a new path comparison method, but who has the energy....

@orangemocha
Copy link
Contributor

@orangemocha
Copy link
Contributor

sorry, the last comment was meant for @dead-horse

make path.resolve work the same as path.normalize
@dead-horse
Copy link
Author

my bad. @orangemocha done

@orangemocha
Copy link
Contributor

Thank you, landed in f6e5740

@UltCombo
Copy link

@dead-horse @orangemocha Indeed, the driver letter casing inconsistency did break the minimatch matching in JSCS (jscs-dev/node-jscs#703).

Sorry if I'm late to the party, but when/why was it decided that drive letters should be normalized to lowercase? As far as I can see, this has mostly been just breaking existing code.

@dead-horse
Copy link
Author

This patch is because path.normalize normalize drive letter to lower case but path.resolve didn't. This bugged me in some situation.

@dead-horse dead-horse deleted the resolve branch October 22, 2014 01:13
@UltCombo
Copy link

@dead-horse makes sense. However, path.dirname still returns an uppercase drive letter (at least in Node 0.11.14) and that's causing a real issue for me, similar to the one you outlined in this comment.

@orangemocha
Copy link
Contributor

@UltCombo for lack for a better solution, I would suggest your normalize the result of path.dirname, or use case-insensitive comparison on Windows.

@UltCombo
Copy link

@orangemocha those were my thoughts as well, thanks. =]

@piscisaureus
Copy link

The patch a05f973 was landed to make some tests pass, but I can't really reconstruct what the motivation was. What I can tell is that the test failure wasn't really related to path.normalize() and path.resolve() [not] changing the drive letter case.

This patch has caused a lot of issues, including #7031 and #7806 and lots of bikeshedding about whether uppercase is the right case or lowercase.

I propose reverting the offending patch, and just doing what node does everywhere else (which is: path functions don't touch the case) and take another look at the cause of those test failures.

@UltCombo
Copy link

@piscisaureus +1 for reverting and not touching the case.

@llafuente
Copy link

We should use the same case as __dirname or process.pwd(), I'm not sure but I think I never saw it uppercased so assume lowercase.

Also means, always lowercase everywhere. No different behaviour between normalizer, resolve, dirname, __dirname, etc...

Not touching the case could lead to problems between modules, or even, your own code.

Path is windows are case-insensitive so developers should handle it properly.

mypath.toLowerCase().indexOf(test.path.toLowerCase())

I think this is a PITA, so I would even sugest to lowercase everything...

as @UltCombo suggest, I temporary revert and fix all the problem as a whole.

@orangemocha
Copy link
Contributor

I am not opposed to reverting a05f973, as it seems to have done more harm than good. However, I think that the real solution to all these issues is introducing a cross-platform path comparison API, and make it case-insensitive on Windows. I will work on a proposal (though likely not for another couple of weeks).

@UltCombo
Copy link

@llafuente @orangemocha worth noting that the require.cache is case sensitive, even on platforms that have case-insensitive paths. E.g.:

require('./a');
require('./A');

Both requires will resolve to the same file on Windows, initialize it twice and generate two entries in the require cache. The same issue would happen with different drive letter casing, or, actually, any casing discrepancy in the path.

I believe we can keep the original path casing, and do path comparisons in a case-insensitive manner in Windows, through a new cross-platform path comparison API as suggested by @orangemocha.

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.

8 participants