Skip to content

Commit

Permalink
path: refactor normalizeArray()
Browse files Browse the repository at this point in the history
The normalizeArray() function now avoids using the slow Array#splice()
method to improve performance and now also filters out empty path parts.

Code that pre-filtered empty parts has been removed.

PR-URL: nodejs/node-v0.x-archive#8724
Reviewed-by: Trevor Norris <[email protected]>
  • Loading branch information
nwoltman authored and trevnorris committed Nov 26, 2014
1 parent 9d9ed61 commit e0a0e91
Showing 1 changed file with 23 additions and 41 deletions.
64 changes: 23 additions & 41 deletions lib/path.js
Original file line number Diff line number Diff line change
Expand Up @@ -25,33 +25,30 @@ var util = require('util');


// resolves . and .. elements in a path array with directory names there
// must be no slashes, empty elements, or device names (c:\) in the array
// must be no slashes or device names (c:\) in the array
// (so also no leading and trailing slashes - it does not distinguish
// relative and absolute paths)
function normalizeArray(parts, allowAboveRoot) {
// if the path tries to go above the root, `up` ends up > 0
var up = 0;
for (var i = parts.length - 1; i >= 0; i--) {
var last = parts[i];
if (last === '.') {
parts.splice(i, 1);
} else if (last === '..') {
parts.splice(i, 1);
up++;
} else if (up) {
parts.splice(i, 1);
up--;
}
}
var res = [];
for (var i = 0; i < parts.length; i++) {
var p = parts[i];

// ignore empty parts
if (!p || p === '.')
continue;

// if the path is allowed to go above the root, restore leading ..s
if (allowAboveRoot) {
for (; up--; up) {
parts.unshift('..');
if (p === '..') {
if (res.length && res[res.length - 1] !== '..') {
res.pop();
} else if (allowAboveRoot) {
res.push('..');
}
} else {
res.push(p);
}
}

return parts;
return res;
}

// Regex to split a windows path into three parts: [*, device, slash,
Expand Down Expand Up @@ -153,12 +150,7 @@ win32.resolve = function() {
// fails)

// Normalize the tail path

function f(p) {
return !!p;
}

resolvedTail = normalizeArray(resolvedTail.split(/[\\\/]+/).filter(f),
resolvedTail = normalizeArray(resolvedTail.split(/[\\\/]+/),
!resolvedAbsolute).join('\\');

// If device is a drive letter, we'll normalize to lower case.
Expand Down Expand Up @@ -186,9 +178,7 @@ win32.normalize = function(path) {
}

// Normalize the tail path
tail = normalizeArray(tail.split(/[\\\/]+/).filter(function(p) {
return !!p;
}), !isAbsolute).join('\\');
tail = normalizeArray(tail.split(/[\\\/]+/), !isAbsolute).join('\\');

if (!tail && !isAbsolute) {
tail = '.';
Expand Down Expand Up @@ -453,9 +443,8 @@ posix.resolve = function() {
// handle relative paths to be safe (might happen when process.cwd() fails)

// Normalize the path
resolvedPath = normalizeArray(resolvedPath.split('/').filter(function(p) {
return !!p;
}), !resolvedAbsolute).join('/');
resolvedPath = normalizeArray(resolvedPath.split('/'),
!resolvedAbsolute).join('/');

return ((resolvedAbsolute ? '/' : '') + resolvedPath) || '.';
};
Expand All @@ -464,17 +453,10 @@ posix.resolve = function() {
// posix version
posix.normalize = function(path) {
var isAbsolute = posix.isAbsolute(path),
trailingSlash = path.substr(-1) === '/',
segments = path.split('/'),
nonEmptySegments = [];
trailingSlash = path.substr(-1) === '/';

// Normalize the path
for (var i = 0; i < segments.length; i++) {
if (segments[i]) {
nonEmptySegments.push(segments[i]);
}
}
path = normalizeArray(nonEmptySegments, !isAbsolute).join('/');
path = normalizeArray(path.split('/'), !isAbsolute).join('/');

if (!path && !isAbsolute) {
path = '.';
Expand Down

0 comments on commit e0a0e91

Please sign in to comment.