Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

Commit 736c8fb

Browse files
committed
refactor($location): move file://+win path fix to $location
The urlResolve method was fixed to automatically remove the volume label from path names to fix issues with the file protocol on windows where $location.path() was returning paths where the first segment would be the volume name, such as "/C:/mypath". See #4942 and #4928 However, the solution was specific to the $location non- HTML5 mode, and was implemented at a lower level of abstraction than it should have been. This refactor moves the fix to inside of the LocationHashBangUrl $$parse method. Closes #5041
1 parent 9475622 commit 736c8fb

File tree

4 files changed

+53
-47
lines changed

4 files changed

+53
-47
lines changed

src/ng/location.js

+40
Original file line numberDiff line numberDiff line change
@@ -179,7 +179,47 @@ function LocationHashbangUrl(appBase, hashPrefix) {
179179
hashPrefix);
180180
}
181181
parseAppUrl(withoutHashUrl, this, appBase);
182+
183+
this.$$path = removeWindowsDriveName(this.$$path, withoutHashUrl, appBase);
184+
182185
this.$$compose();
186+
187+
/*
188+
* In Windows, on an anchor node on documents loaded from
189+
* the filesystem, the browser will return a pathname
190+
* prefixed with the drive name ('/C:/path') when a
191+
* pathname without a drive is set:
192+
* * a.setAttribute('href', '/foo')
193+
* * a.pathname === '/C:/foo' //true
194+
*
195+
* Inside of Angular, we're always using pathnames that
196+
* do not include drive names for routing.
197+
*/
198+
function removeWindowsDriveName (path, url, base) {
199+
/*
200+
Matches paths for file protocol on windows,
201+
such as /C:/foo/bar, and captures only /foo/bar.
202+
*/
203+
var windowsFilePathExp = /^\/?.*?:(\/.*)/;
204+
205+
var firstPathSegmentMatch;
206+
207+
//Get the relative path from the input URL.
208+
if (url.indexOf(base) === 0) {
209+
url = url.replace(base, '');
210+
}
211+
212+
/*
213+
* The input URL intentionally contains a
214+
* first path segment that ends with a colon.
215+
*/
216+
if (windowsFilePathExp.exec(url)) {
217+
return path;
218+
}
219+
220+
firstPathSegmentMatch = windowsFilePathExp.exec(path);
221+
return firstPathSegmentMatch ? firstPathSegmentMatch[1] : path;
222+
}
183223
};
184224

185225
/**

src/ng/urlUtils.js

+4-44
Original file line numberDiff line numberDiff line change
@@ -7,11 +7,6 @@
77
// exactly the behavior needed here. There is little value is mocking these out for this
88
// service.
99
var urlParsingNode = document.createElement("a");
10-
/*
11-
Matches paths for file protocol on windows,
12-
such as /C:/foo/bar, and captures only /foo/bar.
13-
*/
14-
var windowsFilePathExp = /^\/?.*?:(\/.*)/;
1510
var originUrl = urlResolve(window.location.href, true);
1611

1712

@@ -68,8 +63,7 @@ var originUrl = urlResolve(window.location.href, true);
6863
*
6964
*/
7065
function urlResolve(url, base) {
71-
var href = url,
72-
pathname;
66+
var href = url;
7367

7468
if (msie) {
7569
// Normalize before parse. Refer Implementation Notes on why this is
@@ -80,21 +74,6 @@ function urlResolve(url, base) {
8074

8175
urlParsingNode.setAttribute('href', href);
8276

83-
/*
84-
* In Windows, on an anchor node on documents loaded from
85-
* the filesystem, the browser will return a pathname
86-
* prefixed with the drive name ('/C:/path') when a
87-
* pathname without a drive is set:
88-
* * a.setAttribute('href', '/foo')
89-
* * a.pathname === '/C:/foo' //true
90-
*
91-
* Inside of Angular, we're always using pathnames that
92-
* do not include drive names for routing.
93-
*/
94-
95-
pathname = removeWindowsDriveName(urlParsingNode.pathname, url, base);
96-
pathname = (pathname.charAt(0) === '/') ? pathname : '/' + pathname;
97-
9877
// urlParsingNode provides the UrlUtils interface - http://url.spec.whatwg.org/#urlutils
9978
return {
10079
href: urlParsingNode.href,
@@ -104,11 +83,12 @@ function urlResolve(url, base) {
10483
hash: urlParsingNode.hash ? urlParsingNode.hash.replace(/^#/, '') : '',
10584
hostname: urlParsingNode.hostname,
10685
port: urlParsingNode.port,
107-
pathname: pathname
86+
pathname: (urlParsingNode.pathname.charAt(0) === '/')
87+
? urlParsingNode.pathname
88+
: '/' + urlParsingNode.pathname
10889
};
10990
}
11091

111-
11292
/**
11393
* Parse a request URL and determine whether this is a same-origin request as the application document.
11494
*
@@ -121,23 +101,3 @@ function urlIsSameOrigin(requestUrl) {
121101
return (parsed.protocol === originUrl.protocol &&
122102
parsed.host === originUrl.host);
123103
}
124-
125-
function removeWindowsDriveName (path, url, base) {
126-
var firstPathSegmentMatch;
127-
128-
//Get the relative path from the input URL.
129-
if (url.indexOf(base) === 0) {
130-
url = url.replace(base, '');
131-
}
132-
133-
/*
134-
* The input URL intentionally contains a
135-
* first path segment that ends with a colon.
136-
*/
137-
if (windowsFilePathExp.exec(url)) {
138-
return path;
139-
}
140-
141-
firstPathSegmentMatch = windowsFilePathExp.exec(path);
142-
return firstPathSegmentMatch ? firstPathSegmentMatch[1] : path;
143-
}

test/ng/locationSpec.js

+1-2
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,6 @@ describe('$location', function() {
3838
if ($sniffer.msie) return;
3939
//reset urlParsingNode
4040
urlParsingNode = urlParsingNodePlaceholder;
41-
expect(urlParsingNode.pathname).not.toBe('/C:/foo');
4241
}));
4342

4443

@@ -324,7 +323,7 @@ describe('$location', function() {
324323
});
325324

326325

327-
it('should parse hashband url into path and search', function() {
326+
it('should parse hashbang url into path and search', function() {
328327
expect(url.protocol()).toBe('http');
329328
expect(url.host()).toBe('www.server.org');
330329
expect(url.port()).toBe(1234);

test/ng/urlUtilsSpec.js

+8-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
'use strict';
22

33
describe('urlUtils', function() {
4-
describe('parse', function() {
4+
describe('urlResolve', function() {
55
it('should normalize a relative url', function () {
66
expect(urlResolve("foo").href).toMatch(/^https?:\/\/[^/]+\/foo$/);
77
});
@@ -14,6 +14,13 @@ describe('urlUtils', function() {
1414
expect(parsed.hostname).not.toBe("");
1515
expect(parsed.pathname).not.toBe("");
1616
});
17+
18+
19+
it('should return pathname as / if empty path provided', function () {
20+
//IE counts / as empty, necessary to use / so that pathname is not context.html
21+
var parsed = urlResolve('/');
22+
expect(parsed.pathname).toBe('/');
23+
})
1724
});
1825

1926
describe('isSameOrigin', function() {

0 commit comments

Comments
 (0)