|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169 |
- /*
- 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;
- 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;
- }
-
- class Path {
- constructor(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}`
- );
- }
- }
-
- toString() {
- return this.v.join('/');
- }
-
- hasParent() {
- return this.v.length > 0;
- }
-
- parent() {
- 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);
- }
-
- elements() {
- return this.v.slice();
- }
-
- name() {
- return this.v.slice(-1)[0];
- }
-
- contains(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;
- }
-
- ancestorOf(other) {
- return other.contains(this) && other.length !== this.length;
- }
-
- descendantOf(other) {
- return this.contains(other) && other.length !== this.length;
- }
-
- commonPrefixPath(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);
- }
-
- static compare(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;
- }
- }
-
- ['push', 'pop', 'shift', 'unshift', 'splice'].forEach(fn => {
- Object.defineProperty(Path.prototype, fn, {
- value(...args) {
- return this.v[fn](...args);
- }
- });
- });
-
- 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;
- }
- };
|