160 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			160 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /*
 | |
|  Copyright 2012-2015, Yahoo Inc.
 | |
|  Copyrights licensed under the New BSD License. See the accompanying LICENSE file for terms.
 | |
|  */
 | |
| 'use strict';
 | |
| 
 | |
| const path = require('path');
 | |
| let parsePath = path.parse;
 | |
| let SEP = path.sep || /* istanbul ignore next */ '/';
 | |
| const origParser = parsePath;
 | |
| const origSep = SEP;
 | |
| 
 | |
| function makeRelativeNormalizedPath(str, sep) {
 | |
|     const parsed = parsePath(str);
 | |
|     let root = parsed.root;
 | |
|     let dir;
 | |
|     let file = parsed.base;
 | |
|     let quoted;
 | |
|     let pos;
 | |
| 
 | |
|     // handle a weird windows case separately
 | |
|     if (sep === '\\') {
 | |
|         pos = root.indexOf(':\\');
 | |
|         if (pos >= 0) {
 | |
|             root = root.substring(0, pos + 2);
 | |
|         }
 | |
|     }
 | |
|     dir = parsed.dir.substring(root.length);
 | |
| 
 | |
|     if (str === '') {
 | |
|         return [];
 | |
|     }
 | |
| 
 | |
|     if (sep !== '/') {
 | |
|         quoted = new RegExp(sep.replace(/\W/g, '\\$&'), 'g');
 | |
|         dir = dir.replace(quoted, '/');
 | |
|         file = file.replace(quoted, '/'); // excessively paranoid?
 | |
|     }
 | |
| 
 | |
|     if (dir !== '') {
 | |
|         dir = dir + '/' + file;
 | |
|     } else {
 | |
|         dir = file;
 | |
|     }
 | |
|     if (dir.substring(0, 1) === '/') {
 | |
|         dir = dir.substring(1);
 | |
|     }
 | |
|     dir = dir.split(/\/+/);
 | |
|     return dir;
 | |
| }
 | |
| 
 | |
| function Path(strOrArray) {
 | |
|     if (Array.isArray(strOrArray)) {
 | |
|         this.v = strOrArray;
 | |
|     } else if (typeof strOrArray === 'string') {
 | |
|         this.v = makeRelativeNormalizedPath(strOrArray, SEP);
 | |
|     } else {
 | |
|         throw new Error(
 | |
|             'Invalid Path argument must be string or array:' + strOrArray
 | |
|         );
 | |
|     }
 | |
| }
 | |
| 
 | |
| Path.prototype.toString = function() {
 | |
|     return this.v.join('/');
 | |
| };
 | |
| 
 | |
| Path.prototype.hasParent = function() {
 | |
|     return this.v.length > 0;
 | |
| };
 | |
| 
 | |
| Path.prototype.parent = function() {
 | |
|     if (!this.hasParent()) {
 | |
|         throw new Error('Unable to get parent for 0 elem path');
 | |
|     }
 | |
|     const p = this.v.slice();
 | |
|     p.pop();
 | |
|     return new Path(p);
 | |
| };
 | |
| 
 | |
| Path.prototype.elements = function() {
 | |
|     return this.v.slice();
 | |
| };
 | |
| 
 | |
| Path.prototype.contains = function(other) {
 | |
|     let i;
 | |
|     if (other.length > this.length) {
 | |
|         return false;
 | |
|     }
 | |
|     for (i = 0; i < other.length; i += 1) {
 | |
|         if (this.v[i] !== other.v[i]) {
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
|     return true;
 | |
| };
 | |
| 
 | |
| Path.prototype.ancestorOf = function(other) {
 | |
|     return other.contains(this) && other.length !== this.length;
 | |
| };
 | |
| 
 | |
| Path.prototype.descendantOf = function(other) {
 | |
|     return this.contains(other) && other.length !== this.length;
 | |
| };
 | |
| 
 | |
| Path.prototype.commonPrefixPath = function(other) {
 | |
|     const len = this.length > other.length ? other.length : this.length;
 | |
|     let i;
 | |
|     const ret = [];
 | |
| 
 | |
|     for (i = 0; i < len; i += 1) {
 | |
|         if (this.v[i] === other.v[i]) {
 | |
|             ret.push(this.v[i]);
 | |
|         } else {
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
|     return new Path(ret);
 | |
| };
 | |
| 
 | |
| ['push', 'pop', 'shift', 'unshift', 'splice'].forEach(f => {
 | |
|     Path.prototype[f] = function(...args) {
 | |
|         const v = this.v;
 | |
|         return v[f](...args);
 | |
|     };
 | |
| });
 | |
| 
 | |
| Path.compare = function(a, b) {
 | |
|     const al = a.length;
 | |
|     const bl = b.length;
 | |
|     if (al < bl) {
 | |
|         return -1;
 | |
|     }
 | |
|     if (al > bl) {
 | |
|         return 1;
 | |
|     }
 | |
|     const astr = a.toString();
 | |
|     const bstr = b.toString();
 | |
|     return astr < bstr ? -1 : astr > bstr ? 1 : 0;
 | |
| };
 | |
| 
 | |
| Object.defineProperty(Path.prototype, 'length', {
 | |
|     enumerable: true,
 | |
|     get() {
 | |
|         return this.v.length;
 | |
|     }
 | |
| });
 | |
| 
 | |
| module.exports = Path;
 | |
| Path.tester = {
 | |
|     setParserAndSep(p, sep) {
 | |
|         parsePath = p;
 | |
|         SEP = sep;
 | |
|     },
 | |
|     reset() {
 | |
|         parsePath = origParser;
 | |
|         SEP = origSep;
 | |
|     }
 | |
| };
 |