You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							233 lines
						
					
					
						
							7.3 KiB
						
					
					
				
			
		
		
	
	
							233 lines
						
					
					
						
							7.3 KiB
						
					
					
				'use strict'; | 
						|
 | 
						|
var test = require('tape'); | 
						|
 | 
						|
var path = require('path'); | 
						|
var fs = require('fs'); | 
						|
var os = require('os'); | 
						|
 | 
						|
var which = require('which'); | 
						|
 | 
						|
var npmPath = require('../'); | 
						|
 | 
						|
var SEP = npmPath.SEPARATOR; | 
						|
var PATH = npmPath.PATH; | 
						|
 | 
						|
var level0 = path.join(__dirname, 'fixture', 'level0'); | 
						|
var level1 = path.join(level0, 'node_modules', 'level1'); | 
						|
var level2 = path.join(level1, 'node_modules', 'level2'); | 
						|
 | 
						|
var level = [level0, level1, level2]; | 
						|
var binPath = level.map(function (levelPath) { | 
						|
  return path.join(levelPath, 'node_modules', '.bin'); | 
						|
}); | 
						|
 | 
						|
test('exports separator', function (t) { | 
						|
  t.ok(npmPath.SEPARATOR); | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('exports $PATH key', function (t) { | 
						|
  t.ok(npmPath.PATH); | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('includes current node executable dir', function (t) { | 
						|
  var level0Path = npmPath.getSync({ cwd: level0 }); | 
						|
  t.notEqual(level0Path.indexOf(path.dirname(process.execPath) + SEP), -1); | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('async version works', function (t) { | 
						|
  var isAsync = false; | 
						|
  npmPath.get({ cwd: level0 }, function (err, level0Path) { | 
						|
    t.ifError(err); | 
						|
    t.ok(isAsync); | 
						|
    t.notEqual(level0Path.indexOf(path.dirname(process.execPath) + SEP), -1); | 
						|
    t.end(); | 
						|
  }); | 
						|
  isAsync = true; // can only be set if above callback not synchronous | 
						|
}); | 
						|
 | 
						|
test('no fn == sync', function (t) { | 
						|
  var level0Path = npmPath.get({ cwd: level0 }); | 
						|
  t.notEqual(level0Path.indexOf(path.dirname(process.execPath) + SEP), -1); | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('sync options is optional', function (t) { | 
						|
  var newPath = npmPath.get(); | 
						|
  t.notEqual(newPath.indexOf(path.dirname(process.execPath) + SEP), -1); | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('async options is optional', function (t) { | 
						|
  var isAsync = false; | 
						|
  npmPath.get(function (err, newPath) { | 
						|
    t.ifError(err); | 
						|
    t.notEqual(newPath.indexOf(path.dirname(process.execPath) + SEP), -1); | 
						|
    t.ok(isAsync); | 
						|
    t.end(); | 
						|
  }); | 
						|
  isAsync = true; // can only be set if above callback not synchronous | 
						|
}); | 
						|
 | 
						|
test('includes bin from sibling dirs', { skip: true }, function (t) { | 
						|
  t.test('from existing sibling directory', function (t) { | 
						|
    var level1Path = npmPath.getSync({ cwd: path.join(level[0], 'test') }); | 
						|
    t.notEqual(level1Path.indexOf(binPath[0] + SEP), -1, 'should include level 0 .bin'); | 
						|
    t.equal(level1Path.indexOf(binPath[2] + SEP), -1, 'should not include child paths'); | 
						|
    t.end(); | 
						|
  }); | 
						|
 | 
						|
  t.test('from existing sibling directory async', function (t) { | 
						|
    npmPath({ cwd: path.join(level[0], 'test') }, function (err, level1Path) { | 
						|
      t.ifError(err); | 
						|
      t.notEqual(level1Path.indexOf(binPath[0] + SEP), -1, 'should include level 0 .bin'); | 
						|
      t.equal(level1Path.indexOf(binPath[2] + SEP), -1, 'should not include child paths'); | 
						|
      t.end(); | 
						|
    }); | 
						|
  }); | 
						|
}); | 
						|
 | 
						|
test('includes all .bin dirs in all parent node_modules folders', function (t) { | 
						|
  t.test('no nesting', function (t) { | 
						|
    var level0Path = npmPath.getSync({ cwd: level[0] }); | 
						|
    t.notEqual(level0Path.indexOf(binPath[0] + SEP), -1, 'should include level 0 .bin'); | 
						|
    t.equal(level0Path.indexOf(binPath[1] + SEP), -1, 'should not include child paths'); | 
						|
    t.equal(level0Path.indexOf(binPath[2] + SEP), -1, 'should not include child paths'); | 
						|
    t.end(); | 
						|
  }); | 
						|
 | 
						|
  t.test('1 level of nesting', function (t) { | 
						|
    var level1Path = npmPath.getSync({ cwd: level[1] }); | 
						|
    t.notEqual(level1Path.indexOf(binPath[0] + SEP), -1, 'should include level 0 .bin'); | 
						|
    t.notEqual(level1Path.indexOf(binPath[1] + SEP), -1, 'should include level 1 .bin'); | 
						|
    t.equal(level1Path.indexOf(binPath[2] + SEP), -1, 'should not include child paths'); | 
						|
    t.end(); | 
						|
  }); | 
						|
 | 
						|
  t.test('2 levels of nesting', function (t) { | 
						|
    var level1Path = npmPath.getSync({ cwd: level[2] }); | 
						|
    t.notEqual(level1Path.indexOf(binPath[0] + SEP), -1, 'should include level 0 .bin'); | 
						|
    t.notEqual(level1Path.indexOf(binPath[1] + SEP), -1, 'should include level 1 .bin'); | 
						|
    t.notEqual(level1Path.indexOf(binPath[2] + SEP), -1, 'should include level 2 .bin'); | 
						|
    t.end(); | 
						|
  }); | 
						|
 | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('handles directories with node_modules in the name', function (t) { | 
						|
  var trickyL0 = level[0].replace('level0', 'level0_node_modules'); | 
						|
  var trickyL1 = level[1].replace('level0', 'level0_node_modules'); | 
						|
  var trickyL2 = level[2].replace('level0', 'level0_node_modules'); | 
						|
 | 
						|
  t.test('no nesting', function (t) { | 
						|
    var level0Path = npmPath.getSync({ cwd: trickyL0 }); | 
						|
    t.notEqual(level0Path.indexOf(path.join(trickyL0, 'node_modules', '.bin') + SEP), -1, 'should include level 0 .bin'); | 
						|
    t.end(); | 
						|
  }); | 
						|
 | 
						|
  t.test('1 level of nesting', function (t) { | 
						|
    var level1Path = npmPath.getSync({ cwd: trickyL1 }); | 
						|
 | 
						|
    t.notEqual(level1Path.indexOf(path.join(trickyL0, 'node_modules', '.bin') + SEP), -1, 'should include level 0 .bin'); | 
						|
    t.notEqual(level1Path.indexOf(path.join(trickyL1, 'node_modules', '.bin') + SEP), -1, 'should include level 1 .bin'); | 
						|
    t.end(); | 
						|
  }); | 
						|
 | 
						|
  t.test('2 levels of nesting', function (t) { | 
						|
    var level2Path = npmPath.getSync({ cwd: trickyL2 }); | 
						|
 | 
						|
    t.notEqual(level2Path.indexOf(path.join(trickyL0, 'node_modules', '.bin') + SEP), -1, 'should include level 0 .bin'); | 
						|
    t.notEqual(level2Path.indexOf(path.join(trickyL1, 'node_modules', '.bin') + SEP), -1, 'should include level 1 .bin'); | 
						|
    t.notEqual(level2Path.indexOf(path.join(trickyL2, 'node_modules', '.bin') + SEP), -1, 'should include level 1 .bin'); | 
						|
    t.end(); | 
						|
  }); | 
						|
 | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('can set path', function (t) { | 
						|
  var oldPath = process.env[PATH]; | 
						|
  npmPath.set.sync(); | 
						|
  var newPath = process.env[PATH]; | 
						|
  t.notDeepEqual(oldPath, newPath); | 
						|
  process.env[PATH] = oldPath; | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('includes node-gyp bundled with current npm', { skip: true }, function (t) { | 
						|
  var oldPath = process.env[PATH]; | 
						|
  npmPath(); | 
						|
  var newGypPath = which.sync('node-gyp'); | 
						|
  t.ok(newGypPath); | 
						|
  t.ok(fs.existsSync(newGypPath)); | 
						|
  t.notEqual(newGypPath.indexOf(path.join('npm', 'bin', 'node-gyp-bin') + SEP), -1); | 
						|
  process.env[PATH] = oldPath; | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('remove duplicated entries', function (t) { | 
						|
  var pathWithDuplicates = '/bin:/sbin:/bin'; | 
						|
 | 
						|
  var newPath = npmPath({ env: { PATH: pathWithDuplicates } }).split(':'); | 
						|
 | 
						|
  // Set size is equal to array length if there are no duplicates | 
						|
  t.equal(newPath.length, new Set(newPath).size); | 
						|
 | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('can set path to npm root to use for node-gyp lookup', { skip: true }, function (t) { | 
						|
  var oldPath = process.env[PATH]; | 
						|
  var pathToNpm = path.resolve(fs.realpathSync(which.sync('npm')), '..', '..'); | 
						|
 | 
						|
  var tmpFile = path.join(os.tmpdir(), 'npm-path-custom-npm'); | 
						|
  try { | 
						|
    fs.unlinkSync(tmpFile); | 
						|
  } catch (err) { | 
						|
    err; // ignore | 
						|
  } | 
						|
 | 
						|
  fs.linkSync(pathToNpm, tmpFile); | 
						|
  var newPath = npmPath.get({ | 
						|
    npm: tmpFile | 
						|
  }); | 
						|
  t.notEqual(newPath.indexOf(path.join(tmpFile, 'bin', 'node-gyp-bin') + SEP), -1); | 
						|
  process.env[PATH] = oldPath; | 
						|
  fs.unlinkSync(tmpFile); | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('error if passing bad path to npm root', { skip: true }, function (t) { | 
						|
  var oldPath = process.env[PATH]; | 
						|
 | 
						|
  var tmpFile = path.join(os.tmpdir(), 'npm-path-custom-npm'); | 
						|
  try { | 
						|
    fs.unlinkSync(tmpFile); | 
						|
  } catch (err) { | 
						|
    err; // ignore | 
						|
  } | 
						|
  t.throws(function () { | 
						|
    npmPath.get({ | 
						|
      npm: tmpFile | 
						|
    }); | 
						|
  }); | 
						|
  process.env[PATH] = oldPath; | 
						|
  t.end(); | 
						|
}); | 
						|
 | 
						|
test('no error if no npm on existing path', function (t) { | 
						|
  var oldPath = process.env[PATH]; | 
						|
 | 
						|
  process.env[PATH] = ''; | 
						|
 | 
						|
  var newPath = npmPath.get(); | 
						|
 | 
						|
  t.equal(newPath.indexOf(path.join('bin', 'node-gyp-bin') + SEP), -1); | 
						|
 | 
						|
  process.env[PATH] = oldPath; | 
						|
  t.end(); | 
						|
}); |