From b488be127a8cf1e59eb257db3f8eaf6efdb0f275 Mon Sep 17 00:00:00 2001 From: Ryan Dahl Date: Mon, 21 Nov 2011 09:48:45 -0800 Subject: [PATCH] Include NPM, update .pkg to install it. .msi update coming soon. --- LICENSE | 6 + deps/npm/.gitignore | 15 + deps/npm/.gitmodules | 51 + deps/npm/.npmignore | 11 + deps/npm/AUTHORS | 50 + deps/npm/CHANGES | 1 + deps/npm/LICENSE | 61 + deps/npm/Makefile | 125 ++ deps/npm/README.md | 274 ++++ deps/npm/bin/npm-cli.js | 77 ++ deps/npm/bin/npm-g.cmd | 6 + deps/npm/bin/npm-get-uid-gid.js | 16 + deps/npm/bin/npm.cmd | 6 + deps/npm/bin/npm_g.cmd | 6 + deps/npm/bin/read-package-json.js | 22 + deps/npm/cli.js | 2 + deps/npm/configure | 33 + deps/npm/doc/api/author.md | 1 + deps/npm/doc/api/bin.md | 13 + deps/npm/doc/api/bugs.md | 19 + deps/npm/doc/api/commands.md | 22 + deps/npm/doc/api/config.md | 45 + deps/npm/doc/api/deprecate.md | 32 + deps/npm/doc/api/docs.md | 19 + deps/npm/doc/api/edit.md | 24 + deps/npm/doc/api/explore.md | 18 + deps/npm/doc/api/find.md | 1 + deps/npm/doc/api/get.md | 1 + deps/npm/doc/api/help-search.md | 30 + deps/npm/doc/api/home.md | 1 + deps/npm/doc/api/init.md | 29 + deps/npm/doc/api/install.md | 19 + deps/npm/doc/api/link.md | 33 + deps/npm/doc/api/list.md | 1 + deps/npm/doc/api/ln.md | 1 + deps/npm/doc/api/load.md | 26 + deps/npm/doc/api/ls.md | 50 + deps/npm/doc/api/npm.md | 115 ++ deps/npm/doc/api/outdated.md | 13 + deps/npm/doc/api/owner.md | 31 + deps/npm/doc/api/pack.md | 19 + deps/npm/doc/api/prefix.md | 15 + deps/npm/doc/api/prune.md | 17 + deps/npm/doc/api/publish.md | 30 + deps/npm/doc/api/rebuild.md | 16 + deps/npm/doc/api/restart.md | 22 + deps/npm/doc/api/rm.md | 1 + deps/npm/doc/api/root.md | 15 + deps/npm/doc/api/run-script.md | 27 + deps/npm/doc/api/search.md | 35 + deps/npm/doc/api/set.md | 1 + deps/npm/doc/api/start.md | 13 + deps/npm/doc/api/stop.md | 13 + deps/npm/doc/api/submodule.md | 28 + deps/npm/doc/api/tag.md | 23 + deps/npm/doc/api/test.md | 16 + deps/npm/doc/api/uninstall.md | 16 + deps/npm/doc/api/unpublish.md | 20 + deps/npm/doc/api/update.md | 11 + deps/npm/doc/api/version.md | 18 + deps/npm/doc/api/view.md | 93 ++ deps/npm/doc/api/whoami.md | 15 + deps/npm/doc/cli/adduser.md | 36 + deps/npm/doc/cli/author.md | 1 + deps/npm/doc/cli/bin.md | 17 + deps/npm/doc/cli/bugs.md | 38 + deps/npm/doc/cli/build.md | 22 + deps/npm/doc/cli/bundle.md | 14 + deps/npm/doc/cli/cache.md | 70 + deps/npm/doc/cli/changelog.md | 36 + deps/npm/doc/cli/coding-style.md | 190 +++ deps/npm/doc/cli/completion.md | 29 + deps/npm/doc/cli/config.md | 665 +++++++++ deps/npm/doc/cli/deprecate.md | 24 + deps/npm/doc/cli/developers.md | 172 +++ deps/npm/doc/cli/docs.md | 38 + deps/npm/doc/cli/edit.md | 35 + deps/npm/doc/cli/explore.md | 40 + deps/npm/doc/cli/faq.md | 223 ++++ deps/npm/doc/cli/find.md | 1 + deps/npm/doc/cli/folders.md | 209 +++ deps/npm/doc/cli/get.md | 1 + deps/npm/doc/cli/global.md | 1 + deps/npm/doc/cli/help-search.md | 35 + deps/npm/doc/cli/help.md | 38 + deps/npm/doc/cli/home.md | 1 + deps/npm/doc/cli/init.md | 24 + deps/npm/doc/cli/install.md | 201 +++ deps/npm/doc/cli/json.md | 472 +++++++ deps/npm/doc/cli/link.md | 57 + deps/npm/doc/cli/list.md | 55 + deps/npm/doc/cli/ln.md | 1 + deps/npm/doc/cli/ls.md | 1 + deps/npm/doc/cli/npm.md | 155 +++ deps/npm/doc/cli/outdated.md | 17 + deps/npm/doc/cli/owner.md | 32 + deps/npm/doc/cli/pack.md | 25 + deps/npm/doc/cli/prefix.md | 17 + deps/npm/doc/cli/prune.md | 21 + deps/npm/doc/cli/publish.md | 30 + deps/npm/doc/cli/rebuild.md | 20 + deps/npm/doc/cli/registry.md | 92 ++ deps/npm/doc/cli/removing-npm.md | 54 + deps/npm/doc/cli/restart.md | 22 + deps/npm/doc/cli/rm.md | 1 + deps/npm/doc/cli/root.md | 17 + deps/npm/doc/cli/run-script.md | 21 + deps/npm/doc/cli/scripts.md | 182 +++ deps/npm/doc/cli/search.md | 39 + deps/npm/doc/cli/semver.md | 130 ++ deps/npm/doc/cli/set.md | 1 + deps/npm/doc/cli/star.md | 22 + deps/npm/doc/cli/start.md | 18 + deps/npm/doc/cli/stop.md | 18 + deps/npm/doc/cli/submodule.md | 28 + deps/npm/doc/cli/tag.md | 17 + deps/npm/doc/cli/test.md | 21 + deps/npm/doc/cli/uninstall.md | 19 + deps/npm/doc/cli/unpublish.md | 32 + deps/npm/doc/cli/update.md | 21 + deps/npm/doc/cli/version.md | 27 + deps/npm/doc/cli/view.md | 85 ++ deps/npm/doc/cli/whoami.md | 15 + deps/npm/html/api/GubbleBum-Blocky.ttf | Bin 0 -> 10292 bytes deps/npm/html/api/style.css | 336 +++++ deps/npm/html/doc/GubbleBum-Blocky.ttf | Bin 0 -> 10292 bytes deps/npm/html/doc/style.css | 336 +++++ deps/npm/html/docfoot.html | 34 + deps/npm/html/dochead.html | 8 + deps/npm/html/favicon.ico | Bin 0 -> 7094 bytes deps/npm/html/index.html | 86 ++ deps/npm/html/n-64.png | Bin 0 -> 679 bytes deps/npm/html/n-large.png | Bin 0 -> 699 bytes deps/npm/html/npm-16.png | Bin 0 -> 145 bytes deps/npm/html/npm-256-square.png | Bin 0 -> 3290 bytes deps/npm/html/npm-256w.png | Bin 0 -> 3169 bytes deps/npm/html/npm-64-square.png | Bin 0 -> 2908 bytes deps/npm/html/npm-fin.png | Bin 0 -> 47789 bytes deps/npm/html/npm-large.png | Bin 0 -> 483909 bytes deps/npm/html/npm.png | Bin 0 -> 1164 bytes deps/npm/lib/adduser.js | 50 + deps/npm/lib/bin.js | 19 + deps/npm/lib/bugs.js | 50 + deps/npm/lib/build.js | 186 +++ deps/npm/lib/cache.js | 735 ++++++++++ deps/npm/lib/completion.js | 245 ++++ deps/npm/lib/config.js | 286 ++++ deps/npm/lib/deprecate.js | 48 + deps/npm/lib/docs.js | 46 + deps/npm/lib/edit.js | 31 + deps/npm/lib/explore.js | 34 + deps/npm/lib/faq.js | 8 + deps/npm/lib/get.js | 12 + deps/npm/lib/help-search.js | 183 +++ deps/npm/lib/help.js | 150 +++ deps/npm/lib/init.js | 240 ++++ deps/npm/lib/install.js | 608 +++++++++ deps/npm/lib/link.js | 158 +++ deps/npm/lib/ls.js | 169 +++ deps/npm/lib/npm.js | 448 +++++++ deps/npm/lib/outdated.js | 146 ++ deps/npm/lib/owner.js | 188 +++ deps/npm/lib/pack.js | 66 + deps/npm/lib/prefix.js | 10 + deps/npm/lib/prune.js | 40 + deps/npm/lib/publish.js | 166 +++ deps/npm/lib/rebuild.js | 68 + deps/npm/lib/restart.js | 1 + deps/npm/lib/root.js | 11 + deps/npm/lib/run-script.js | 100 ++ deps/npm/lib/search.js | 222 ++++ deps/npm/lib/set.js | 12 + deps/npm/lib/star.js | 34 + deps/npm/lib/start.js | 1 + deps/npm/lib/stop.js | 1 + deps/npm/lib/submodule.js | 93 ++ deps/npm/lib/tag.js | 17 + deps/npm/lib/test.js | 1 + deps/npm/lib/unbuild.js | 85 ++ deps/npm/lib/uninstall.js | 55 + deps/npm/lib/unpublish.js | 69 + deps/npm/lib/update.js | 39 + deps/npm/lib/utils/cmd-shim.js | 98 ++ deps/npm/lib/utils/completion.sh | 43 + .../lib/utils/completion/file-completion.js | 29 + .../lib/utils/completion/installed-deep.js | 46 + .../lib/utils/completion/installed-shallow.js | 79 ++ .../lib/utils/completion/remote-packages.js | 57 + deps/npm/lib/utils/completion/users.js | 22 + deps/npm/lib/utils/config-defs.js | 299 +++++ deps/npm/lib/utils/error-handler.js | 264 ++++ deps/npm/lib/utils/excludes.js | 145 ++ deps/npm/lib/utils/exec.js | 106 ++ deps/npm/lib/utils/fetch.js | 62 + deps/npm/lib/utils/find-prefix.js | 50 + deps/npm/lib/utils/find.js | 53 + deps/npm/lib/utils/get-agent.js | 62 + deps/npm/lib/utils/get.js | 6 + deps/npm/lib/utils/ini.js | 333 +++++ deps/npm/lib/utils/lifecycle.js | 278 ++++ deps/npm/lib/utils/link.js | 30 + deps/npm/lib/utils/load-package-defaults.js | 75 ++ deps/npm/lib/utils/log.js | 185 +++ deps/npm/lib/utils/mkdir-p.js | 191 +++ .../lib/utils/npm-registry-client/adduser.js | 100 ++ deps/npm/lib/utils/npm-registry-client/get.js | 171 +++ .../lib/utils/npm-registry-client/index.js | 16 + .../lib/utils/npm-registry-client/publish.js | 163 +++ .../lib/utils/npm-registry-client/request.js | 242 ++++ .../npm/lib/utils/npm-registry-client/star.js | 32 + deps/npm/lib/utils/npm-registry-client/tag.js | 8 + .../utils/npm-registry-client/unpublish.js | 98 ++ deps/npm/lib/utils/output.js | 156 +++ deps/npm/lib/utils/promise-chain.js | 39 + deps/npm/lib/utils/prompt.js | 69 + deps/npm/lib/utils/read-installed.js | 295 ++++ deps/npm/lib/utils/read-json.js | 496 +++++++ deps/npm/lib/utils/relativize.js | 70 + deps/npm/lib/utils/set.js | 25 + deps/npm/lib/utils/sha.js | 51 + deps/npm/lib/utils/tar.js | 551 ++++++++ deps/npm/lib/utils/uid-number.js | 55 + deps/npm/lib/version.js | 66 + deps/npm/lib/view.js | 244 ++++ deps/npm/lib/whoami.js | 13 + deps/npm/lib/xmas.js | 54 + deps/npm/node_modules/abbrev/README.md | 23 + deps/npm/node_modules/abbrev/lib/abbrev.js | 106 ++ deps/npm/node_modules/abbrev/package.json | 8 + deps/npm/node_modules/block-stream/README.md | 14 + .../block-stream/bench/block-stream.js | 68 + .../block-stream/bench/dropper-pause.js | 70 + .../block-stream/bench/dropper.js | 68 + .../node_modules/block-stream/block-stream.js | 209 +++ .../node_modules/block-stream/package.json | 23 + .../node_modules/block-stream/test/basic.js | 27 + .../block-stream/test/nopad-thorough.js | 68 + .../node_modules/block-stream/test/nopad.js | 57 + .../block-stream/test/pause-resume.js | 73 + .../block-stream/test/thorough.js | 68 + deps/npm/node_modules/fstream/.gitignore | 3 + deps/npm/node_modules/fstream/README.md | 76 ++ .../npm/node_modules/fstream/examples/pipe.js | 113 ++ .../node_modules/fstream/examples/reader.js | 29 + .../fstream/examples/symlink-write.js | 9 + deps/npm/node_modules/fstream/fstream.js | 31 + deps/npm/node_modules/fstream/lib/abstract.js | 82 ++ deps/npm/node_modules/fstream/lib/collect.js | 67 + .../node_modules/fstream/lib/dir-reader.js | 180 +++ .../node_modules/fstream/lib/dir-writer.js | 165 +++ .../node_modules/fstream/lib/file-reader.js | 147 ++ .../node_modules/fstream/lib/file-writer.js | 95 ++ deps/npm/node_modules/fstream/lib/get-type.js | 32 + .../node_modules/fstream/lib/link-reader.js | 54 + .../node_modules/fstream/lib/link-writer.js | 82 ++ .../node_modules/fstream/lib/proxy-reader.js | 87 ++ .../node_modules/fstream/lib/proxy-writer.js | 109 ++ deps/npm/node_modules/fstream/lib/reader.js | 234 ++++ deps/npm/node_modules/fstream/lib/writer.js | 324 +++++ deps/npm/node_modules/fstream/package.json | 21 + deps/npm/node_modules/graceful-fs/LICENSE | 23 + deps/npm/node_modules/graceful-fs/README.md | 5 + .../node_modules/graceful-fs/graceful-fs.js | 39 + .../npm/node_modules/graceful-fs/package.json | 16 + deps/npm/node_modules/inherits/LICENSE | 26 + deps/npm/node_modules/inherits/README.md | 51 + .../npm/node_modules/inherits/inherits-old.js | 40 + deps/npm/node_modules/inherits/inherits.js | 29 + deps/npm/node_modules/inherits/package.json | 8 + deps/npm/node_modules/ini/LICENSE | 23 + deps/npm/node_modules/ini/README.md | 71 + deps/npm/node_modules/ini/ini.js | 97 ++ deps/npm/node_modules/ini/package.json | 24 + .../node_modules/ini/test/fixtures/foo.ini | 18 + deps/npm/node_modules/ini/test/foo.js | 40 + deps/npm/node_modules/minimatch/.gitmodules | 3 + deps/npm/node_modules/minimatch/LICENSE | 23 + deps/npm/node_modules/minimatch/README.md | 107 ++ deps/npm/node_modules/minimatch/minimatch.js | 399 ++++++ .../minimatch/node_modules/lru-cache/LICENSE | 23 + .../node_modules/lru-cache/README.md | 12 + .../node_modules/lru-cache/lib/lru-cache.js | 151 +++ .../node_modules/lru-cache/package.json | 13 + deps/npm/node_modules/minimatch/package.json | 32 + deps/npm/node_modules/minimatch/test/basic.js | 141 ++ deps/npm/node_modules/mkdirp/LICENSE | 21 + deps/npm/node_modules/mkdirp/README.markdown | 21 + deps/npm/node_modules/mkdirp/examples/pow.js | 6 + deps/npm/node_modules/mkdirp/index.js | 36 + deps/npm/node_modules/mkdirp/package.json | 23 + deps/npm/node_modules/mkdirp/test/chmod.js | 39 + deps/npm/node_modules/mkdirp/test/clobber.js | 37 + deps/npm/node_modules/mkdirp/test/mkdirp.js | 28 + deps/npm/node_modules/mkdirp/test/race.js | 41 + deps/npm/node_modules/mkdirp/test/rel.js | 32 + deps/npm/node_modules/node-uuid/LICENSE.md | 3 + deps/npm/node_modules/node-uuid/README.md | 100 ++ deps/npm/node_modules/node-uuid/package.json | 12 + .../node-uuid/test/benchmark-native.c | 34 + .../node_modules/node-uuid/test/benchmark.js | 27 + .../npm/node_modules/node-uuid/test/test.html | 14 + deps/npm/node_modules/node-uuid/test/test.js | 83 ++ deps/npm/node_modules/node-uuid/uuid.js | 80 ++ deps/npm/node_modules/nopt/.gitignore | 0 deps/npm/node_modules/nopt/LICENSE | 23 + deps/npm/node_modules/nopt/README.md | 208 +++ deps/npm/node_modules/nopt/bin/nopt.js | 44 + .../node_modules/nopt/examples/my-program.js | 30 + deps/npm/node_modules/nopt/lib/nopt.js | 552 ++++++++ deps/npm/node_modules/nopt/package.json | 12 + deps/npm/node_modules/proto-list/LICENSE | 23 + deps/npm/node_modules/proto-list/README.md | 3 + deps/npm/node_modules/proto-list/package.json | 9 + .../npm/node_modules/proto-list/proto-list.js | 94 ++ deps/npm/node_modules/request/LICENSE | 55 + deps/npm/node_modules/request/README.md | 193 +++ deps/npm/node_modules/request/main.js | 506 +++++++ deps/npm/node_modules/request/mimetypes.js | 146 ++ deps/npm/node_modules/request/package.json | 15 + .../request/tests/googledoodle.png | Bin 0 -> 38510 bytes deps/npm/node_modules/request/tests/run.sh | 6 + deps/npm/node_modules/request/tests/server.js | 57 + .../node_modules/request/tests/test-body.js | 90 ++ .../node_modules/request/tests/test-errors.js | 30 + .../node_modules/request/tests/test-pipes.js | 167 +++ .../request/tests/test-timeout.js | 87 ++ deps/npm/node_modules/rimraf/AUTHORS | 5 + deps/npm/node_modules/rimraf/LICENSE | 23 + deps/npm/node_modules/rimraf/README.md | 32 + deps/npm/node_modules/rimraf/fiber.js | 86 ++ deps/npm/node_modules/rimraf/package.json | 9 + deps/npm/node_modules/rimraf/rimraf.js | 140 ++ deps/npm/node_modules/rimraf/test/run.sh | 10 + deps/npm/node_modules/rimraf/test/setup.sh | 47 + .../node_modules/rimraf/test/test-async.js | 5 + .../node_modules/rimraf/test/test-fiber.js | 15 + .../npm/node_modules/rimraf/test/test-sync.js | 3 + deps/npm/node_modules/semver/LICENSE | 23 + deps/npm/node_modules/semver/README.md | 119 ++ deps/npm/node_modules/semver/bin/semver | 71 + deps/npm/node_modules/semver/package.json | 11 + deps/npm/node_modules/semver/semver.js | 303 +++++ deps/npm/node_modules/semver/test.js | 397 ++++++ deps/npm/node_modules/slide/.npmignore | 1 + deps/npm/node_modules/slide/LICENSE | 23 + deps/npm/node_modules/slide/README.md | 32 + deps/npm/node_modules/slide/index.js | 1 + .../slide/lib/async-map-ordered.js | 65 + deps/npm/node_modules/slide/lib/async-map.js | 56 + deps/npm/node_modules/slide/lib/bind-actor.js | 16 + deps/npm/node_modules/slide/lib/chain.js | 20 + deps/npm/node_modules/slide/lib/slide.js | 3 + deps/npm/node_modules/slide/package.json | 19 + deps/npm/node_modules/tar/.gitignore | 4 + deps/npm/node_modules/tar/README.md | 50 + .../node_modules/tar/examples/extracter.js | 11 + deps/npm/node_modules/tar/examples/reader.js | 36 + deps/npm/node_modules/tar/lib/buffer-entry.js | 30 + deps/npm/node_modules/tar/lib/entry-writer.js | 166 +++ deps/npm/node_modules/tar/lib/entry.js | 212 +++ .../tar/lib/extended-header-writer.js | 165 +++ .../node_modules/tar/lib/extended-header.js | 139 ++ deps/npm/node_modules/tar/lib/extract.js | 64 + .../tar/lib/global-header-writer.js | 14 + deps/npm/node_modules/tar/lib/header.js | 385 ++++++ deps/npm/node_modules/tar/lib/pack.js | 212 +++ deps/npm/node_modules/tar/lib/parse.js | 253 ++++ deps/npm/node_modules/tar/old/README.md | 1 + deps/npm/node_modules/tar/old/doc/example.js | 24 + deps/npm/node_modules/tar/old/generator.js | 387 ++++++ deps/npm/node_modules/tar/old/parser.js | 344 +++++ deps/npm/node_modules/tar/old/tar.js | 74 ++ .../tar/old/test/test-generator.js | 13 + .../tar/old/test/test-generator.tar | Bin 0 -> 3260 bytes .../tar/old/test/test-generator.txt | Bin 0 -> 3261 bytes .../node_modules/tar/old/test/test-parser.js | 28 + .../node_modules/tar/old/test/test-tar.tar | Bin 0 -> 3072 bytes .../node_modules/tar/old/test/test-tar.txt | Bin 0 -> 3073 bytes deps/npm/node_modules/tar/package.json | 26 + deps/npm/node_modules/tar/tar.js | 172 +++ deps/npm/node_modules/tar/test/extract.js | 406 ++++++ .../node_modules/tar/test/fixtures/200.tar | Bin 0 -> 3072 bytes .../node_modules/tar/test/fixtures/200L.hex | 50 + ...cccccccccccccccccccccccccccccccccccccccccc | 1 + .../tar/test/fixtures/200longlink.tar | Bin 0 -> 3584 bytes .../tar/test/fixtures/200longname.tar | Bin 0 -> 3072 bytes deps/npm/node_modules/tar/test/fixtures/a.hex | 14 + deps/npm/node_modules/tar/test/fixtures/a.tar | Bin 0 -> 2048 bytes deps/npm/node_modules/tar/test/fixtures/a.txt | 1 + deps/npm/node_modules/tar/test/fixtures/b.hex | 14 + deps/npm/node_modules/tar/test/fixtures/b.tar | Bin 0 -> 2048 bytes deps/npm/node_modules/tar/test/fixtures/b.txt | 1 + deps/npm/node_modules/tar/test/fixtures/c.hex | 74 ++ deps/npm/node_modules/tar/test/fixtures/c.tar | Bin 0 -> 18432 bytes deps/npm/node_modules/tar/test/fixtures/c.txt | 1 + .../npm/node_modules/tar/test/fixtures/cc.txt | 1 + .../node_modules/tar/test/fixtures/foo.hex | 14 + .../npm/node_modules/tar/test/fixtures/foo.js | 1 + .../node_modules/tar/test/fixtures/foo.tar | Bin 0 -> 2048 bytes .../node_modules/tar/test/fixtures/hardlink-1 | 1 + .../node_modules/tar/test/fixtures/hardlink-2 | 1 + .../node_modules/tar/test/fixtures/omega.hex | 22 + .../node_modules/tar/test/fixtures/omega.tar | Bin 0 -> 3072 bytes .../node_modules/tar/test/fixtures/omega.txt | 1 + .../tar/test/fixtures/omegapax.tar | Bin 0 -> 5120 bytes .../tar/test/fixtures/packtest/omega.txt | 1 + .../tar/test/fixtures/packtest/star.4.html | 1184 +++++++++++++++++ .../tar/test/fixtures/packtest/Ω.txt | 1 + ...cccccccccccccccccccccccccccccccccccccccccc | 1 + .../node_modules/tar/test/fixtures/symlink | 1 + .../npm/node_modules/tar/test/fixtures/Ω.txt | 1 + deps/npm/node_modules/tar/test/header.js | 183 +++ deps/npm/node_modules/tar/test/pack.js | 953 +++++++++++++ deps/npm/node_modules/tar/test/parse.js | 359 +++++ deps/npm/node_modules/which/LICENSE | 23 + deps/npm/node_modules/which/README.md | 5 + deps/npm/node_modules/which/bin/which | 14 + deps/npm/node_modules/which/package.json | 17 + deps/npm/node_modules/which/which.js | 67 + deps/npm/package.json | 77 ++ deps/npm/scripts/clean-old.sh | 165 +++ deps/npm/scripts/doc-build.sh | 71 + deps/npm/scripts/index-build.js | 62 + deps/npm/scripts/install.sh | 282 ++++ deps/npm/test/common.js | 7 + .../test/disabled/bundlerecurs/package.json | 4 + deps/npm/test/disabled/failer/package.json | 5 + deps/npm/test/disabled/fast/package.json | 9 + .../test/disabled/package-config/package.json | 4 + deps/npm/test/disabled/package-config/test.js | 17 + deps/npm/test/disabled/slow/package.json | 9 + deps/npm/test/disabled/startstop/package.json | 3 + .../test/packages/npm-test-blerg/package.json | 4 + deps/npm/test/packages/npm-test-blerg/test.js | 5 + .../packages/npm-test-env-reader/package.json | 14 + .../test/packages/npm-test-env-reader/test.sh | 3 + .../npm-test-missing-bindir/package.json | 4 + .../packages/npm-test-missing-bindir/test.js | 5 + .../packages/npm-test-private/package.json | 4 + .../npm-test-test-package/package.json | 5 + .../packages/npm-test-url-dep/package.json | 4 + deps/npm/test/run | 138 ++ deps/npm/test/update-test.sh | 59 + tools/osx-pkg-postinstall.sh | 5 + tools/osx-pkg.pmdoc/01local-contents.xml | 1 + tools/osx-pkg.pmdoc/01local.xml | 2 +- tools/osx-pkg.pmdoc/02npm-contents.xml | 1 + tools/osx-pkg.pmdoc/02npm.xml | 1 + tools/osx-pkg.pmdoc/index.xml | 10 +- 449 files changed, 33131 insertions(+), 4 deletions(-) create mode 100644 deps/npm/.gitignore create mode 100644 deps/npm/.gitmodules create mode 100644 deps/npm/.npmignore create mode 100644 deps/npm/AUTHORS create mode 120000 deps/npm/CHANGES create mode 100644 deps/npm/LICENSE create mode 100644 deps/npm/Makefile create mode 100644 deps/npm/README.md create mode 100755 deps/npm/bin/npm-cli.js create mode 100644 deps/npm/bin/npm-g.cmd create mode 100755 deps/npm/bin/npm-get-uid-gid.js create mode 100644 deps/npm/bin/npm.cmd create mode 100644 deps/npm/bin/npm_g.cmd create mode 100755 deps/npm/bin/read-package-json.js create mode 100755 deps/npm/cli.js create mode 100755 deps/npm/configure create mode 120000 deps/npm/doc/api/author.md create mode 100644 deps/npm/doc/api/bin.md create mode 100644 deps/npm/doc/api/bugs.md create mode 100644 deps/npm/doc/api/commands.md create mode 100644 deps/npm/doc/api/config.md create mode 100644 deps/npm/doc/api/deprecate.md create mode 100644 deps/npm/doc/api/docs.md create mode 100644 deps/npm/doc/api/edit.md create mode 100644 deps/npm/doc/api/explore.md create mode 120000 deps/npm/doc/api/find.md create mode 120000 deps/npm/doc/api/get.md create mode 100644 deps/npm/doc/api/help-search.md create mode 120000 deps/npm/doc/api/home.md create mode 100644 deps/npm/doc/api/init.md create mode 100644 deps/npm/doc/api/install.md create mode 100644 deps/npm/doc/api/link.md create mode 120000 deps/npm/doc/api/list.md create mode 120000 deps/npm/doc/api/ln.md create mode 100644 deps/npm/doc/api/load.md create mode 100644 deps/npm/doc/api/ls.md create mode 100644 deps/npm/doc/api/npm.md create mode 100644 deps/npm/doc/api/outdated.md create mode 100644 deps/npm/doc/api/owner.md create mode 100644 deps/npm/doc/api/pack.md create mode 100644 deps/npm/doc/api/prefix.md create mode 100644 deps/npm/doc/api/prune.md create mode 100644 deps/npm/doc/api/publish.md create mode 100644 deps/npm/doc/api/rebuild.md create mode 100644 deps/npm/doc/api/restart.md create mode 120000 deps/npm/doc/api/rm.md create mode 100644 deps/npm/doc/api/root.md create mode 100644 deps/npm/doc/api/run-script.md create mode 100644 deps/npm/doc/api/search.md create mode 120000 deps/npm/doc/api/set.md create mode 100644 deps/npm/doc/api/start.md create mode 100644 deps/npm/doc/api/stop.md create mode 100644 deps/npm/doc/api/submodule.md create mode 100644 deps/npm/doc/api/tag.md create mode 100644 deps/npm/doc/api/test.md create mode 100644 deps/npm/doc/api/uninstall.md create mode 100644 deps/npm/doc/api/unpublish.md create mode 100644 deps/npm/doc/api/update.md create mode 100644 deps/npm/doc/api/version.md create mode 100644 deps/npm/doc/api/view.md create mode 100644 deps/npm/doc/api/whoami.md create mode 100644 deps/npm/doc/cli/adduser.md create mode 120000 deps/npm/doc/cli/author.md create mode 100644 deps/npm/doc/cli/bin.md create mode 100644 deps/npm/doc/cli/bugs.md create mode 100644 deps/npm/doc/cli/build.md create mode 100644 deps/npm/doc/cli/bundle.md create mode 100644 deps/npm/doc/cli/cache.md create mode 100644 deps/npm/doc/cli/changelog.md create mode 100644 deps/npm/doc/cli/coding-style.md create mode 100644 deps/npm/doc/cli/completion.md create mode 100644 deps/npm/doc/cli/config.md create mode 100644 deps/npm/doc/cli/deprecate.md create mode 100644 deps/npm/doc/cli/developers.md create mode 100644 deps/npm/doc/cli/docs.md create mode 100644 deps/npm/doc/cli/edit.md create mode 100644 deps/npm/doc/cli/explore.md create mode 100644 deps/npm/doc/cli/faq.md create mode 120000 deps/npm/doc/cli/find.md create mode 100644 deps/npm/doc/cli/folders.md create mode 120000 deps/npm/doc/cli/get.md create mode 120000 deps/npm/doc/cli/global.md create mode 100644 deps/npm/doc/cli/help-search.md create mode 100644 deps/npm/doc/cli/help.md create mode 120000 deps/npm/doc/cli/home.md create mode 100644 deps/npm/doc/cli/init.md create mode 100644 deps/npm/doc/cli/install.md create mode 100644 deps/npm/doc/cli/json.md create mode 100644 deps/npm/doc/cli/link.md create mode 100644 deps/npm/doc/cli/list.md create mode 120000 deps/npm/doc/cli/ln.md create mode 120000 deps/npm/doc/cli/ls.md create mode 100644 deps/npm/doc/cli/npm.md create mode 100644 deps/npm/doc/cli/outdated.md create mode 100644 deps/npm/doc/cli/owner.md create mode 100644 deps/npm/doc/cli/pack.md create mode 100644 deps/npm/doc/cli/prefix.md create mode 100644 deps/npm/doc/cli/prune.md create mode 100644 deps/npm/doc/cli/publish.md create mode 100644 deps/npm/doc/cli/rebuild.md create mode 100644 deps/npm/doc/cli/registry.md create mode 100644 deps/npm/doc/cli/removing-npm.md create mode 100644 deps/npm/doc/cli/restart.md create mode 120000 deps/npm/doc/cli/rm.md create mode 100644 deps/npm/doc/cli/root.md create mode 100644 deps/npm/doc/cli/run-script.md create mode 100644 deps/npm/doc/cli/scripts.md create mode 100644 deps/npm/doc/cli/search.md create mode 100644 deps/npm/doc/cli/semver.md create mode 120000 deps/npm/doc/cli/set.md create mode 100644 deps/npm/doc/cli/star.md create mode 100644 deps/npm/doc/cli/start.md create mode 100644 deps/npm/doc/cli/stop.md create mode 100644 deps/npm/doc/cli/submodule.md create mode 100644 deps/npm/doc/cli/tag.md create mode 100644 deps/npm/doc/cli/test.md create mode 100644 deps/npm/doc/cli/uninstall.md create mode 100644 deps/npm/doc/cli/unpublish.md create mode 100644 deps/npm/doc/cli/update.md create mode 100644 deps/npm/doc/cli/version.md create mode 100644 deps/npm/doc/cli/view.md create mode 100644 deps/npm/doc/cli/whoami.md create mode 100755 deps/npm/html/api/GubbleBum-Blocky.ttf create mode 100644 deps/npm/html/api/style.css create mode 100755 deps/npm/html/doc/GubbleBum-Blocky.ttf create mode 100644 deps/npm/html/doc/style.css create mode 100644 deps/npm/html/docfoot.html create mode 100644 deps/npm/html/dochead.html create mode 100644 deps/npm/html/favicon.ico create mode 100644 deps/npm/html/index.html create mode 100644 deps/npm/html/n-64.png create mode 100644 deps/npm/html/n-large.png create mode 100644 deps/npm/html/npm-16.png create mode 100644 deps/npm/html/npm-256-square.png create mode 100644 deps/npm/html/npm-256w.png create mode 100644 deps/npm/html/npm-64-square.png create mode 100644 deps/npm/html/npm-fin.png create mode 100644 deps/npm/html/npm-large.png create mode 100644 deps/npm/html/npm.png create mode 100644 deps/npm/lib/adduser.js create mode 100644 deps/npm/lib/bin.js create mode 100644 deps/npm/lib/bugs.js create mode 100644 deps/npm/lib/build.js create mode 100644 deps/npm/lib/cache.js create mode 100644 deps/npm/lib/completion.js create mode 100644 deps/npm/lib/config.js create mode 100644 deps/npm/lib/deprecate.js create mode 100644 deps/npm/lib/docs.js create mode 100644 deps/npm/lib/edit.js create mode 100644 deps/npm/lib/explore.js create mode 100644 deps/npm/lib/faq.js create mode 100644 deps/npm/lib/get.js create mode 100644 deps/npm/lib/help-search.js create mode 100644 deps/npm/lib/help.js create mode 100644 deps/npm/lib/init.js create mode 100644 deps/npm/lib/install.js create mode 100644 deps/npm/lib/link.js create mode 100644 deps/npm/lib/ls.js create mode 100644 deps/npm/lib/npm.js create mode 100644 deps/npm/lib/outdated.js create mode 100644 deps/npm/lib/owner.js create mode 100644 deps/npm/lib/pack.js create mode 100644 deps/npm/lib/prefix.js create mode 100644 deps/npm/lib/prune.js create mode 100644 deps/npm/lib/publish.js create mode 100644 deps/npm/lib/rebuild.js create mode 100644 deps/npm/lib/restart.js create mode 100644 deps/npm/lib/root.js create mode 100644 deps/npm/lib/run-script.js create mode 100644 deps/npm/lib/search.js create mode 100644 deps/npm/lib/set.js create mode 100644 deps/npm/lib/star.js create mode 100644 deps/npm/lib/start.js create mode 100644 deps/npm/lib/stop.js create mode 100644 deps/npm/lib/submodule.js create mode 100644 deps/npm/lib/tag.js create mode 100644 deps/npm/lib/test.js create mode 100644 deps/npm/lib/unbuild.js create mode 100644 deps/npm/lib/uninstall.js create mode 100644 deps/npm/lib/unpublish.js create mode 100644 deps/npm/lib/update.js create mode 100644 deps/npm/lib/utils/cmd-shim.js create mode 100755 deps/npm/lib/utils/completion.sh create mode 100644 deps/npm/lib/utils/completion/file-completion.js create mode 100644 deps/npm/lib/utils/completion/installed-deep.js create mode 100644 deps/npm/lib/utils/completion/installed-shallow.js create mode 100644 deps/npm/lib/utils/completion/remote-packages.js create mode 100644 deps/npm/lib/utils/completion/users.js create mode 100644 deps/npm/lib/utils/config-defs.js create mode 100644 deps/npm/lib/utils/error-handler.js create mode 100644 deps/npm/lib/utils/excludes.js create mode 100644 deps/npm/lib/utils/exec.js create mode 100644 deps/npm/lib/utils/fetch.js create mode 100644 deps/npm/lib/utils/find-prefix.js create mode 100644 deps/npm/lib/utils/find.js create mode 100644 deps/npm/lib/utils/get-agent.js create mode 100644 deps/npm/lib/utils/get.js create mode 100644 deps/npm/lib/utils/ini.js create mode 100644 deps/npm/lib/utils/lifecycle.js create mode 100644 deps/npm/lib/utils/link.js create mode 100644 deps/npm/lib/utils/load-package-defaults.js create mode 100644 deps/npm/lib/utils/log.js create mode 100644 deps/npm/lib/utils/mkdir-p.js create mode 100644 deps/npm/lib/utils/npm-registry-client/adduser.js create mode 100644 deps/npm/lib/utils/npm-registry-client/get.js create mode 100644 deps/npm/lib/utils/npm-registry-client/index.js create mode 100644 deps/npm/lib/utils/npm-registry-client/publish.js create mode 100644 deps/npm/lib/utils/npm-registry-client/request.js create mode 100644 deps/npm/lib/utils/npm-registry-client/star.js create mode 100644 deps/npm/lib/utils/npm-registry-client/tag.js create mode 100644 deps/npm/lib/utils/npm-registry-client/unpublish.js create mode 100644 deps/npm/lib/utils/output.js create mode 100644 deps/npm/lib/utils/promise-chain.js create mode 100644 deps/npm/lib/utils/prompt.js create mode 100644 deps/npm/lib/utils/read-installed.js create mode 100644 deps/npm/lib/utils/read-json.js create mode 100644 deps/npm/lib/utils/relativize.js create mode 100644 deps/npm/lib/utils/set.js create mode 100644 deps/npm/lib/utils/sha.js create mode 100644 deps/npm/lib/utils/tar.js create mode 100644 deps/npm/lib/utils/uid-number.js create mode 100644 deps/npm/lib/version.js create mode 100644 deps/npm/lib/view.js create mode 100644 deps/npm/lib/whoami.js create mode 100644 deps/npm/lib/xmas.js create mode 100644 deps/npm/node_modules/abbrev/README.md create mode 100644 deps/npm/node_modules/abbrev/lib/abbrev.js create mode 100644 deps/npm/node_modules/abbrev/package.json create mode 100644 deps/npm/node_modules/block-stream/README.md create mode 100644 deps/npm/node_modules/block-stream/bench/block-stream.js create mode 100644 deps/npm/node_modules/block-stream/bench/dropper-pause.js create mode 100644 deps/npm/node_modules/block-stream/bench/dropper.js create mode 100644 deps/npm/node_modules/block-stream/block-stream.js create mode 100644 deps/npm/node_modules/block-stream/package.json create mode 100644 deps/npm/node_modules/block-stream/test/basic.js create mode 100644 deps/npm/node_modules/block-stream/test/nopad-thorough.js create mode 100644 deps/npm/node_modules/block-stream/test/nopad.js create mode 100644 deps/npm/node_modules/block-stream/test/pause-resume.js create mode 100644 deps/npm/node_modules/block-stream/test/thorough.js create mode 100644 deps/npm/node_modules/fstream/.gitignore create mode 100644 deps/npm/node_modules/fstream/README.md create mode 100644 deps/npm/node_modules/fstream/examples/pipe.js create mode 100644 deps/npm/node_modules/fstream/examples/reader.js create mode 100644 deps/npm/node_modules/fstream/examples/symlink-write.js create mode 100644 deps/npm/node_modules/fstream/fstream.js create mode 100644 deps/npm/node_modules/fstream/lib/abstract.js create mode 100644 deps/npm/node_modules/fstream/lib/collect.js create mode 100644 deps/npm/node_modules/fstream/lib/dir-reader.js create mode 100644 deps/npm/node_modules/fstream/lib/dir-writer.js create mode 100644 deps/npm/node_modules/fstream/lib/file-reader.js create mode 100644 deps/npm/node_modules/fstream/lib/file-writer.js create mode 100644 deps/npm/node_modules/fstream/lib/get-type.js create mode 100644 deps/npm/node_modules/fstream/lib/link-reader.js create mode 100644 deps/npm/node_modules/fstream/lib/link-writer.js create mode 100644 deps/npm/node_modules/fstream/lib/proxy-reader.js create mode 100644 deps/npm/node_modules/fstream/lib/proxy-writer.js create mode 100644 deps/npm/node_modules/fstream/lib/reader.js create mode 100644 deps/npm/node_modules/fstream/lib/writer.js create mode 100644 deps/npm/node_modules/fstream/package.json create mode 100644 deps/npm/node_modules/graceful-fs/LICENSE create mode 100644 deps/npm/node_modules/graceful-fs/README.md create mode 100644 deps/npm/node_modules/graceful-fs/graceful-fs.js create mode 100644 deps/npm/node_modules/graceful-fs/package.json create mode 100644 deps/npm/node_modules/inherits/LICENSE create mode 100644 deps/npm/node_modules/inherits/README.md create mode 100644 deps/npm/node_modules/inherits/inherits-old.js create mode 100644 deps/npm/node_modules/inherits/inherits.js create mode 100644 deps/npm/node_modules/inherits/package.json create mode 100644 deps/npm/node_modules/ini/LICENSE create mode 100644 deps/npm/node_modules/ini/README.md create mode 100644 deps/npm/node_modules/ini/ini.js create mode 100644 deps/npm/node_modules/ini/package.json create mode 100644 deps/npm/node_modules/ini/test/fixtures/foo.ini create mode 100644 deps/npm/node_modules/ini/test/foo.js create mode 100644 deps/npm/node_modules/minimatch/.gitmodules create mode 100644 deps/npm/node_modules/minimatch/LICENSE create mode 100644 deps/npm/node_modules/minimatch/README.md create mode 100644 deps/npm/node_modules/minimatch/minimatch.js create mode 100644 deps/npm/node_modules/minimatch/node_modules/lru-cache/LICENSE create mode 100644 deps/npm/node_modules/minimatch/node_modules/lru-cache/README.md create mode 100644 deps/npm/node_modules/minimatch/node_modules/lru-cache/lib/lru-cache.js create mode 100644 deps/npm/node_modules/minimatch/node_modules/lru-cache/package.json create mode 100644 deps/npm/node_modules/minimatch/package.json create mode 100644 deps/npm/node_modules/minimatch/test/basic.js create mode 100644 deps/npm/node_modules/mkdirp/LICENSE create mode 100644 deps/npm/node_modules/mkdirp/README.markdown create mode 100644 deps/npm/node_modules/mkdirp/examples/pow.js create mode 100644 deps/npm/node_modules/mkdirp/index.js create mode 100644 deps/npm/node_modules/mkdirp/package.json create mode 100644 deps/npm/node_modules/mkdirp/test/chmod.js create mode 100644 deps/npm/node_modules/mkdirp/test/clobber.js create mode 100644 deps/npm/node_modules/mkdirp/test/mkdirp.js create mode 100644 deps/npm/node_modules/mkdirp/test/race.js create mode 100644 deps/npm/node_modules/mkdirp/test/rel.js create mode 100644 deps/npm/node_modules/node-uuid/LICENSE.md create mode 100644 deps/npm/node_modules/node-uuid/README.md create mode 100644 deps/npm/node_modules/node-uuid/package.json create mode 100644 deps/npm/node_modules/node-uuid/test/benchmark-native.c create mode 100644 deps/npm/node_modules/node-uuid/test/benchmark.js create mode 100644 deps/npm/node_modules/node-uuid/test/test.html create mode 100644 deps/npm/node_modules/node-uuid/test/test.js create mode 100644 deps/npm/node_modules/node-uuid/uuid.js create mode 100644 deps/npm/node_modules/nopt/.gitignore create mode 100644 deps/npm/node_modules/nopt/LICENSE create mode 100644 deps/npm/node_modules/nopt/README.md create mode 100755 deps/npm/node_modules/nopt/bin/nopt.js create mode 100755 deps/npm/node_modules/nopt/examples/my-program.js create mode 100644 deps/npm/node_modules/nopt/lib/nopt.js create mode 100644 deps/npm/node_modules/nopt/package.json create mode 100644 deps/npm/node_modules/proto-list/LICENSE create mode 100644 deps/npm/node_modules/proto-list/README.md create mode 100644 deps/npm/node_modules/proto-list/package.json create mode 100644 deps/npm/node_modules/proto-list/proto-list.js create mode 100644 deps/npm/node_modules/request/LICENSE create mode 100644 deps/npm/node_modules/request/README.md create mode 100644 deps/npm/node_modules/request/main.js create mode 100644 deps/npm/node_modules/request/mimetypes.js create mode 100644 deps/npm/node_modules/request/package.json create mode 100644 deps/npm/node_modules/request/tests/googledoodle.png create mode 100755 deps/npm/node_modules/request/tests/run.sh create mode 100644 deps/npm/node_modules/request/tests/server.js create mode 100644 deps/npm/node_modules/request/tests/test-body.js create mode 100644 deps/npm/node_modules/request/tests/test-errors.js create mode 100644 deps/npm/node_modules/request/tests/test-pipes.js create mode 100644 deps/npm/node_modules/request/tests/test-timeout.js create mode 100644 deps/npm/node_modules/rimraf/AUTHORS create mode 100644 deps/npm/node_modules/rimraf/LICENSE create mode 100644 deps/npm/node_modules/rimraf/README.md create mode 100644 deps/npm/node_modules/rimraf/fiber.js create mode 100644 deps/npm/node_modules/rimraf/package.json create mode 100644 deps/npm/node_modules/rimraf/rimraf.js create mode 100644 deps/npm/node_modules/rimraf/test/run.sh create mode 100644 deps/npm/node_modules/rimraf/test/setup.sh create mode 100644 deps/npm/node_modules/rimraf/test/test-async.js create mode 100644 deps/npm/node_modules/rimraf/test/test-fiber.js create mode 100644 deps/npm/node_modules/rimraf/test/test-sync.js create mode 100644 deps/npm/node_modules/semver/LICENSE create mode 100644 deps/npm/node_modules/semver/README.md create mode 100755 deps/npm/node_modules/semver/bin/semver create mode 100644 deps/npm/node_modules/semver/package.json create mode 100644 deps/npm/node_modules/semver/semver.js create mode 100644 deps/npm/node_modules/semver/test.js create mode 100644 deps/npm/node_modules/slide/.npmignore create mode 100644 deps/npm/node_modules/slide/LICENSE create mode 100644 deps/npm/node_modules/slide/README.md create mode 100644 deps/npm/node_modules/slide/index.js create mode 100644 deps/npm/node_modules/slide/lib/async-map-ordered.js create mode 100644 deps/npm/node_modules/slide/lib/async-map.js create mode 100644 deps/npm/node_modules/slide/lib/bind-actor.js create mode 100644 deps/npm/node_modules/slide/lib/chain.js create mode 100644 deps/npm/node_modules/slide/lib/slide.js create mode 100644 deps/npm/node_modules/slide/package.json create mode 100644 deps/npm/node_modules/tar/.gitignore create mode 100644 deps/npm/node_modules/tar/README.md create mode 100644 deps/npm/node_modules/tar/examples/extracter.js create mode 100644 deps/npm/node_modules/tar/examples/reader.js create mode 100644 deps/npm/node_modules/tar/lib/buffer-entry.js create mode 100644 deps/npm/node_modules/tar/lib/entry-writer.js create mode 100644 deps/npm/node_modules/tar/lib/entry.js create mode 100644 deps/npm/node_modules/tar/lib/extended-header-writer.js create mode 100644 deps/npm/node_modules/tar/lib/extended-header.js create mode 100644 deps/npm/node_modules/tar/lib/extract.js create mode 100644 deps/npm/node_modules/tar/lib/global-header-writer.js create mode 100644 deps/npm/node_modules/tar/lib/header.js create mode 100644 deps/npm/node_modules/tar/lib/pack.js create mode 100644 deps/npm/node_modules/tar/lib/parse.js create mode 100644 deps/npm/node_modules/tar/old/README.md create mode 100644 deps/npm/node_modules/tar/old/doc/example.js create mode 100644 deps/npm/node_modules/tar/old/generator.js create mode 100644 deps/npm/node_modules/tar/old/parser.js create mode 100644 deps/npm/node_modules/tar/old/tar.js create mode 100644 deps/npm/node_modules/tar/old/test/test-generator.js create mode 100644 deps/npm/node_modules/tar/old/test/test-generator.tar create mode 100644 deps/npm/node_modules/tar/old/test/test-generator.txt create mode 100644 deps/npm/node_modules/tar/old/test/test-parser.js create mode 100644 deps/npm/node_modules/tar/old/test/test-tar.tar create mode 100644 deps/npm/node_modules/tar/old/test/test-tar.txt create mode 100644 deps/npm/node_modules/tar/package.json create mode 100644 deps/npm/node_modules/tar/tar.js create mode 100644 deps/npm/node_modules/tar/test/extract.js create mode 100644 deps/npm/node_modules/tar/test/fixtures/200.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/200L.hex create mode 100644 deps/npm/node_modules/tar/test/fixtures/200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc create mode 100644 deps/npm/node_modules/tar/test/fixtures/200longlink.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/200longname.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/a.hex create mode 100644 deps/npm/node_modules/tar/test/fixtures/a.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/a.txt create mode 100644 deps/npm/node_modules/tar/test/fixtures/b.hex create mode 100644 deps/npm/node_modules/tar/test/fixtures/b.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/b.txt create mode 100644 deps/npm/node_modules/tar/test/fixtures/c.hex create mode 100644 deps/npm/node_modules/tar/test/fixtures/c.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/c.txt create mode 100644 deps/npm/node_modules/tar/test/fixtures/cc.txt create mode 100644 deps/npm/node_modules/tar/test/fixtures/foo.hex create mode 100644 deps/npm/node_modules/tar/test/fixtures/foo.js create mode 100644 deps/npm/node_modules/tar/test/fixtures/foo.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/hardlink-1 create mode 100644 deps/npm/node_modules/tar/test/fixtures/hardlink-2 create mode 100644 deps/npm/node_modules/tar/test/fixtures/omega.hex create mode 100644 deps/npm/node_modules/tar/test/fixtures/omega.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/omega.txt create mode 100644 deps/npm/node_modules/tar/test/fixtures/omegapax.tar create mode 100644 deps/npm/node_modules/tar/test/fixtures/packtest/omega.txt create mode 100644 deps/npm/node_modules/tar/test/fixtures/packtest/star.4.html create mode 100644 deps/npm/node_modules/tar/test/fixtures/packtest/Ω.txt create mode 100644 deps/npm/node_modules/tar/test/fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc create mode 120000 deps/npm/node_modules/tar/test/fixtures/symlink create mode 100644 deps/npm/node_modules/tar/test/fixtures/Ω.txt create mode 100644 deps/npm/node_modules/tar/test/header.js create mode 100644 deps/npm/node_modules/tar/test/pack.js create mode 100644 deps/npm/node_modules/tar/test/parse.js create mode 100644 deps/npm/node_modules/which/LICENSE create mode 100644 deps/npm/node_modules/which/README.md create mode 100755 deps/npm/node_modules/which/bin/which create mode 100644 deps/npm/node_modules/which/package.json create mode 100644 deps/npm/node_modules/which/which.js create mode 100644 deps/npm/package.json create mode 100644 deps/npm/scripts/clean-old.sh create mode 100755 deps/npm/scripts/doc-build.sh create mode 100644 deps/npm/scripts/index-build.js create mode 100644 deps/npm/scripts/install.sh create mode 100644 deps/npm/test/common.js create mode 100644 deps/npm/test/disabled/bundlerecurs/package.json create mode 100644 deps/npm/test/disabled/failer/package.json create mode 100644 deps/npm/test/disabled/fast/package.json create mode 100644 deps/npm/test/disabled/package-config/package.json create mode 100755 deps/npm/test/disabled/package-config/test.js create mode 100644 deps/npm/test/disabled/slow/package.json create mode 100644 deps/npm/test/disabled/startstop/package.json create mode 100644 deps/npm/test/packages/npm-test-blerg/package.json create mode 100644 deps/npm/test/packages/npm-test-blerg/test.js create mode 100644 deps/npm/test/packages/npm-test-env-reader/package.json create mode 100755 deps/npm/test/packages/npm-test-env-reader/test.sh create mode 100644 deps/npm/test/packages/npm-test-missing-bindir/package.json create mode 100644 deps/npm/test/packages/npm-test-missing-bindir/test.js create mode 100644 deps/npm/test/packages/npm-test-private/package.json create mode 100644 deps/npm/test/packages/npm-test-test-package/package.json create mode 100644 deps/npm/test/packages/npm-test-url-dep/package.json create mode 100755 deps/npm/test/run create mode 100755 deps/npm/test/update-test.sh create mode 100644 tools/osx-pkg-postinstall.sh create mode 100644 tools/osx-pkg.pmdoc/01local-contents.xml create mode 100644 tools/osx-pkg.pmdoc/02npm-contents.xml create mode 100644 tools/osx-pkg.pmdoc/02npm.xml diff --git a/LICENSE b/LICENSE index d8d88d62cc..204ff4b93d 100644 --- a/LICENSE +++ b/LICENSE @@ -78,3 +78,9 @@ The externally maintained libraries used by Node are: - deps/zlib copyright 1995-2010 Jean-loup Gailly and Mark Adler licensed under a permissive free software license. See deps/zlib/LICENSE. + + - deps/npm NPM is a package manager program copyright 2009, 2010, 2011 + Isaac Z. Schlueter and licensed under MIT. NPM includes several + subpackages MIT or Apache licenses, see deps/npm/LICENSE for more + information. NPM is included in the Node .msi and .pkg distributions + but not in the Node binary itself. diff --git a/deps/npm/.gitignore b/deps/npm/.gitignore new file mode 100644 index 0000000000..e3424f5f6d --- /dev/null +++ b/deps/npm/.gitignore @@ -0,0 +1,15 @@ +*.swp +test/bin +test/output.log +test/packages/*/node_modules +test/packages/npm-test-depends-on-spark/which-spark.log +test/packages/test-package/random-data.txt +test/root +node_modules/ronn +node_modules/.bin +npm-debug.log +html/api/*.html +html/doc/*.html +man/ +doc/*/index.md +./npmrc diff --git a/deps/npm/.gitmodules b/deps/npm/.gitmodules new file mode 100644 index 0000000000..b629bd2936 --- /dev/null +++ b/deps/npm/.gitmodules @@ -0,0 +1,51 @@ +[submodule "node_modules/semver"] + path = node_modules/semver + url = https://github.com/isaacs/node-semver.git +[submodule "node_modules/abbrev"] + path = node_modules/abbrev + url = https://github.com/isaacs/abbrev-js.git +[submodule "node_modules/nopt"] + path = node_modules/nopt + url = https://github.com/isaacs/nopt.git +[submodule "node_modules/node-uuid"] + path = node_modules/node-uuid + url = https://github.com/broofa/node-uuid +[submodule "node_modules/minimatch"] + path = node_modules/minimatch + url = https://github.com/isaacs/minimatch.git +[submodule "node_modules/graceful-fs"] + path = node_modules/graceful-fs + url = https://github.com/isaacs/node-graceful-fs.git +[submodule "node_modules/slide"] + path = node_modules/slide + url = https://github.com/isaacs/slide-flow-control.git +[submodule "node_modules/rimraf"] + path = node_modules/rimraf + url = https://github.com/isaacs/rimraf.git +[submodule "node_modules/proto-list"] + path = node_modules/proto-list + url = https://github.com/isaacs/proto-list.git +[submodule "node_modules/ini"] + path = node_modules/ini + url = https://github.com/isaacs/ini.git +[submodule "node_modules/which"] + path = node_modules/which + url = https://github.com/isaacs/node-which.git +[submodule "node_modules/request"] + path = node_modules/request + url = https://github.com/isaacs/request.git +[submodule "node_modules/tar"] + path = node_modules/tar + url = git://github.com/isaacs/node-tar.git +[submodule "node_modules/fstream"] + path = node_modules/fstream + url = git://github.com/isaacs/fstream.git +[submodule "node_modules/inherits"] + path = node_modules/inherits + url = git://github.com/isaacs/inherits.git +[submodule "node_modules/block-stream"] + path = node_modules/block-stream + url = git://github.com/isaacs/block-stream.git +[submodule "node_modules/mkdirp"] + path = node_modules/mkdirp + url = git://github.com/isaacs/node-mkdirp.git diff --git a/deps/npm/.npmignore b/deps/npm/.npmignore new file mode 100644 index 0000000000..c0a5f571a1 --- /dev/null +++ b/deps/npm/.npmignore @@ -0,0 +1,11 @@ +*.swp +test/bin +test/output.log +test/packages/*/node_modules +test/packages/npm-test-depends-on-spark/which-spark.log +test/packages/test-package/random-data.txt +test/root +node_modules/ronn +node_modules/.bin +npm-debug.log +./npmrc diff --git a/deps/npm/AUTHORS b/deps/npm/AUTHORS new file mode 100644 index 0000000000..59d7257e81 --- /dev/null +++ b/deps/npm/AUTHORS @@ -0,0 +1,50 @@ +# Authors sorted by whether or not they're me +Isaac Z. Schlueter (http://blog.izs.me/) +Steve Steiner (http://websaucesoftware.com/blog/) +Mikeal Rogers (http://www.mikealrogers.com/) +Aaron Blohowiak (http://aaronblohowiak.com/) +Martyn Smith (http://dollyfish.net.nz/) +Mathias Pettersson (http://mape.me/) +Brian Hammond (http://fictorial.com/) +Charlie Robbins (http://www.charlierobbins.com/) +Francisco Treacy (http://franciscotreacy.com/) +Cliffano Subagio (http://blog.cliffano.com/) +Christian Eager (http://perpenduum.com) +Dav Glass (http://blog.davglass.com) +Alex K. Wolfe +James Sanders (http://james-sanders.com/) +Reid Burke (http://reidburke.com/) +Arlo Breault (http://thoughtherder.com/) +Timo Derstappen (http://teemow.com) +Bradley Meck +Bart Teeuwisse (http://thecodemill.biz/) +Ben Noordhuis (http://bnoordhuis.nl/) +Tor Valamo (http://www.magnimedia.no/) +Whyme.Lyu <5longluna@gmail.com> (http://whyme.kuantu.com/) +Olivier Melcher +Tomaž Muraus (http://www.tomaz-muraus.info) +Evan Meagher (http://evanmeagher.net/) +Orlando Vazquez (http://2wycked.net/) +George Miroshnykov +Geoff Flarity (http://ca.linkedin.com/pub/geoff-flarity/a/536/43a) +Pete Kruckenberg +Laurie Harper (http://laurie.holoweb.net/) +Chris Wong +Max Goodman (http://chromacode.com/) +Scott Bronson +Federico Romero +Visnu Pitiyanuvath (http://visnup.com) +Irakli Gozalishvili (http://jeditoolkit.com/) +Mark Cahill (http://www.tiemonster.info/) +Zearin +Iain Sproat +Trent Mick (http://trentm.com/) +Felix Geisendörfer (http://www.debuggable.com/) +Conny Brunnkvist (http://twitter.com/connyb) +Will Elwood (https://github.com/welwood08) +Oleg Efimov (http://sannis.ru) +Martin Cooper +Jameson Little +cspotcode +Maciej Małecki +Stephen Sugden diff --git a/deps/npm/CHANGES b/deps/npm/CHANGES new file mode 120000 index 0000000000..6b73a61615 --- /dev/null +++ b/deps/npm/CHANGES @@ -0,0 +1 @@ +doc/cli/changelog.md \ No newline at end of file diff --git a/deps/npm/LICENSE b/deps/npm/LICENSE new file mode 100644 index 0000000000..71ca385bdf --- /dev/null +++ b/deps/npm/LICENSE @@ -0,0 +1,61 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter (the "Author") +All rights reserved. + +MIT +no-false-attribs License + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +Distributions of all or part of the Software intended to be used +by the recipients as they would use the unmodified Software, +containing modifications that substantially alter, remove, or +disable functionality of the Software, outside of the documented +configuration mechanisms provided by the Software, shall be +modified such that the Author's bug reporting email addresses and +urls are either replaced with the contact information of the +parties responsible for the changes, or removed entirely. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. + + +Except where noted, this license applies to any and all software +programs and associated documentation files created by the +Author, when distributed with the Software. + +"Node.js" and "node" trademark Joyent, Inc. npm is not officially +part of the Node.js project, and is neither owned by nor +officially affiliated with Joyent, Inc. + +Packages published in the npm registry are not part of npm +itself, are the sole property of their respective maintainers, +and are not covered by this license. + +"npm Logo" created by Mathias Pettersson and Brian Hammond, +used with permission. + +This program includes a BSDTar/LibArchive version 2.8.3-1 binary, +originally distributed as part of the MinGW suite, compiled for +Win32, according to the terms of the BSD license. +See deps/basic-bsdtar-2.8.3-1-ming32-bin/basic-bsdtar.LICENSE. + +This program uses "node-uuid", Copyright (c) 2010 Robert Kieffer, +according to the terms of the MIT license. + +This program uses "request", Copyright (c) 2011 Mikeal Rogers, +according to the terms of the Apache license. diff --git a/deps/npm/Makefile b/deps/npm/Makefile new file mode 100644 index 0000000000..e10381baf7 --- /dev/null +++ b/deps/npm/Makefile @@ -0,0 +1,125 @@ +SHELL = bash + +markdowns = $(shell find doc -name '*.md' | grep -v 'index') README.md + +cli_mandocs = $(shell find doc/cli -name '*.md' \ + |sed 's|.md|.1|g' \ + |sed 's|doc/cli/|man/man1/|g' ) \ + man/man1/README.1 \ + man/man1/index.1 + +api_mandocs = $(shell find doc/api -name '*.md' \ + |sed 's|.md|.3|g' \ + |sed 's|doc/api/|man/man3/|g' ) + +cli_htmldocs = $(shell find doc/cli -name '*.md' \ + |grep -v 'index.md' \ + |sed 's|.md|.html|g' \ + |sed 's|doc/cli/|html/doc/|g' ) \ + html/doc/README.html \ + html/doc/index.html + +api_htmldocs = $(shell find doc/api -name '*.md' \ + |sed 's|.md|.html|g' \ + |sed 's|doc/api/|html/api/|g' ) + +mandocs = $(api_mandocs) $(cli_mandocs) + +htmldocs = $(api_htmldocs) $(cli_htmldocs) + +all: submodules doc + +submodules: + ! [ -d .git ] || git submodule update --init --recursive + +latest: submodules + @echo "Installing latest published npm" + @echo "Use 'make install' or 'make link' to install the code" + @echo "in this folder that you're looking at right now." + node cli.js install -g -f npm + +install: all + node cli.js install -g -f + +# backwards compat +dev: install + +link: uninstall + node cli.js link -f + +clean: doc-clean uninstall + rm npmrc + node cli.js cache clean + +uninstall: submodules + node cli.js rm npm -g -f + +doc: $(mandocs) $(htmldocs) + +docclean: doc-clean +doc-clean: + rm -rf \ + node_modules/ronn \ + node_modules/.bin/ronn \ + .building_ronn \ + doc/cli/index.md \ + doc/api/index.md \ + $(api_mandocs) \ + $(cli_mandocs) \ + $(api_htmldocs) \ + $(cli_htmldocs) \ + &>/dev/null || true + +# use `npm install ronn` for this to work. +man/man1/README.1: README.md scripts/doc-build.sh package.json + scripts/doc-build.sh $< $@ + +man/man1/%.1: doc/cli/%.md scripts/doc-build.sh package.json + @[ -d man/man1 ] || mkdir -p man/man1 + scripts/doc-build.sh $< $@ + +man/man3/%.3: doc/api/%.md scripts/doc-build.sh package.json + @[ -d man/man3 ] || mkdir -p man/man3 + scripts/doc-build.sh $< $@ + +html/doc/README.html: README.md html/dochead.html html/docfoot.html scripts/doc-build.sh package.json + scripts/doc-build.sh $< $@ + +html/doc/%.html: doc/cli/%.md html/dochead.html html/docfoot.html scripts/doc-build.sh package.json + scripts/doc-build.sh $< $@ + +html/api/%.html: doc/api/%.md html/dochead.html html/docfoot.html scripts/doc-build.sh package.json + scripts/doc-build.sh $< $@ + +doc/cli/index.md: $(markdowns) scripts/index-build.js scripts/doc-build.sh package.json + node scripts/index-build.js > $@ + +node_modules/ronn: + node cli.js install https://github.com/isaacs/ronnjs/tarball/master + +doc: man + +man: $(cli_docs) $(api_docs) + +test: submodules + node cli.js test + +version: link + git add package.json &&\ + git ci -m v$(shell npm -v) + +publish: link + git tag -s -m v$(shell npm -v) v$(shell npm -v) &&\ + git push origin master --tags &&\ + npm publish &&\ + make doc-publish + +docpublish: doc-publish +doc-publish: doc + rsync -vazu --stats --no-implied-dirs --delete html/doc/ npmjs.org:/var/www/npmjs.org/public/doc + rsync -vazu --stats --no-implied-dirs --delete html/api/ npmjs.org:/var/www/npmjs.org/public/api + +sandwich: + @[ $$(whoami) = "root" ] && (echo "ok"; echo "ham" > sandwich) || echo "make it yourself" + +.PHONY: all latest install dev link doc clean uninstall test man doc-publish doc-clean docclean docpublish diff --git a/deps/npm/README.md b/deps/npm/README.md new file mode 100644 index 0000000000..df1fdfc25c --- /dev/null +++ b/deps/npm/README.md @@ -0,0 +1,274 @@ +npm(1) -- node package manager +============================== + +## SYNOPSIS + +This is just enough info to get you up and running. + +Much more info available via `npm help` once it's installed. + +## IMPORTANT + +**You need node v0.4 or higher to run this program.** + +To install an old **and unsupported** version of npm that works on node 0.3 +and prior, clone the git repo and dig through the old tags and branches. + +## Simple Install (Unix only, sorry) + +To install npm with one command, do this: + + curl http://npmjs.org/install.sh | sh + +To skip the npm 0.x cleanup, do this: + + curl http://npmjs.org/install.sh | clean=no sh + +To say "yes" to the 0.x cleanup, but skip the prompt: + + curl http://npmjs.org/install.sh | clean=yes sh + +If you get permission errors, see the section below, entitled +"Permission Errors on Installation". + +## Installing on Windows -- Experimental + +Yes, this sucks. A convenient one-liner is coming soon. + +### Step 1: Drop the node.exe somewhere + +You will probably need the latest version of node, **at least** version +`0.5.8` or higher. You can get it from +. + +### Step 2 (optional): Update the %PATH% environment variable + +Update your `%PATH%` environment variable in System Properties: +Advanced: Environment, so that it includes the `bin` folder you chose. +The entries are separated by semicolons. + +You *may* be able to do this from the command line using `set` and +`setx`. `cd` into the `bin` folder you created in step 1, and do this: + + set path=%PATH%;%CD% + setx path "%PATH%" + +This will have the added advantage that you'll be able to simply type +`npm` or `node` in any project folder to access those commands. + +If you decide not to update the PATH, and put the node.exe file in +`C:\node\node.exe`, then the npm executable will end up `C:\node\npm.cmd`, +and you'll have to type `C:\node\npm ` to use it. + +### Step 3: Install git + +If you don't already have git, +[install it](https://git.wiki.kernel.org/index.php/MSysGit:InstallMSysGit). + +Run `git --version` to make sure that it's at least version 1.7.6. + +### Step 4: install npm + +Lastly, **after** node.exe, git, and your %PATH% have *all* been set up +properly, install npm itself: + + git config --system http.sslcainfo /bin/curl-ca-bundle.crt + git clone --recursive git://github.com/isaacs/npm.git + cd npm + node cli.js install npm -gf + +## Permission Errors (`EACCES` or `EACCESS`) on Installation + +On Windows, you may need to run the command prompt in elevated +permission mode. (Right-click on cmd.exe, Run as Administrator.) + +On Unix, you may need to run as root, or use `sudo`. + +**Note**: You would need to `sudo` the `sh`, **not** the `curl`. Fetching +stuff from the internet typically doesn't require elevated permissions. +Running it might. + +I highly recommend that you first download the file, and make sure that +it is what you expect, and *then* run it. + + curl -O http://npmjs.org/install.sh + # inspect file.. + sudo sh install.sh + +## Installing on Cygwin + +No. + +## Dev Install + +To install the latest **unstable** development version from git: + + git clone https://github.com/isaacs/npm.git + cd npm + git submodule update --init --recursive + sudo make install # (or: `node cli.js install -gf`) + +If you're sitting in the code folder reading this document in your +terminal, then you've already got the code. Just do: + + git submodule update --init --recursive + sudo make install + +and npm will install itself. + +If you don't have make, and don't have curl or git, and ALL you have is +this code and node, you can probably do this: + + git submodule update --init --recursive + sudo node ./cli.js install -g + +Note that github tarballs **do not contain submodules**, so +those won't work. You'll have to also fetch the appropriate submodules +listed in the .gitmodules file. + +## Permissions when Using npm to Install Other Stuff + +**tl;dr** + +* Use `sudo` for greater safety. Or don't, if you prefer not to. +* npm will downgrade permissions if it's root before running any build + scripts that package authors specified. + +### More details... + +As of version 0.3, it is recommended to run npm as root. +This allows npm to change the user identifier to the `nobody` user prior +to running any package build or test commands. + +If you are not the root user, or if you are on a platform that does not +support uid switching, then npm will not attempt to change the userid. + +If you would like to ensure that npm **always** runs scripts as the +"nobody" user, and have it fail if it cannot downgrade permissions, then +set the following configuration param: + + npm config set unsafe-perm false + +This will prevent running in unsafe mode, even as non-root users. + +## Uninstalling + +So sad to see you go. + + sudo npm uninstall npm -g + +Or, if that fails, + + sudo make uninstall + +## More Severe Uninstalling + +Usually, the above instructions are sufficient. That will remove +npm, but leave behind anything you've installed. + +If you would like to remove all the packages that you have installed, +then you can use the `npm ls` command to find them, and then `npm rm` to +remove them. + +To remove cruft left behind by npm 0.x, you can use the included +`clean-old.sh` script file. You can run it conveniently like this: + + npm explore npm -g -- sh scripts/clean-old.sh + +npm uses two configuration files, one for per-user configs, and another +for global (every-user) configs. You can view them by doing: + + npm config get userconfig # defaults to ~/.npmrc + npm config get globalconfig # defaults to /usr/local/etc/npmrc + +Uninstalling npm does not remove configuration files by default. You +must remove them yourself manually if you want them gone. Note that +this means that future npm installs will not remember the settings that +you have chosen. + +## Using npm Programmatically + +If you would like to use npm programmatically, you can do that. +It's not very well documented, but it *is* rather simple. + + var npm = require("npm") + npm.load(myConfigObject, function (er) { + if (er) return handlError(er) + npm.commands.install(["some", "args"], function (er, data) { + if (er) return commandFailed(er) + // command succeeded, and data might have some info + }) + npm.on("log", function (message) { .... }) + }) + +The `load` function takes an object hash of the command-line configs. +The various `npm.commands.` functions take an **array** of +positional argument **strings**. The last argument to any +`npm.commands.` function is a callback. Some commands take other +optional arguments. Read the source. + +You cannot set configs individually for any single npm function at this +time. Since `npm` is a singleton, any call to `npm.config.set` will +change the value for *all* npm commands in that process. + +See `./bin/npm-cli.js` for an example of pulling config values off of the +command line arguments using nopt. You may also want to check out `npm +help config` to learn about all the options you can set there. + +## More Docs + +Check out the [docs](http://npmjs.org/doc/), +especially the +[faq](http://npmjs.org/doc/faq.html). + +You can use the `npm help` command to read any of them. + +If you're a developer, and you want to use npm to publish your program, +you should +[read this](http://npmjs.org/doc/developers.html) + +## Legal Stuff + +"npm" and "the npm registry" are owned by Isaac Z. Schlueter. All +rights not explicitly granted in the MIT license are reserved. See the +included LICENSE file for more details. + +"Node.js" and "node" are trademarks owned by Joyent, Inc. npm is not +officially part of the Node.js project, and is neither owned by nor +officially affiliated with Joyent, Inc. + +The packages in the npm registry are not part of npm itself, and are the +sole property of their respective maintainers. While every effort is +made to ensure accountability, there is absolutely no guarantee, +warrantee, or assertion made as to the quality, fitness for a specific +purpose, or lack of malice in any given npm package. Modules +published on the npm registry are not affiliated with or endorsed by +Joyent, Inc., Isaac Z. Schlueter, Ryan Dahl, or the Node.js project. + +If you have a complaint about a package in the npm registry, and cannot +resolve it with the package owner, please express your concerns to +Isaac Z. Schlueter at . + +### In plain english + +This is mine; not my employer's, not Node's, not Joyent's, not Ryan +Dahl's. + +If you publish something, it's yours, and you are solely accountable +for it. Not me, not Node, not Joyent, not Ryan Dahl. + +If other people publish something, it's theirs. Not mine, not Node's, +not Joyent's, not Ryan Dahl's. + +Yes, you can publish something evil. It will be removed promptly if +reported, and we'll lose respect for you. But there is no vetting +process for published modules. + +If this concerns you, inspect the source before using packages. + +## SEE ALSO + +* npm(1) +* npm-faq(1) +* npm-help(1) +* npm-index(1) diff --git a/deps/npm/bin/npm-cli.js b/deps/npm/bin/npm-cli.js new file mode 100755 index 0000000000..db6db2fbd2 --- /dev/null +++ b/deps/npm/bin/npm-cli.js @@ -0,0 +1,77 @@ +#!/usr/bin/env node +;(function () { // wrapper in case we're in module_context mode + +// windows: running "npm blah" in this folder will invoke WSH, not node. +if (typeof WScript !== "undefined") { + WScript.echo("npm does not work when run\n" + +"with the Windows Scripting Host\n\n" + +"'cd' to a different directory,\n" + +"or type 'npm.cmd ',\n" + +"or type 'node npm '.") + WScript.quit(1) + return +} + +var log = require("../lib/utils/log.js") +log.waitForConfig() +log.info("ok", "it worked if it ends with") + +var fs = require("graceful-fs") + , path = require("path") + , npm = require("../lib/npm.js") + , ini = require("../lib/utils/ini.js") + , errorHandler = require("../lib/utils/error-handler.js") + + , configDefs = require("../lib/utils/config-defs.js") + , shorthands = configDefs.shorthands + , types = configDefs.types + , nopt = require("nopt") + +// if npm is called as "npmg" or "npm_g", then +// run in global mode. +if (path.basename(process.argv[1]).slice(-1) === "g") { + process.argv.splice(1, 1, "npm", "-g") +} + +log.verbose(process.argv, "cli") + +var conf = nopt(types, shorthands) +npm.argv = conf.argv.remain +if (npm.deref(npm.argv[0])) npm.command = npm.argv.shift() +else conf.usage = true + + +if (conf.version) { + console.log(npm.version) + return +} + +log.info("npm@"+npm.version, "using") +log.info("node@"+process.version, "using") + +// make sure that this version of node works with this version of npm. +var semver = require("semver") + , nodeVer = process.version + , reqVer = npm.nodeVersionRequired +if (reqVer && !semver.satisfies(nodeVer, reqVer)) { + return errorHandler(new Error( + "npm doesn't work with node " + nodeVer + + "\nRequired: node@" + reqVer), true) +} + +process.on("uncaughtException", errorHandler) + +if (conf.usage && npm.command !== "help") { + npm.argv.unshift(npm.command) + npm.command = "help" +} + +// now actually fire up npm and run the command. +// this is how to use npm programmatically: +conf._exit = true +npm.load(conf, function (er) { + if (er) return errorHandler(er) + npm.commands[npm.command](npm.argv, errorHandler) +}) + +})() diff --git a/deps/npm/bin/npm-g.cmd b/deps/npm/bin/npm-g.cmd new file mode 100644 index 0000000000..bac9e5f1c4 --- /dev/null +++ b/deps/npm/bin/npm-g.cmd @@ -0,0 +1,6 @@ +:: Created by npm, please don't edit manually. +@IF EXIST "%~dp0"\"node.exe" ( + "%~dp0"\"node.exe" "%~dp0\.\node_modules\npm\bin\npm-cli.js" %* +) ELSE ( + node "%~dp0\.\node_modules\npm\bin\npm-cli.js" %* +) \ No newline at end of file diff --git a/deps/npm/bin/npm-get-uid-gid.js b/deps/npm/bin/npm-get-uid-gid.js new file mode 100755 index 0000000000..390e0f2fc9 --- /dev/null +++ b/deps/npm/bin/npm-get-uid-gid.js @@ -0,0 +1,16 @@ +var argv = process.argv.slice(2) + , user = argv[0] || process.getuid() + , group = argv[1] || process.getgid() + +if (!isNaN(user)) user = +user +if (!isNaN(group)) group = +group + +console.error([user, group]) + +try { + process.setgid(group) + process.setuid(user) + console.log(JSON.stringify({uid:+process.getuid(), gid:+process.getgid()})) +} catch (ex) { + console.log(JSON.stringify({error:ex.message,errno:ex.errno})) +} diff --git a/deps/npm/bin/npm.cmd b/deps/npm/bin/npm.cmd new file mode 100644 index 0000000000..bac9e5f1c4 --- /dev/null +++ b/deps/npm/bin/npm.cmd @@ -0,0 +1,6 @@ +:: Created by npm, please don't edit manually. +@IF EXIST "%~dp0"\"node.exe" ( + "%~dp0"\"node.exe" "%~dp0\.\node_modules\npm\bin\npm-cli.js" %* +) ELSE ( + node "%~dp0\.\node_modules\npm\bin\npm-cli.js" %* +) \ No newline at end of file diff --git a/deps/npm/bin/npm_g.cmd b/deps/npm/bin/npm_g.cmd new file mode 100644 index 0000000000..bac9e5f1c4 --- /dev/null +++ b/deps/npm/bin/npm_g.cmd @@ -0,0 +1,6 @@ +:: Created by npm, please don't edit manually. +@IF EXIST "%~dp0"\"node.exe" ( + "%~dp0"\"node.exe" "%~dp0\.\node_modules\npm\bin\npm-cli.js" %* +) ELSE ( + node "%~dp0\.\node_modules\npm\bin\npm-cli.js" %* +) \ No newline at end of file diff --git a/deps/npm/bin/read-package-json.js b/deps/npm/bin/read-package-json.js new file mode 100755 index 0000000000..8c95d86e8b --- /dev/null +++ b/deps/npm/bin/read-package-json.js @@ -0,0 +1,22 @@ +var argv = process.argv +if (argv.length < 3) { + console.error("Usage: read-package.json [ ...]") + process.exit(1) +} + +var fs = require("fs") + , file = argv[2] + , readJson = require("../lib/utils/read-json") + +readJson(file, function (er, data) { + if (er) throw er + if (argv.length === 3) console.log(data) + else argv.slice(3).forEach(function (field) { + field = field.split(".") + var val = data + field.forEach(function (f) { + val = val[f] + }) + console.log(val) + }) +}) diff --git a/deps/npm/cli.js b/deps/npm/cli.js new file mode 100755 index 0000000000..0df931e35a --- /dev/null +++ b/deps/npm/cli.js @@ -0,0 +1,2 @@ +#!/usr/bin/env node +require("./bin/npm-cli.js") diff --git a/deps/npm/configure b/deps/npm/configure new file mode 100755 index 0000000000..b13c8d0d73 --- /dev/null +++ b/deps/npm/configure @@ -0,0 +1,33 @@ +#!/bin/bash + +# set configurations that will be "sticky" on this system, +# surviving npm self-updates. + +CONFIGS=() +i=0 + +# get the location of this file. +unset CDPATH +CONFFILE=$(cd $(dirname "$0"); pwd -P)/npmrc + +while [ $# -gt 0 ]; do + conf="$1" + case $conf in + --help) + echo "./configure --param=value ..." + exit 0 + ;; + --*) + CONFIGS[$i]="${conf:2}" + ;; + *) + CONFIGS[$i]="$conf" + ;; + esac + let i++ + shift +done + +for c in "${CONFIGS[@]}"; do + echo "$c" >> "$CONFFILE" +done diff --git a/deps/npm/doc/api/author.md b/deps/npm/doc/api/author.md new file mode 120000 index 0000000000..b7a53cb66b --- /dev/null +++ b/deps/npm/doc/api/author.md @@ -0,0 +1 @@ +owner.md \ No newline at end of file diff --git a/deps/npm/doc/api/bin.md b/deps/npm/doc/api/bin.md new file mode 100644 index 0000000000..f3dc48286d --- /dev/null +++ b/deps/npm/doc/api/bin.md @@ -0,0 +1,13 @@ +npm-bin(3) -- Display npm bin folder +==================================== + +## SYNOPSIS + + npm.commands.bin(args, cb) + +## DESCRIPTION + +Print the folder where npm will install executables. + +This function should not be used programmatically. Instead, just refer +to the `npm.bin` member. diff --git a/deps/npm/doc/api/bugs.md b/deps/npm/doc/api/bugs.md new file mode 100644 index 0000000000..cc4db8f9ec --- /dev/null +++ b/deps/npm/doc/api/bugs.md @@ -0,0 +1,19 @@ +npm-bugs(3) -- Bugs for a package in a web browser maybe +======================================================== + +## SYNOPSIS + + npm.commands.bugs(package, callback) + +## DESCRIPTION + +This command tries to guess at the likely location of a package's +bug tracker URL, and then tries to open it using the `--browser` +config param. + +Like other commands, the first parameter is an array. This command only +uses the first element, which is expected to be a package name with an +optional version number. + +This command will launch a browser, so this command may not be the most +friendly for programmatic use. diff --git a/deps/npm/doc/api/commands.md b/deps/npm/doc/api/commands.md new file mode 100644 index 0000000000..eb7545639d --- /dev/null +++ b/deps/npm/doc/api/commands.md @@ -0,0 +1,22 @@ +npm-commands(3) -- npm commands +=============================== + +## SYNOPSIS + + npm.commands[](args, callback) + +## DESCRIPTION + +npm comes with a full set of commands, and each of the commands takes a +similar set of arguments. + +In general, all commands on the command object take an **array** of positional +argument **strings**. The last argument to any function is a callback. Some +commands are special and take other optional arguments. + +All commands have their own man page. See `man npm-` for command-line +usage, or `man 3 npm-` for programmatic usage. + +## SEE ALSO + +* npm-index(1) diff --git a/deps/npm/doc/api/config.md b/deps/npm/doc/api/config.md new file mode 100644 index 0000000000..7ae2274281 --- /dev/null +++ b/deps/npm/doc/api/config.md @@ -0,0 +1,45 @@ +npm-config(3) -- Manage the npm configuration files +=================================================== + +## SYNOPSIS + + npm.commands.config(args, callback) + var val = npm.config.get(key) + npm.config.set(key, val) + +## DESCRIPTION + +This function acts much the same way as the command-line version. The first +element in the array tells config what to do. Possible values are: + +* `set` + + Sets a config parameter. The second element in `args` is interpreted as the + key, and the third element is interpreted as the value. + +* `get` + + Gets the value of a config parameter. The second element in `args` is the + key to get the value of. + +* `delete` (`rm` or `del`) + + Deletes a parameter from the config. The second element in `args` is the + key to delete. + +* `list` (`ls`) + + Show all configs that aren't secret. No parameters necessary. + +* `edit`: + + Opens the config file in the default editor. This command isn't very useful + programmatically, but it is made available. + +To programmatically access npm configuration settings, or set them for +the duration of a program, use the `npm.config.set` and `npm.config.get` +functions instead. + +## SEE ALSO + +* npm(3) diff --git a/deps/npm/doc/api/deprecate.md b/deps/npm/doc/api/deprecate.md new file mode 100644 index 0000000000..ac94fd7a9f --- /dev/null +++ b/deps/npm/doc/api/deprecate.md @@ -0,0 +1,32 @@ +npm-deprecate(3) -- Deprecate a version of a package +==================================================== + +## SYNOPSIS + + npm.commands.deprecate(args, callback) + +## DESCRIPTION + +This command will update the npm registry entry for a package, providing +a deprecation warning to all who attempt to install it. + +The 'args' parameter must have exactly two elements: + +* `package[@version]` + + The `version` portion is optional, and may be either a range, or a + specific version, or a tag. + +* `message` + + The warning message that will be printed whenever a user attempts to + install the package. + +Note that you must be the package owner to deprecate something. See the +`owner` and `adduser` help topics. + +## SEE ALSO + +* npm-publish(3) +* npm-unpublish(3) +* npm-registry(1) diff --git a/deps/npm/doc/api/docs.md b/deps/npm/doc/api/docs.md new file mode 100644 index 0000000000..2c5fc5e632 --- /dev/null +++ b/deps/npm/doc/api/docs.md @@ -0,0 +1,19 @@ +npm-docs(3) -- Docs for a package in a web browser maybe +======================================================== + +## SYNOPSIS + + npm.commands.docs(package, callback) + +## DESCRIPTION + +This command tries to guess at the likely location of a package's +documentation URL, and then tries to open it using the `--browser` +config param. + +Like other commands, the first parameter is an array. This command only +uses the first element, which is expected to be a package name with an +optional version number. + +This command will launch a browser, so this command may not be the most +friendly for programmatic use. diff --git a/deps/npm/doc/api/edit.md b/deps/npm/doc/api/edit.md new file mode 100644 index 0000000000..b13fbb8578 --- /dev/null +++ b/deps/npm/doc/api/edit.md @@ -0,0 +1,24 @@ +npm-edit(3) -- Edit an installed package +======================================== + +## SYNOPSIS + + npm.commands.edit(package, callback) + +## DESCRIPTION + +Opens the package folder in the default editor (or whatever you've +configured as the npm `editor` config -- see `npm help config`.) + +After it has been edited, the package is rebuilt so as to pick up any +changes in compiled packages. + +For instance, you can do `npm install connect` to install connect +into your package, and then `npm.commands.edit(["connect"], callback)` +to make a few changes to your locally installed copy. + +The first parameter is a string array with a single element, the package +to open. The package can optionally have a version number attached. + +Since this command opens an editor in a new process, be careful about where +and how this is used. diff --git a/deps/npm/doc/api/explore.md b/deps/npm/doc/api/explore.md new file mode 100644 index 0000000000..a239f3df31 --- /dev/null +++ b/deps/npm/doc/api/explore.md @@ -0,0 +1,18 @@ +npm-explore(3) -- Browse an installed package +============================================= + +## SYNOPSIS + + npm.commands.explore(args, callback) + +## DESCRIPTION + +Spawn a subshell in the directory of the installed package specified. + +If a command is specified, then it is run in the subshell, which then +immediately terminates. + +Note that the package is *not* automatically rebuilt afterwards, so be +sure to use `npm rebuild ` if you make any changes. + +The first element in the 'args' parameter must be a package name. After that is the optional command, which can be any number of strings. All of the strings will be combined into one, space-delimited command. diff --git a/deps/npm/doc/api/find.md b/deps/npm/doc/api/find.md new file mode 120000 index 0000000000..5b3debb8f1 --- /dev/null +++ b/deps/npm/doc/api/find.md @@ -0,0 +1 @@ +ls.md \ No newline at end of file diff --git a/deps/npm/doc/api/get.md b/deps/npm/doc/api/get.md new file mode 120000 index 0000000000..3dc8737366 --- /dev/null +++ b/deps/npm/doc/api/get.md @@ -0,0 +1 @@ +config.md \ No newline at end of file diff --git a/deps/npm/doc/api/help-search.md b/deps/npm/doc/api/help-search.md new file mode 100644 index 0000000000..5c00cfc177 --- /dev/null +++ b/deps/npm/doc/api/help-search.md @@ -0,0 +1,30 @@ +npm-help-search(3) -- Search the help pages +=========================================== + +## SYNOPSIS + + npm.commands.helpSearch(args, [silent,] callback) + +## DESCRIPTION + +This command is rarely useful, but it exists in the rare case that it is. + +This command takes an array of search terms and returns the help pages that +match in order of best match. + +If there is only one match, then npm displays that help section. If there +are multiple results, the results are printed to the screen formatted and the +array of results is returned. Each result is an object with these properties: + +* hits: + A map of args to number of hits on that arg. For example, {"npm": 3} +* found: + Total number of unique args that matched. +* totalHits: + Total number of hits. +* lines: + An array of all matching lines (and some adjacent lines). +* file: + Name of the file that matched + +The silent parameter is not neccessary not used, but it may in the future. diff --git a/deps/npm/doc/api/home.md b/deps/npm/doc/api/home.md new file mode 120000 index 0000000000..8828313f5b --- /dev/null +++ b/deps/npm/doc/api/home.md @@ -0,0 +1 @@ +docs.md \ No newline at end of file diff --git a/deps/npm/doc/api/init.md b/deps/npm/doc/api/init.md new file mode 100644 index 0000000000..5afc11b3ba --- /dev/null +++ b/deps/npm/doc/api/init.md @@ -0,0 +1,29 @@ +npm init(3) -- Interactively create a package.json file +======================================================= + +## SYNOPSIS + + npm.commands.init(args, callback) + +## DESCRIPTION + +This will ask you a bunch of questions, and then write a package.json for you. + +It attempts to make reasonable guesses about what you want things to be set to, +and then writes a package.json file with the options you've selected. + +If you already have a package.json file, it'll read that first, and default to +the options in there. + +It is strictly additive, so it does not delete options from your package.json +without a really good reason to do so. + +Since this function expects to be run on the command-line, it doesn't work very +well as a programmatically. The best option is to roll your own, and since +JavaScript makes it stupid simple to output formatted JSON, that is the +preferred method. If you're sure you want to handle command-line prompting, +then go ahead and use this programmatically. + +## SEE ALSO + +npm-json(1) diff --git a/deps/npm/doc/api/install.md b/deps/npm/doc/api/install.md new file mode 100644 index 0000000000..12f665a76c --- /dev/null +++ b/deps/npm/doc/api/install.md @@ -0,0 +1,19 @@ +npm-install(3) -- install a package programmatically +==================================================== + +## SYNOPSIS + + npm.commands.install([where,] packages, callback) + +## DESCRIPTION + +This acts much the same ways as installing on the command-line. + +The 'where' parameter is optional and only used internally, and it specifies +where the packages should be installed to. + +The 'packages' parameter is an array of strings. Each element in the array is +the name of a package to be installed. + +Finally, 'callback' is a function that will be called when all packages have been +installed or when an error has been encountered. diff --git a/deps/npm/doc/api/link.md b/deps/npm/doc/api/link.md new file mode 100644 index 0000000000..ad8cefcab3 --- /dev/null +++ b/deps/npm/doc/api/link.md @@ -0,0 +1,33 @@ +npm-link(3) -- Symlink a package folder +======================================= + +## SYNOPSIS + + npm.command.link(callback) + npm.command.link(packages, callback) + +## DESCRIPTION + +Package linking is a two-step process. + +Without parameters, link will create a globally-installed +symbolic link from `prefix/package-name` to the current folder. + +With a parameters, link will create a symlink from the local `node_modules` +folder to the global symlink. + +When creating tarballs for `npm publish`, the linked packages are +"snapshotted" to their current state by resolving the symbolic links. + +This is +handy for installing your own stuff, so that you can work on it and test it +iteratively without having to continually rebuild. + +For example: + + npm.commands.link(cb) # creates global link from the cwd + # (say redis package) + npm.commands.link('redis', cb) # link-install the package + +Now, any changes to the redis package will be reflected in +the package in the current working directory diff --git a/deps/npm/doc/api/list.md b/deps/npm/doc/api/list.md new file mode 120000 index 0000000000..5b3debb8f1 --- /dev/null +++ b/deps/npm/doc/api/list.md @@ -0,0 +1 @@ +ls.md \ No newline at end of file diff --git a/deps/npm/doc/api/ln.md b/deps/npm/doc/api/ln.md new file mode 120000 index 0000000000..243f994145 --- /dev/null +++ b/deps/npm/doc/api/ln.md @@ -0,0 +1 @@ +link.md \ No newline at end of file diff --git a/deps/npm/doc/api/load.md b/deps/npm/doc/api/load.md new file mode 100644 index 0000000000..a95a6b295d --- /dev/null +++ b/deps/npm/doc/api/load.md @@ -0,0 +1,26 @@ +npm-load(3) -- Load config settings +=================================== + +## SYNOPSIS + + npm.load(conf, cb) + +## DESCRIPTION + +npm.load() must be called before any other function call. Both parameters are +optional, but the second is recommended. + +The first parameter is an object hash of command-line config params, and the +second parameter is a callback that will be called when npm is loaded and +ready to serve. + +The first parameter should follow a similar structure as the package.json +config object. + +For example, to emulate the --dev flag, pass an object that looks like this: + + { + "dev": true + } + +For a list of all the available command-line configs, see `npm help config` diff --git a/deps/npm/doc/api/ls.md b/deps/npm/doc/api/ls.md new file mode 100644 index 0000000000..a6c0a13821 --- /dev/null +++ b/deps/npm/doc/api/ls.md @@ -0,0 +1,50 @@ +npm-ls(3) -- List installed packages +====================================== + +## SYNOPSIS + + npm.commands.ls(args, [silent,] callback) + +## DESCRIPTION + +This command will print to stdout all the versions of packages that are +installed, as well as their dependencies, in a tree-structure. It will also +return that data using the callback. + +This command does not take any arguments, but args must be defined. +Beyond that, if any arguments are passed in, npm will politely warn that it +does not take positional arguments, though you may set config flags +like with any other command, such as `global` to list global packages. + +It will print out extraneous, missing, and invalid packages. + +If the silent parameter is set to true, nothing will be output to the screen, +but the data will still be returned. + +## CONFIGURATION + +### long + +* Default: false +* Type: Boolean + +Show extended information. + +### parseable + +* Default: false +* Type: Boolean + +Show parseable output instead of tree view. + +### global + +* Default: false +* Type: Boolean + +List packages in the global install prefix instead of in the current +project. + +Note, if parseable is set or long isn't set, then duplicates will be trimmed. +This means that if a submodule a same dependency as a parent module, then the +dependency will only be output once. diff --git a/deps/npm/doc/api/npm.md b/deps/npm/doc/api/npm.md new file mode 100644 index 0000000000..a2f034c4b7 --- /dev/null +++ b/deps/npm/doc/api/npm.md @@ -0,0 +1,115 @@ +npm(3) -- node package manager +============================== + +## SYNOPSIS + + var npm = require("npm") + npm.load(configObject, function (er, npm) { + // use the npm object, now that it's loaded. + + npm.config.set(key, val) + val = npm.config.get(key) + + console.log("prefix = %s", npm.prefix) + + npm.commands.install(["package"], cb) + }) + +## VERSION + +@VERSION@ + +## DESCRIPTION + +This is the API documentation for npm. +To find documentation of the command line +client, see `npm(1)`. + +Prior to using npm's commands, +`npm.load()` must be called with an object hash of +top-level configs. In the npm command line client, +this set of configs is parsed from the command line options. Additional +configuration params are loaded from two configuration files. See +`npm-config(1)` for more information. + +After that, each of the functions are accessible in the +commands object: `npm.commands.`. See `npm-index(1)` for a list of +all possible commands. + +All commands on the command object take an **array** of positional argument +**strings**. The last argument to any function is a callback. Some +commands take other optional arguments. + +Configs cannot currently be set on a per function basis, as each call to +npm.config.set will change the value for *all* npm commands in that process. + +To find API documentation for a specific command, run the `npm apihelp` +command. + +## METHODS AND PROPERTIES + +* `npm.load(configs, cb)` + + Load the configuration params, and call the `cb` function once the + globalconfig and userconfig files have been loaded as well, or on + nextTick if they've already been loaded. + +* `npm.config` + + An object for accessing npm configuration parameters. + + * `npm.config.get(key)` + * `npm.config.set(key, val)` + * `npm.config.del(key)` + +* `npm.dir` or `npm.root` + + The `node_modules` directory where npm will operate. + +* `npm.prefix` + + The prefix where npm is operating. (Most often the current working + directory.) + +* `npm.cache` + + The place where npm keeps JSON and tarballs it fetches from the + registry (or uploads to the registry). + +* `npm.tmp` + + npm's temporary working directory. + +* `npm.deref` + + Get the "real" name for a command that has either an alias or + abbreviation. + +## MAGIC + +For each of the methods in the `npm.commands` hash, a method is added to +the npm object, which takes a set of positional string arguments rather +than an array and a callback. + +If the last argument is a callback, then it will use the supplied +callback. However, if no callback is provided, then it will print out +the error or results. + +For example, this would work in a node repl: + + > npm = require("npm") + > npm.load() // wait a sec... + > npm.install("dnode", "express") + +Note that that *won't* work in a node program, since the `install` +method will get called before the configuration load is completed. + +## ABBREVS + +In order to support `npm ins foo` instead of `npm install foo`, the +`npm.commands` object has a set of abbreviations as well as the full +method names. Use the `npm.deref` method to find the real name. + +For example: + + var cmd = npm.deref("unp") // cmd === "unpublish" diff --git a/deps/npm/doc/api/outdated.md b/deps/npm/doc/api/outdated.md new file mode 100644 index 0000000000..89f4cf6faa --- /dev/null +++ b/deps/npm/doc/api/outdated.md @@ -0,0 +1,13 @@ +npm-outdated(3) -- Check for outdated packages +============================================== + +## SYNOPSIS + + npm.commands.outdated([packages,] callback) + +## DESCRIPTION + +This command will check the registry to see if the specified packages are +currently outdated. + +If the 'packages' parameter is left out, npm will check all packages. diff --git a/deps/npm/doc/api/owner.md b/deps/npm/doc/api/owner.md new file mode 100644 index 0000000000..de203c072a --- /dev/null +++ b/deps/npm/doc/api/owner.md @@ -0,0 +1,31 @@ +npm-owner(3) -- Manage package owners +===================================== + +## SYNOPSIS + + npm.commands.owner(args, callback) + +## DESCRIPTION + +The first element of the 'args' parameter defines what to do, and the subsequent +elements depend on the action. Possible values for the action are (order of +parameters are given in parenthesis): + +* ls (package): + List all the users who have access to modify a package and push new versions. + Handy when you need to know who to bug for help. +* add (user, package): + Add a new user as a maintainer of a package. This user is enabled to modify + metadata, publish new versions, and add other owners. +* rm (user, package): + Remove a user from the package owner list. This immediately revokes their + privileges. + +Note that there is only one level of access. Either you can modify a package, +or you can't. Future versions may contain more fine-grained access levels, but +that is not implemented at this time. + +## SEE ALSO + +* npm-publish(3) +* npm-registry(1) diff --git a/deps/npm/doc/api/pack.md b/deps/npm/doc/api/pack.md new file mode 100644 index 0000000000..cb339c0c42 --- /dev/null +++ b/deps/npm/doc/api/pack.md @@ -0,0 +1,19 @@ +npm-pack(3) -- Create a tarball from a package +============================================== + +## SYNOPSIS + + npm.commands.pack([packages,] callback) + +## DESCRIPTION + +For anything that's installable (that is, a package folder, tarball, +tarball url, name@tag, name@version, or name), this command will fetch +it to the cache, and then copy the tarball to the current working +directory as `-.tgz`, and then write the filenames out to +stdout. + +If the same package is specified multiple times, then the file will be +overwritten the second time. + +If no arguments are supplied, then npm packs the current package folder. diff --git a/deps/npm/doc/api/prefix.md b/deps/npm/doc/api/prefix.md new file mode 100644 index 0000000000..806dd4b6cb --- /dev/null +++ b/deps/npm/doc/api/prefix.md @@ -0,0 +1,15 @@ +npm-prefix(3) -- Display prefix +=============================== + +## SYNOPSIS + + npm.commands.prefix(args, callback) + +## DESCRIPTION + +Print the prefix to standard out. + +'args' is never used and callback is never called with data. +'args' must be present or things will break. + +This function is not useful programmatically diff --git a/deps/npm/doc/api/prune.md b/deps/npm/doc/api/prune.md new file mode 100644 index 0000000000..2a4f177485 --- /dev/null +++ b/deps/npm/doc/api/prune.md @@ -0,0 +1,17 @@ +npm-prune(3) -- Remove extraneous packages +========================================== + +## SYNOPSIS + + npm.commands.prune([packages,] callback) + +## DESCRIPTION + +This command removes "extraneous" packages. + +The first parameter is optional, and it specifies packages to be removed. + +No packages are specified, then all packages will be checked. + +Extraneous packages are packages that are not listed on the parent +package's dependencies list. diff --git a/deps/npm/doc/api/publish.md b/deps/npm/doc/api/publish.md new file mode 100644 index 0000000000..a743303f88 --- /dev/null +++ b/deps/npm/doc/api/publish.md @@ -0,0 +1,30 @@ +npm-publish(3) -- Publish a package +=================================== + +## SYNOPSIS + + npm.commands.publish([packages,] callback) + +## DESCRIPTION + +Publishes a package to the registry so that it can be installed by name. +Possible values in the 'packages' array are: + +* ``: + A folder containing a package.json file + +* ``: + A url or file path to a gzipped tar archive containing a single folder + with a package.json file inside. + +If the package array is empty, npm will try to publish something in the +current working directory. + +This command could fails if one of the packages specified already exists in +the registry. Overwrites when the "force" environment variable is set. + +## SEE ALSO + +* npm-registry(1) +* npm-adduser(1) +* npm-owner(3) diff --git a/deps/npm/doc/api/rebuild.md b/deps/npm/doc/api/rebuild.md new file mode 100644 index 0000000000..8b8989806a --- /dev/null +++ b/deps/npm/doc/api/rebuild.md @@ -0,0 +1,16 @@ +npm-rebuild(3) -- Rebuild a package +=================================== + +## SYNOPSIS + + npm.commands.rebuild([packages,] callback) + +## DESCRIPTION + +This command runs the `npm build` command on each of the matched packages. This is useful +when you install a new version of node, and must recompile all your C++ addons with +the new binary. If no 'packages' parameter is specify, every package will be rebuilt. + +## CONFIGURATION + +See `npm help build` diff --git a/deps/npm/doc/api/restart.md b/deps/npm/doc/api/restart.md new file mode 100644 index 0000000000..c40704438e --- /dev/null +++ b/deps/npm/doc/api/restart.md @@ -0,0 +1,22 @@ +npm-restart(3) -- Start a package +================================= + +## SYNOPSIS + + npm.commands.restart(packages, callback) + +## DESCRIPTION + +This runs a package's "restart" script, if one was provided. +Otherwise it runs package's "stop" script, if one was provided, and then +the "start" script. + +If no version is specified, then it restarts the "active" version. + +npm can run tests on multiple packages. Just specify multiple packages +in the `packages` parameter. + +## SEE ALSO + +* npm-start(3) +* npm-stop(3) diff --git a/deps/npm/doc/api/rm.md b/deps/npm/doc/api/rm.md new file mode 120000 index 0000000000..32d3b511f9 --- /dev/null +++ b/deps/npm/doc/api/rm.md @@ -0,0 +1 @@ +uninstall.md \ No newline at end of file diff --git a/deps/npm/doc/api/root.md b/deps/npm/doc/api/root.md new file mode 100644 index 0000000000..1c3ab56402 --- /dev/null +++ b/deps/npm/doc/api/root.md @@ -0,0 +1,15 @@ +npm-root(3) -- Display npm root +=============================== + +## SYNOPSIS + + npm.commands.root(args, callback) + +## DESCRIPTION + +Print the effective `node_modules` folder to standard out. + +'args' is never used and callback is never called with data. +'args' must be present or things will break. + +This function is not useful programmatically. diff --git a/deps/npm/doc/api/run-script.md b/deps/npm/doc/api/run-script.md new file mode 100644 index 0000000000..f15900ecbc --- /dev/null +++ b/deps/npm/doc/api/run-script.md @@ -0,0 +1,27 @@ +npm-run-script(3) -- Run arbitrary package scripts +================================================== + +## SYNOPSIS + + npm.commands.run-script(args, callback) + +## DESCRIPTION + +This runs an arbitrary command from a package's "scripts" object. + +It is used by the test, start, restart, and stop commands, but can be +called directly, as well. + +The 'args' parameter is an array of strings. Behavior depends on the number +of elements. If there is only one element, npm assumes that the element +represents a command to be run on the local repository. If there is more than +one element, then the first is assumed to be the package and the second is +assumed to be the command to run. All other elements are ignored. + +## SEE ALSO + +* npm-scripts(1) +* npm-test(3) +* npm-start(3) +* npm-restart(3) +* npm-stop(3) diff --git a/deps/npm/doc/api/search.md b/deps/npm/doc/api/search.md new file mode 100644 index 0000000000..30651d76a4 --- /dev/null +++ b/deps/npm/doc/api/search.md @@ -0,0 +1,35 @@ +npm-search(3) -- Search for packages +==================================== + +## SYNOPSIS + + npm.commands.search(searchTerms, [silent,] [staleness,] callback) + +## DESCRIPTION + +Search the registry for packages matching the search terms. The available parameters are: + +* searchTerms: + Array of search terms. These terms are case-insensitive. +* silent: + If true, npm will not log anything to the console. +* staleness: + This is the threshold for stale packages. "Fresh" packages are not refreshed + from the registry. This value is measured in seconds. +* callback: + Returns an object where each key is the name of a package, and the value + is information about that package along with a 'words' property, which is + a space-delimited string of all of the interesting words in that package. + The only properties included are those that are searched, which generally include: + + * name + * description + * maintainers + * url + * keywords + +A search on the registry excludes any result that does not match all of the +search terms. It also removes any items from the results that contain an +excluded term (the "searchexclude" config). The search is case insensitive +and doesn't try to read your mind (it doesn't do any verb tense matching or the +like). diff --git a/deps/npm/doc/api/set.md b/deps/npm/doc/api/set.md new file mode 120000 index 0000000000..3dc8737366 --- /dev/null +++ b/deps/npm/doc/api/set.md @@ -0,0 +1 @@ +config.md \ No newline at end of file diff --git a/deps/npm/doc/api/start.md b/deps/npm/doc/api/start.md new file mode 100644 index 0000000000..74491146aa --- /dev/null +++ b/deps/npm/doc/api/start.md @@ -0,0 +1,13 @@ +npm-start(3) -- Start a package +=============================== + +## SYNOPSIS + + npm.commands.start(packages, callback) + +## DESCRIPTION + +This runs a package's "start" script, if one was provided. + +npm can run tests on multiple packages. Just specify multiple packages +in the `packages` parameter. diff --git a/deps/npm/doc/api/stop.md b/deps/npm/doc/api/stop.md new file mode 100644 index 0000000000..0f8333d351 --- /dev/null +++ b/deps/npm/doc/api/stop.md @@ -0,0 +1,13 @@ +npm-stop(3) -- Stop a package +============================= + +## SYNOPSIS + + npm.commands.stop(packages, callback) + +## DESCRIPTION + +This runs a package's "stop" script, if one was provided. + +npm can run stop on multiple packages. Just specify multiple packages +in the `packages` parameter. diff --git a/deps/npm/doc/api/submodule.md b/deps/npm/doc/api/submodule.md new file mode 100644 index 0000000000..2d8bafaa31 --- /dev/null +++ b/deps/npm/doc/api/submodule.md @@ -0,0 +1,28 @@ +npm-submodule(3) -- Add a package as a git submodule +==================================================== + +## SYNOPSIS + + npm.commands.submodule(packages, callback) + +## DESCRIPTION + +For each package specified, npm will check if it has a git repository url +in its package.json description then add it as a git submodule at +`node_modules/`. + +This is a convenience only. From then on, it's up to you to manage +updates by using the appropriate git commands. npm will stubbornly +refuse to update, modify, or remove anything with a `.git` subfolder +in it. + +This command also does not install missing dependencies, if the package +does not include them in its git repository. If `npm ls` reports that +things are missing, you can either install, link, or submodule them yourself, +or you can do `npm explore -- npm install` to install the +dependencies into the submodule folder. + +## SEE ALSO + +* npm help json +* git help submodule diff --git a/deps/npm/doc/api/tag.md b/deps/npm/doc/api/tag.md new file mode 100644 index 0000000000..b5a3d7faa6 --- /dev/null +++ b/deps/npm/doc/api/tag.md @@ -0,0 +1,23 @@ +npm-tag(3) -- Tag a published version +===================================== + +## SYNOPSIS + + npm.commands.tag(package@version, tag, callback) + +## DESCRIPTION + +Tags the specified version of the package with the specified tag, or the +`--tag` config if not specified. + +The 'package@version' is an array of strings, but only the first two elements are +currently used. + +The first element must be in the form package@version, where package +is the package name and version is the version number (much like installing a +specific version). + +The second element is the name of the tag to tag this version with. If this +parameter is missing or falsey (empty), the default froom the config will be +used. For more information about how to set this config, check +`man 3 npm-config` for programmatic usage or `man npm-config` for cli usage. diff --git a/deps/npm/doc/api/test.md b/deps/npm/doc/api/test.md new file mode 100644 index 0000000000..bc48dcc35f --- /dev/null +++ b/deps/npm/doc/api/test.md @@ -0,0 +1,16 @@ +npm-test(3) -- Test a package +============================= + +## SYNOPSIS + + npm.commands.test(packages, callback) + +## DESCRIPTION + +This runs a package's "test" script, if one was provided. + +To run tests as a condition of installation, set the `npat` config to +true. + +npm can run tests on multiple packages. Just specify multiple packages +in the `packages` parameter. diff --git a/deps/npm/doc/api/uninstall.md b/deps/npm/doc/api/uninstall.md new file mode 100644 index 0000000000..4505295b3b --- /dev/null +++ b/deps/npm/doc/api/uninstall.md @@ -0,0 +1,16 @@ +npm-uninstall(3) -- uninstall a package programmatically +======================================================== + +## SYNOPSIS + + npm.commands.uninstall(packages, callback) + +## DESCRIPTION + +This acts much the same ways as uninstalling on the command-line. + +The 'packages' parameter is an array of strings. Each element in the array is +the name of a package to be uninstalled. + +Finally, 'callback' is a function that will be called when all packages have been +uninstalled or when an error has been encountered. diff --git a/deps/npm/doc/api/unpublish.md b/deps/npm/doc/api/unpublish.md new file mode 100644 index 0000000000..6cbc5c1f37 --- /dev/null +++ b/deps/npm/doc/api/unpublish.md @@ -0,0 +1,20 @@ +npm-unpublish(3) -- Remove a package from the registry +====================================================== + +## SYNOPSIS + + npm.commands.unpublish(package, callback) + +## DESCRIPTION + +This removes a package version from the registry, deleting its +entry and removing the tarball. + +The package parameter must be defined. + +Only the first element in the package parameter is used. If there is no first +element, then npm assumes that the package at the current working directory +is what is meant. + +If no version is specified, or if all versions are removed then +the root package entry is removed from the registry entirely. diff --git a/deps/npm/doc/api/update.md b/deps/npm/doc/api/update.md new file mode 100644 index 0000000000..bf02fd3c84 --- /dev/null +++ b/deps/npm/doc/api/update.md @@ -0,0 +1,11 @@ +npm-update(3) -- Update a package +================================= + +## SYNOPSIS + npm.commands.update(packages, callback) + +# DESCRIPTION + +Updates a package, upgrading it to the latest version. It also installs any missing packages. + +The 'packages' argument is an array of packages to update. The 'callback' parameter will be called when done or when an error occurs. diff --git a/deps/npm/doc/api/version.md b/deps/npm/doc/api/version.md new file mode 100644 index 0000000000..bd40102c45 --- /dev/null +++ b/deps/npm/doc/api/version.md @@ -0,0 +1,18 @@ +npm-version(3) -- Bump a package version +======================================== + +## SYNOPSIS + + npm.commands.version(newversion, callback) + +## DESCRIPTION + +Run this in a package directory to bump the version and write the new +data back to the package.json file. + +If run in a git repo, it will also create a version commit and tag, and +fail if the repo is not clean. + +Like all other commands, this function takes a string array as its first +parameter. The difference, however, is this function will fail if it does +not have exactly one element. The only element should be a version number. diff --git a/deps/npm/doc/api/view.md b/deps/npm/doc/api/view.md new file mode 100644 index 0000000000..fd0076c967 --- /dev/null +++ b/deps/npm/doc/api/view.md @@ -0,0 +1,93 @@ +npm-view(3) -- View registry info +================================= + +## SYNOPSIS + + npm.commands.view(args, [silent,] callback) + +## DESCRIPTION + +This command shows data about a package and prints it to the stream +referenced by the `outfd` config, which defaults to stdout. + +The "args" parameter is an ordered list that closely resembles the command-line +usage. The elements should be ordered such that the first element is +the package and version (package@version). The version is optional. After that, +the rest of the parameters are fields with optional subfields ("field.subfield") +which can be used to get only the information desired from the registry. + +The callback will be passed all of the data returned by the query. + +For example, to get the package registry entry for the `connect` package, +you can do this: + + npm.commands.view(["connect"], callback) + +If no version is specified, "latest" is assumed. + +Field names can be specified after the package descriptor. +For example, to show the dependencies of the `ronn` package at version +0.3.5, you could do the following: + + npm.commands.view(["ronn@0.3.5", "dependencies"], callback) + +You can view child field by separating them with a period. +To view the git repository URL for the latest version of npm, you could +do this: + + npm.commands.view(["npm", "repository.url"], callback) + +For fields that are arrays, requesting a non-numeric field will return +all of the values from the objects in the list. For example, to get all +the contributor names for the "express" project, you can do this: + + npm.commands.view(["express", "contributors.email"], callback) + +You may also use numeric indices in square braces to specifically select +an item in an array field. To just get the email address of the first +contributor in the list, you can do this: + + npm.commands.view(["express", "contributors[0].email"], callback) + +Multiple fields may be specified, and will be printed one after another. +For exampls, to get all the contributor names and email addresses, you +can do this: + + npm.commands.view(["express", "contributors.name", "contributors.email"], callback) + +"Person" fields are shown as a string if they would be shown as an +object. So, for example, this will show the list of npm contributors in +the shortened string format. (See `npm help json` for more on this.) + + npm.commands.view(["npm", "contributors"], callback) + +If a version range is provided, then data will be printed for every +matching version of the package. This will show which version of jsdom +was required by each matching version of yui3: + + npm.commands.view(["yui3@'>0.5.4'", "dependencies.jsdom"], callback) + +## OUTPUT + +If only a single string field for a single version is output, then it +will not be colorized or quoted, so as to enable piping the output to +another command. + +If the version range matches multiple versions, than each printed value +will be prefixed with the version it applies to. + +If multiple fields are requested, than each of them are prefixed with +the field name. + +Console output can be disabled by setting the 'silent' parameter to true. + +## RETURN VALUE + +The data returned will be an object in this formation: + + { : + { : + , ... } + , ... } + +corresponding to the list of fields selected. diff --git a/deps/npm/doc/api/whoami.md b/deps/npm/doc/api/whoami.md new file mode 100644 index 0000000000..598a1ab1a3 --- /dev/null +++ b/deps/npm/doc/api/whoami.md @@ -0,0 +1,15 @@ +npm-whoami(3) -- Display npm username +===================================== + +## SYNOPSIS + + npm.commands.whoami(args, callback) + +## DESCRIPTION + +Print the `username` config to standard output. + +'args' is never used and callback is never called with data. +'args' must be present or things will break. + +This function is not useful programmatically diff --git a/deps/npm/doc/cli/adduser.md b/deps/npm/doc/cli/adduser.md new file mode 100644 index 0000000000..51aa6f6a3d --- /dev/null +++ b/deps/npm/doc/cli/adduser.md @@ -0,0 +1,36 @@ +npm-adduser(1) -- Add a registry user account +============================================= + +## SYNOPSIS + + npm adduser + +## DESCRIPTION + +Create or verify a user named `` in the npm registry, and +save the credentials to the `.npmrc` file. + +The username, password, and email are read in from prompts. + +You may use this command to change your email address, but not username +or password. + +To reset your password, go to + +You may use this command multiple times with the same user account to +authorize on a new machine. + +## CONFIGURATION + +### registry + +Default: http://registry.npmjs.org/ + +The base URL of the npm package registry. + +## SEE ALSO + +* npm-registry(1) +* npm-config(1) +* npm-owner(1) +* npm-whoami(1) diff --git a/deps/npm/doc/cli/author.md b/deps/npm/doc/cli/author.md new file mode 120000 index 0000000000..b7a53cb66b --- /dev/null +++ b/deps/npm/doc/cli/author.md @@ -0,0 +1 @@ +owner.md \ No newline at end of file diff --git a/deps/npm/doc/cli/bin.md b/deps/npm/doc/cli/bin.md new file mode 100644 index 0000000000..2c2e7c4772 --- /dev/null +++ b/deps/npm/doc/cli/bin.md @@ -0,0 +1,17 @@ +npm-bin(1) -- Display npm bin folder +==================================== + +## SYNOPSIS + + npm bin + +## DESCRIPTION + +Print the folder where npm will install executables. + +## SEE ALSO + +* npm-prefix(1) +* npm-root(1) +* npm-folders(1) +* npm-config(1) diff --git a/deps/npm/doc/cli/bugs.md b/deps/npm/doc/cli/bugs.md new file mode 100644 index 0000000000..2e57cc891b --- /dev/null +++ b/deps/npm/doc/cli/bugs.md @@ -0,0 +1,38 @@ +npm-bugs(1) -- Bugs for a package in a web browser maybe +======================================================== + +## SYNOPSIS + + npm bugs + +## DESCRIPTION + +This command tries to guess at the likely location of a package's +bug tracker URL, and then tries to open it using the `--browser` +config param. + +## CONFIGURATION + +### browser + +* Default: OS X: `"open"`, others: `"google-chrome"` +* Type: String + +The browser that is called by the `npm bugs` command to open websites. + +### registry + +* Default: https://registry.npmjs.org/ +* Type: url + +The base URL of the npm package registry. + + +## SEE ALSO + +* npm-docs(1) +* npm-view(1) +* npm-publish(1) +* npm-registry(1) +* npm-config(1) +* npm-json(1) diff --git a/deps/npm/doc/cli/build.md b/deps/npm/doc/cli/build.md new file mode 100644 index 0000000000..978f4a6d62 --- /dev/null +++ b/deps/npm/doc/cli/build.md @@ -0,0 +1,22 @@ +npm-build(1) -- Build a package +=============================== + +## SYNOPSIS + + npm build + +* ``: + A folder containing a `package.json` file in its root. + +## DESCRIPTION + +This is the plumbing command called by `npm link` and `npm install`. + +It should generally not be called directly. + +## SEE ALSO + +* npm-install(1) +* npm-link(1) +* npm-scripts(1) +* npm-json(1) diff --git a/deps/npm/doc/cli/bundle.md b/deps/npm/doc/cli/bundle.md new file mode 100644 index 0000000000..69b3d83e45 --- /dev/null +++ b/deps/npm/doc/cli/bundle.md @@ -0,0 +1,14 @@ +npm-bundle(1) -- REMOVED +======================== + +## DESCRIPTION + +The `npm bundle` command has been removed in 1.0, for the simple reason +that it is no longer necessary, as the default behavior is now to +install packages into the local space. + +Just use `npm install` now to do what `npm bundle` used to do. + +## SEE ALSO + +* npm-install(1) diff --git a/deps/npm/doc/cli/cache.md b/deps/npm/doc/cli/cache.md new file mode 100644 index 0000000000..1fa128ad44 --- /dev/null +++ b/deps/npm/doc/cli/cache.md @@ -0,0 +1,70 @@ +npm-cache(1) -- Manipulates packages cache +========================================== + +## SYNOPSIS + + npm cache add + npm cache add + npm cache add + npm cache add @ + + npm cache ls [] + + npm cache clean [] + +## DESCRIPTION + +Used to add, list, or clear the npm cache folder. + +* add: + Add the specified package to the local cache. This command is primarily + intended to be used internally by npm, but it can provide a way to + add data to the local installation cache explicitly. + +* ls: + Show the data in the cache. Argument is a path to show in the cache + folder. Works a bit like the `find` program, but limited by the + `depth` config. + +* clean: + Delete data out of the cache folder. If an argument is provided, then + it specifies a subpath to delete. If no argument is provided, then + the entire cache is cleared. + +## DETAILS + +npm stores cache data in `$HOME/.npm`. For each package that is added +to the cache, three pieces of information are stored in +`{cache}/{name}/{version}`: + +* .../package/: + A folder containing the package contents as they appear in the tarball. +* .../package.json: + The package.json file, as npm sees it, with overlays applied and a _id attribute. +* .../package.tgz: + The tarball for that version. + +Additionally, whenever a registry request is made, a `.cache.json` file +is placed at the corresponding URI, to store the ETag and the requested +data. + +Commands that make non-essential registry requests (such as `search` and +`view`, or the completion scripts) generally specify a minimum timeout. +If the `.cache.json` file is younger than the specified timeout, then +they do not make an HTTP request to the registry. + +## CONFIGURATION + +### cache + +Default: `$HOME/.npm` on Posix, or `$HOME/npm-cache` on Windows. + +The root cache folder. + +## SEE ALSO + +* npm-folders(1) +* npm-config(1) +* npm-install(1) +* npm-publish(1) +* npm-pack(1) diff --git a/deps/npm/doc/cli/changelog.md b/deps/npm/doc/cli/changelog.md new file mode 100644 index 0000000000..0115405ca0 --- /dev/null +++ b/deps/npm/doc/cli/changelog.md @@ -0,0 +1,36 @@ +npm-changelog(1) -- Changes +=========================== + +## HISTORY + +### 1.0 +* Greatly simplified folder structure +* Install locally (bundle by default) +* Drastic rearchitecture + +### 0.3 +* More correct permission/uid handling when running as root +* Require node 0.4.0 +* Reduce featureset +* Packages without "main" modules don't export modules +* Remove support for invalid JSON (since node doesn't support it) + +### 0.2 +* First allegedly "stable" release +* Most functionality implemented +* Used shim files and `name@version` symlinks +* Feature explosion +* Kind of a mess + +### 0.1 +* push to beta, and announce +* Solaris and Cygwin support + +### 0.0 +* Lots of sketches and false starts; abandoned a few times +* Core functionality established + +## SEE ALSO + +* npm(1) +* npm-faq(1) diff --git a/deps/npm/doc/cli/coding-style.md b/deps/npm/doc/cli/coding-style.md new file mode 100644 index 0000000000..f0640c85cd --- /dev/null +++ b/deps/npm/doc/cli/coding-style.md @@ -0,0 +1,190 @@ +npm-coding-style(1) -- npm's "funny" coding style +================================================= + +## DESCRIPTION + +npm's coding style is a bit unconventional. It is not different for +difference's sake, but rather a carefully crafted style that is +designed to reduce visual clutter and make bugs more apparent. + +If you want to contribute to npm (which is very encouraged), you should +make your code conform to npm's style. + +## Line Length + +Keep lines shorter than 80 characters. It's better for lines to be +too short than to be too long. Break up long lists, objects, and other +statements onto multiple lines. + +## Indentation + +Two-spaces. Tabs are better, but they look like hell in web browsers +(and on github), and node uses 2 spaces, so that's that. + +Configure your editor appropriately. + +## Curly braces + +Curly braces belong on the same line as the thing that necessitates them. + +Bad: + + function () + { + +Good: + + function () { + +If a block needs to wrap to the next line, use a curly brace. Don't +use it if it doesn't. + +Bad: + + if (foo) { bar() } + while (foo) + bar() + +Good: + + if (foo) bar() + while (foo) { + bar() + } + +## Semicolons + +Don't use them except in four situations: + +* `for (;;)` loops. They're actually required. +* null loops like: `while (something) ;` (But you'd better have a good + reason for doing that.) +* case "foo": doSomething(); break +* In front of a leading ( or [ at the start of the line. + This prevents the expression from being interpreted + as a function call or property access, respectively. + +Some examples of good semicolon usage: + + ;(x || y).doSomething() + ;[a, b, c].forEach(doSomething) + for (var i = 0; i < 10; i ++) { + switch (state) { + case "begin": start(); continue + case "end": finish(); break + default: throw new Error("unknown state") + } + end() + } + +Note that starting lines with `-` and `+` also should be prefixed +with a semicolon, but this is much less common. + +## Comma First + +If there is a list of things separated by commas, and it wraps +across multiple lines, put the comma at the start of the next +line, directly below the token that starts the list. Put the +final token in the list on a line by itself. For example: + + var magicWords = [ "abracadabra" + , "gesundheit" + , "ventrilo" + ] + , spells = { "fireball" : function () { setOnFire() } + , "water" : function () { putOut() } + } + , a = 1 + , b = "abc" + , etc + , somethingElse + +## Whitespace + +Put a single space in front of ( for anything other than a function call. +Also use a single space wherever it makes things more readable. + +Don't leave trailing whitespace at the end of lines. Don't indent empty +lines. Don't use more spaces than are helpful. + +## Functions + +Use named functions. They make stack traces a lot easier to read. + +## Callbacks, Sync/async Style + +Use the asynchronous/non-blocking versions of things as much as possible. +It might make more sense for npm to use the synchronous fs APIs, but this +way, the fs and http and child process stuff all uses the same callback-passing +methodology. + +The callback should always be the last argument in the list. Its first +argument is the Error or null. + +Be very careful never to ever ever throw anything. It's worse than useless. +Just send the error message back as the first argument to the callback. + +## Errors + +Always create a new Error object with your message. Don't just return a +string message to the callback. Stack traces are handy. + +Use the `require("./utils/log").er` function. It takes a callback and an +error message, and returns an object that will report the message in the +event of a failure. It's quite handy. + + function myThing (args, cb) { + getData(args, function (er, data) { + if (er) return log.er(cb, "Couldn't get data")(er) + doSomethingElse(data, cb) + }) + } + function justHasToWork (cb) { + doSomething(log.er(cb, "the doSomething failed.")) + } + +## Logging + +Please clean up logs when they are no longer helpful. In particular, +logging the same object over and over again is not helpful. Logs should +report what's happening so that it's easier to track down where a fault +occurs. + +Use appropriate log levels. The default log() function logs at the +"info" level. See `npm-config(1)` and search for "loglevel". + +## Case, naming, etc. + +Use lowerCamelCase for multiword identifiers when they refer to objects, +functions, methods, members, or anything not specified in this section. + +Use UpperCamelCase for class names (things that you'd pass to "new"). + +Use all-lower-hyphen-css-case for multiword filenames and config keys. + +Use named functions. They make stack traces easier to follow. + +Use CAPS_SNAKE_CASE for constants, things that should never change +and are rarely used. + +Use a single uppercase letter for function names where the function +would normally be anonymous, but needs to call itself recursively. It +makes it clear that it's a "throwaway" function. + +## null, undefined, false, 0 + +Boolean variables and functions should always be either `true` or +`false`. Don't set it to 0 unless it's supposed to be a number. + +When something is intentionally missing or removed, set it to `null`. + +Don't set things to `undefined`. Reserve that value to mean "not yet +set to anything." + +Boolean objects are verboten. + +## SEE ALSO + +* npm-developers(1) +* npm-faq(1) +* npm(1) diff --git a/deps/npm/doc/cli/completion.md b/deps/npm/doc/cli/completion.md new file mode 100644 index 0000000000..48bc50fd87 --- /dev/null +++ b/deps/npm/doc/cli/completion.md @@ -0,0 +1,29 @@ +npm-completion(1) -- Tab Completion for npm +=========================================== + +## SYNOPSIS + + . <(npm completion) + +## DESCRIPTION + +Enables tab-completion in all npm commands. + +The synopsis above +loads the completions into your current shell. Adding it to +your ~/.bashrc or ~/.zshrc will make the completions available +everywhere. + +You may of course also pipe the output of npm completion to a file +such as `/usr/local/etc/bash_completion.d/npm` if you have a system +that will read that file for you. + +When `COMP_CWORD`, `COMP_LINE`, and `COMP_POINT` are defined in the +environment, `npm completion` acts in "plumbing mode", and outputs +completions based on the arguments. + +## SEE ALSO + +* npm-developers(1) +* npm-faq(1) +* npm(1) diff --git a/deps/npm/doc/cli/config.md b/deps/npm/doc/cli/config.md new file mode 100644 index 0000000000..1ede36292e --- /dev/null +++ b/deps/npm/doc/cli/config.md @@ -0,0 +1,665 @@ +npm-config(1) -- Manage the npm configuration file +================================================== + +## SYNOPSIS + + npm config set [--global] + npm config get + npm config delete + npm config list + npm config edit + npm get + npm set [--global] + +## DESCRIPTION + +npm gets its configuration values from 6 sources, in this priority: + +### Command Line Flags + +Putting `--foo bar` on the command line sets the +`foo` configuration parameter to `"bar"`. A `--` argument tells the cli +parser to stop reading flags. A `--flag` parameter that is at the *end* of +the command will be given the value of `true`. + +### Environment Variables + +Any environment variables that start with `npm_config_` will be interpreted +as a configuration parameter. For example, putting `npm_config_foo=bar` in +your environment will set the `foo` configuration parameter to `bar`. Any +environment configurations that are not given a value will be given the value +of `true`. Config values are case-insensitive, so `NPM_CONFIG_FOO=bar` will +work the same. + +### Per-user config file + +`$HOME/.npmrc` (or the `userconfig` param, if set above) + +This file is an ini-file formatted list of `key = value` parameters. + +### Global config file + +`$PREFIX/etc/npmrc` (or the `globalconfig` param, if set above): +This file is an ini-file formatted list of `key = value` parameters + +### Built-in config file + +`path/to/npm/itself/npmrc` + +This is an unchangeable "builtin" +configuration file that npm keeps consistent across updates. Set +fields in here using the `./configure` script that comes with npm. +This is primarily for distribution maintainers to override default +configs in a standard and consistent manner. + +### Default Configs + +A set of configuration parameters that are internal to npm, and are +defaults if nothing else is specified. + +## Sub-commands + +Config supports the following sub-commands: + +### set + + npm config set key value + +Sets the config key to the value. + +If value is omitted, then it sets it to "true". + +### get + + npm config get key + +Echo the config value to stdout. + +### list + + npm config list + +Show all the config settings. + +### delete + + npm config delete key + +Deletes the key from all configuration files. + +### edit + + npm config edit + +Opens the config file in an editor. Use the `--global` flag to edit the +global config. + +## Shorthands and Other CLI Niceties + +The following shorthands are parsed on the command-line: + +* `-v`: `--version` +* `-h`, `-?`, `--help`, `-H`: `--usage` +* `-s`, `--silent`: `--loglevel silent` +* `-d`: `--loglevel info` +* `-dd`, `--verbose`: `--loglevel verbose` +* `-ddd`: `--loglevel silly` +* `-g`: `--global` +* `-l`: `--long` +* `-m`: `--message` +* `-p`, `--porcelain`: `--parseable` +* `-reg`: `--registry` +* `-v`: `--version` +* `-f`: `--force` +* `-l`: `--long` +* `-desc`: `--description` +* `-S`: `--save` +* `-y`: `--yes` +* `-n`: `--yes false` +* `ll` and `la` commands: `ls --long` + +If the specified configuration param resolves unambiguously to a known +configuration parameter, then it is expanded to that configuration +parameter. For example: + + npm ls --par + # same as: + npm ls --parseable + +If multiple single-character shorthands are strung together, and the +resulting combination is unambiguously not some other configuration +param, then it is expanded to its various component pieces. For +example: + + npm ls -gpld + # same as: + npm ls --global --parseable --long --loglevel info + +## Per-Package Config Settings + +When running scripts (see `npm-scripts(1)`) +the package.json "config" keys are overwritten in the environment if +there is a config param of `[@]:`. For example, if +the package.json has this: + + { "name" : "foo" + , "config" : { "port" : "8080" } + , "scripts" : { "start" : "node server.js" } } + +and the server.js is this: + + http.createServer(...).listen(process.env.npm_package_config_port) + +then the user could change the behavior by doing: + + npm config set foo:port 80 + +## Config Settings + +### always-auth + +* Default: false +* Type: Boolean + +Force npm to always require authentication when accessing the registry, +even for `GET` requests. + +### bin-publish + +* Default: false +* Type: Boolean + +If set to true, then binary packages will be created on publish. + +This is the way to opt into the "bindist" behavior described below. + +### bindist + +* Default: Unstable node versions, `null`, otherwise + `"--"` +* Type: String or `null` + +Experimental: on stable versions of node, binary distributions will be +created with this tag. If a user then installs that package, and their +`bindist` tag is found in the list of binary distributions, they will +get that prebuilt version. + +Pre-build node packages have their preinstall, install, and postinstall +scripts stripped (since they are run prior to publishing), and do not +have their `build` directories automatically ignored. + +It's yet to be seen if this is a good idea. + +### browser + +* Default: OS X: `"open"`, others: `"google-chrome"` +* Type: String + +The browser that is called by the `npm docs` command to open websites. + +### ca + +* Default: The npm CA certificate +* Type: String or null + +The Certificate Authority signing certificate that is trusted for SSL +connections to the registry. + +Set to `null` to only allow "known" registrars, or to a specific CA cert +to trust only that specific signing authority. + +See also the `strict-ssl` config. + +### cache + +* Default: Windows: `~/npm-cache`, Posix: `~/.npm` +* Type: path + +The location of npm's cache directory. See `npm-cache(1)` + +### color + +* Default: true on Posix, false on Windows +* Type: Boolean or `"always"` + +If false, never shows colors. If `"always"` then always shows colors. +If true, then only prints color codes for tty file descriptors. + +### depth + +* Default: Infinity +* Type: Number + +The depth to go when recursing directories for `npm ls` and +`npm cache ls`. + +### description + +* Default: true +* Type: Boolean + +Show the description in `npm search` + +### dev + +* Default: false +* Type: Boolean + +Install `dev-dependencies` along with packages. + +Note that `dev-dependencies` are also installed if the `npat` flag is +set. + +### editor + +* Default: `EDITOR` environment variable if set, or `"vi"` on Posix, + or `"notepad"` on Windows. +* Type: path + +The command to run for `npm edit` or `npm config edit`. + +### force + +* Default: false +* Type: Boolean + +Makes various commands more forceful. + +* lifecycle script failure does not block progress. +* publishing clobbers previously published versions. +* skips cache when requesting from the registry. +* prevents checks against clobbering non-npm files. + +### global + +* Default: false +* Type: Boolean + +Operates in "global" mode, so that packages are installed into the +`prefix` folder instead of the current working directory. See +`npm-folders(1)` for more on the differences in behavior. + +* packages are installed into the `prefix/node_modules` folder, instead of the + current working directory. +* bin files are linked to `prefix/bin` +* man pages are linked to `prefix/share/man` + +### globalconfig + +* Default: {prefix}/etc/npmrc +* Type: path + +The config file to read for global config options. + +### globalignorefile + +* Default: {prefix}/etc/npmignore +* Type: path + +The config file to read for global ignore patterns to apply to all users +and all projects. + +If not found, but there is a "gitignore" file in the +same directory, then that will be used instead. + +### group + +* Default: GID of the current process +* Type: String or Number + +The group to use when running package scripts in global mode as the root +user. + +### https-proxy + +* Default: the `HTTPS_PROXY` or `https_proxy` or `HTTP_PROXY` or + `http_proxy` environment variables. +* Type: url + +A proxy to use for outgoing https requests. + +### ignore + +* Default: "" +* Type: string + +A white-space separated list of glob patterns of files to always exclude +from packages when building tarballs. + +### init.version + +* Default: "0.0.0" +* Type: semver + +The value `npm init` should use by default for the package version. + +### init.author.name + +* Default: "0.0.0" +* Type: String + +The value `npm init` should use by default for the package author's name. + +### init.author.email + +* Default: "" +* Type: String + +The value `npm init` should use by default for the package author's email. + +### init.author.url + +* Default: "" +* Type: String + +The value `npm init` should use by default for the package author's homepage. + +### link + +* Default: false +* Type: Boolean + +If true, then local installs will link if there is a suitable globally +installed package. + +Note that this means that local installs can cause things to be +installed into the global space at the same time. The link is only done +if one of the two conditions are met: + +* The package is not already installed globally, or +* the globally installed version is identical to the version that is + being installed locally. + +### logfd + +* Default: stderr file descriptor +* Type: Number or Stream + +The location to write log output. + +### loglevel + +* Default: "warn" +* Type: String +* Values: "silent", "win", "error", "warn", "info", "verbose", "silly" + +What level of logs to report. On failure, *all* logs are written to +`npm-debug.log` in the current working directory. + +### logprefix + +* Default: true on Posix, false on Windows +* Type: Boolean + +Whether or not to prefix log messages with "npm" and the log level. See +also "color" and "loglevel". + +### long + +* Default: false +* Type: Boolean + +Show extended information in `npm ls` + +### message + +* Default: "%s" +* Type: String + +Commit message which is used by `npm version` when creating version commit. + +Any "%s" in the message will be replaced with the version number. + +### node-version + +* Default: process.version +* Type: semver or false + +The node version to use when checking package's "engines" hash. + +### npat + +* Default: false +* Type: Boolean + +Run tests on installation and report results to the +`npaturl`. + +### npaturl + +* Default: Not yet implemented +* Type: url + +The url to report npat test results. + +### onload-script + +* Default: false +* Type: path + +A node module to `require()` when npm loads. Useful for programmatic +usage. + +### outfd + +* Default: standard output file descriptor +* Type: Number or Stream + +Where to write "normal" output. This has no effect on log output. + +### parseable + +* Default: false +* Type: Boolean + +Output parseable results from commands that write to +standard output. + +### prefix + +* Default: node's process.installPrefix +* Type: path + +The location to install global items. If set on the command line, then +it forces non-global commands to run in the specified folder. + +### production + +* Default: false +* Type: Boolean + +Set to true to run in "production" mode. + +1. devDependencies are not installed at the topmost level when running + local `npm install` without any arguments. +2. Set the NODE_ENV="production" for lifecycle scripts. + +### proxy + +* Default: `HTTP_PROXY` or `http_proxy` environment variable, or null +* Type: url + +A proxy to use for outgoing http requests. + +### rebuild-bundle + +* Default: true +* Type: Boolean + +Rebuild bundled dependencies after installation. + +### registry + +* Default: https://registry.npmjs.org/ +* Type: url + +The base URL of the npm package registry. + +### rollback + +* Default: true +* Type: Boolean + +Remove failed installs. + +### save + +* Default: false +* Type: Boolean + +Save installed packages to a package.json file as dependencies. + +Only works if there is already a package.json file present. + +### searchopts + +* Default: "" +* Type: String + +Space-separated options that are always passed to search. + +### searchexclude + +* Default: "" +* Type: String + +Space-separated options that limit the results from search. + +### shell + +* Default: SHELL environment variable, or "bash" on Posix, or "cmd" on + Windows +* Type: path + +The shell to run for the `npm explore` command. + +### strict-ssl + +* Default: true +* Type: Boolean + +Whether or not to do SSL key validation when making requests to the +registry via https. + +See also the `ca` config. + +### tag + +* Default: latest +* Type: String + +If you ask npm to install a package and don't tell it a specific version, then +it will install the specified tag. + +Also the tag that is added to the package@version specified by the `npm +tag` command, if no explicit tag is given. + +### tmp + +* Default: TMPDIR environment variable, or "/tmp" +* Type: path + +Where to store temporary files and folders. All temp files are deleted +on success, but left behind on failure for forensic purposes. + +### unicode + +* Default: true +* Type: Boolean + +When set to true, npm uses unicode characters in the tree output. When +false, it uses ascii characters to draw trees. + +### unsafe-perm + +* Default: false if running as root, true otherwise +* Type: Boolean + +Set to true to suppress the UID/GID switching when running package +scripts. If set explicitly to false, then installing as a non-root user +will fail. + +### usage + +* Default: false +* Type: Boolean + +Set to show short usage output (like the -H output) +instead of complete help when doing `npm-help(1)`. + +### user + +* Default: "nobody" +* Type: String or Number + +The UID to set to when running package scripts as root. + +### username + +* Default: null +* Type: String + +The username on the npm registry. Set with `npm adduser` + +### userconfig + +* Default: ~/.npmrc +* Type: path + +The location of user-level configuration settings. + +### userignorefile + +* Default: ~/.npmignore +* Type: path + +The location of a user-level ignore file to apply to all packages. + +If not found, but there is a .gitignore file in the same directory, then +that will be used instead. + +### umask + +* Default: 022 +* Type: Octal numeric string + +The "umask" value to use when setting the file creation mode on files +and folders. + +Folders and executables are given a mode which is `0777` masked against +this value. Other files are given a mode which is `0666` masked against +this value. Thus, the defaults are `0755` and `0644` respectively. + +### version + +* Default: false +* Type: boolean + +If true, output the npm version and exit successfully. + +Only relevant when specified explicitly on the command line. + +### viewer + +* Default: "man" on Posix, "browser" on Windows +* Type: path + +The program to use to view help content. + +Set to `"browser"` to view html help content in the default web browser. + +### yes + +* Default: null +* Type: Boolean or null + +If set to `null`, then prompt the user for responses in some +circumstances. + +If set to `true`, then answer "yes" to any prompt. If set to `false` +then answer "no" to any prompt. + +## SEE ALSO + +* npm-folders(1) +* npm(1) diff --git a/deps/npm/doc/cli/deprecate.md b/deps/npm/doc/cli/deprecate.md new file mode 100644 index 0000000000..925337f21a --- /dev/null +++ b/deps/npm/doc/cli/deprecate.md @@ -0,0 +1,24 @@ +npm-deprecate(1) -- Deprecate a version of a package +==================================================== + +## SYNOPSIS + + npm deprecate [@] + +## DESCRIPTION + +This command will update the npm registry entry for a package, providing +a deprecation warning to all who attempt to install it. + +It works on version ranges as well as specific versions, so you can do +something like this: + + npm deprecate my-thing@"< 0.2.3" "critical bug fixed in v0.2.3" + +Note that you must be the package owner to deprecate something. See the +`owner` and `adduser` help topics. + +## SEE ALSO + +* npm-publish(1) +* npm-registry(1) diff --git a/deps/npm/doc/cli/developers.md b/deps/npm/doc/cli/developers.md new file mode 100644 index 0000000000..0f0f94c588 --- /dev/null +++ b/deps/npm/doc/cli/developers.md @@ -0,0 +1,172 @@ +npm-developers(1) -- Developer Guide +==================================== + +## DESCRIPTION + +So, you've decided to use npm to develop (and maybe publish/deploy) +your project. + +Fantastic! + +There are a few things that you need to do above the simple steps +that your users will do to install your program. + +## About These Documents + +These are man pages. If you install npm, you should be able to +then do `man npm-thing` to get the documentation on a particular +topic, or `npm help thing` to see the same information. + +## What is a `package` + +A package is: + +* a) a folder containing a program described by a package.json file +* b) a gzipped tarball containing (a) +* c) a url that resolves to (b) +* d) a `@` that is published on the registry with (c) +* e) a `@` that points to (d) +* f) a `` that has a "latest" tag satisfying (e) + +Even if you never publish your package, you can still get a lot of +benefits of using npm if you just want to write a node program (a), and +perhaps if you also want to be able to easily install it elsewhere +after packing it up into a tarball (b). + +## The package.json File + +You need to have a `package.json` file in the root of your project to do +much of anything with npm. That is basically the whole interface. + +See `npm-json(1)` for details about what goes in that file. At the very +least, you need: + +* name: + This should be a string that identifies your project. Please do not + use the name to specify that it runs on node, or is in JavaScript. + You can use the "engines" field to explicitly state the versions of + node (or whatever else) that your program requires, and it's pretty + well assumed that it's javascript. + + It does not necessarily need to match your github repository name. + + So, `node-foo` and `bar-js` are bad names. `foo` or `bar` are better. + +* version: + A semver-compatible version. + +* engines: + Specify the versions of node (or whatever else) that your program + runs on. The node API changes a lot, and there may be bugs or new + functionality that you depend on. Be explicit. + +* author: + Take some credit. + +* scripts: + If you have a special compilation or installation script, then you + should put it in the `scripts` hash. You should definitely have at + least a basic smoke-test command as the "scripts.test" field. + See npm-scripts(1). + +* main: + If you have a single module that serves as the entry point to your + program (like what the "foo" package gives you at require("foo")), + then you need to specify that in the "main" field. + +* directories: + This is a hash of folders. The best ones to include are "lib" and + "doc", but if you specify a folder full of man pages in "man", then + they'll get installed just like these ones. + +You can use `npm init` in the root of your package in order to get you +started with a pretty basic package.json file. See `npm-init(1)` for +more info. + +## Keeping files *out* of your package + +Use a `.npmignore` file to keep stuff out of your package. If there's +no .npmignore file, but there *is* a .gitignore file, then npm will +ignore the stuff matched by the .gitignore file. If you *want* to +include something that is excluded by your .gitignore file, you can +create an empty .npmignore file to override it. + +## Link Packages + +`npm link` is designed to install a development package and see the +changes in real time without having to keep re-installing it. (You do +need to either re-link or `npm rebuild -g` to update compiled packages, +of course.) + +More info at `npm-link(1)`. + +## Before Publishing: Make Sure Your Package Installs and Works + +**This is important.** + +If you can not install it locally, you'll have +problems trying to publish it. Or, worse yet, you'll be able to +publish it, but you'll be publishing a broken or pointless package. +So don't do that. + +In the root of your package, do this: + + npm install . -g + +That'll show you that it's working. If you'd rather just create a symlink +package that points to your working directory, then do this: + + npm link + +Use `npm ls -g` to see if it's there. + +To test a local install, go into some other folder, and then do: + + cd ../some-other-folder + npm install ../my-package + +to install it locally into the node_modules folder in that other place. + +Then go into the node-repl, and try using require("my-thing") to +bring in your module's main module. + +## Create a User Account + +Create a user with the adduser command. It works like this: + + npm adduser + +and then follow the prompts. + +This is documented better in npm-adduser(1). + +## Publish your package + +This part's easy. IN the root of your folder, do this: + + npm publish + +You can give publish a url to a tarball, or a filename of a tarball, +or a path to a folder. + +Note that pretty much **everything in that folder will be exposed** +by default. So, if you have secret stuff in there, use a `.npminclude` +or `.npmignore` file to list out the globs to include/ignore, or publish +from a fresh checkout. + +## Brag about it + +Send emails, write blogs, blab in IRC. + +Tell the world how easy it is to install your program! + +## SEE ALSO + +* npm-faq(1) +* npm(1) +* npm-init(1) +* npm-json(1) +* npm-scripts(1) +* npm-publish(1) +* npm-adduser(1) +* npm-registry(1) diff --git a/deps/npm/doc/cli/docs.md b/deps/npm/doc/cli/docs.md new file mode 100644 index 0000000000..26b2455dd8 --- /dev/null +++ b/deps/npm/doc/cli/docs.md @@ -0,0 +1,38 @@ +npm-docs(1) -- Docs for a package in a web browser maybe +======================================================== + +## SYNOPSIS + + npm docs + npm home + +## DESCRIPTION + +This command tries to guess at the likely location of a package's +documentation URL, and then tries to open it using the `--browser` +config param. + +## CONFIGURATION + +### browser + +* Default: OS X: `"open"`, others: `"google-chrome"` +* Type: String + +The browser that is called by the `npm docs` command to open websites. + +### registry + +* Default: https://registry.npmjs.org/ +* Type: url + +The base URL of the npm package registry. + + +## SEE ALSO + +* npm-view(1) +* npm-publish(1) +* npm-registry(1) +* npm-config(1) +* npm-json(1) diff --git a/deps/npm/doc/cli/edit.md b/deps/npm/doc/cli/edit.md new file mode 100644 index 0000000000..9eaccfc540 --- /dev/null +++ b/deps/npm/doc/cli/edit.md @@ -0,0 +1,35 @@ +npm-edit(1) -- Edit an installed package +======================================== + +## SYNOPSIS + + npm edit [@] + +## DESCRIPTION + +Opens the package folder in the default editor (or whatever you've +configured as the npm `editor` config -- see `npm-config(1)`.) + +After it has been edited, the package is rebuilt so as to pick up any +changes in compiled packages. + +For instance, you can do `npm install connect` to install connect +into your package, and then `npm edit connect` to make a few +changes to your locally installed copy. + +## CONFIGURATION + +### editor + +* Default: `EDITOR` environment variable if set, or `"vi"` on Posix, + or `"notepad"` on Windows. +* Type: path + +The command to run for `npm edit` or `npm config edit`. + +## SEE ALSO + +* npm-folders(1) +* npm-explore(1) +* npm-install(1) +* npm-config(1) diff --git a/deps/npm/doc/cli/explore.md b/deps/npm/doc/cli/explore.md new file mode 100644 index 0000000000..00701b392a --- /dev/null +++ b/deps/npm/doc/cli/explore.md @@ -0,0 +1,40 @@ +npm-explore(1) -- Browse an installed package +============================================= + +## SYNOPSIS + + npm explore [@] [ -- ] + +## DESCRIPTION + +Spawn a subshell in the directory of the installed package specified. + +If a command is specified, then it is run in the subshell, which then +immediately terminates. + +This is particularly handy in the case of git submodules in the +`node_modules` folder: + + npm explore some-dependency -- git pull origin master + +Note that the package is *not* automatically rebuilt afterwards, so be +sure to use `npm rebuild ` if you make any changes. + +## CONFIGURATION + +### shell + +* Default: SHELL environment variable, or "bash" on Posix, or "cmd" on + Windows +* Type: path + +The shell to run for the `npm explore` command. + +## SEE ALSO + +* npm-submodule(1) +* npm-folders(1) +* npm-edit(1) +* npm-rebuild(1) +* npm-build(1) +* npm-install(1) diff --git a/deps/npm/doc/cli/faq.md b/deps/npm/doc/cli/faq.md new file mode 100644 index 0000000000..15bb0c637d --- /dev/null +++ b/deps/npm/doc/cli/faq.md @@ -0,0 +1,223 @@ +npm-faq(1) -- Frequently Asked Questions +======================================== + +## Where can I find these docs in HTML? + +, or run: + + npm config set viewer browser + +to open these documents in your default web browser rather than `man`. + +## It didn't work. + +That's not really a question. + +## Why didn't it work? + +I don't know yet. + +Read the error output, and if you can't figure out what it means, +do what it says and post a bug with all the information it asks for. + +## Where does npm put stuff? + +See `npm-folders(1)` + +tl;dr: + +* Use the `npm root` command to see where modules go, and the `npm bin` + command to see where executables go +* Global installs are different from local installs. If you install + something with the `-g` flag, then its executables go in `npm bin -g` + and its modules go in `npm root -g`. + +## How do I install something everywhere? + +Install it globally by tacking `-g` or `--global` to the command. + +## I installed something globally, but I can't `require()` it + +Install it locally. + +## I don't wanna. + +Check out `npm link`. You might like it. + +## No, I really want 0.x style 'everything global' style. + +Ok, fine. Do this: + + echo 'export NODE_PATH="'$(npm root -g)'"' >> ~/.bashrc + . ~/.bashrc + npm config set global true + +This is not recommended. + +Many things **will not work** if you do this. Make sure you read and +understand `npm-config(1)` and `npm-global(1)` before you complain +about things being broken. + +When you realize what a mistake it was, do this to switch back: + + npm config delete global --local + +## If 'npm' is an acronym, why is it never capitalized? + +Contrary to the belief of many, "npm" is not in fact an abbreviation for +"Node Package Manager". It is a recursive bacronymic abbreviation for +"npm is not an acronym". (If it was "ninaa", then it would be an +acronym, and thus incorrectly named.) + +"NPM", however, *is* an acronym (more precisely, a capitonym) for the +National Association of Pastoral Musicians. You can learn more +about them at . + +In software, "NPM" is a non-parametric mapping utility written by +Chris Rorden. You can analyze pictures of brains with it. Learn more +about the (capitalized) NPM program at . + +The first seed that eventually grew into this flower was a bash utility +named "pm", which was a shortened descendent of "pkgmakeinst", a +bash function that was used to install various different things on different +platforms, most often using Yahoo's `yinst`. If `npm` was ever an +acronym for anything, it was `node pm` or maybe `new pm`. + +So, in all seriousness, the "npm" project is named after its command-line +utility, which was organically selected to be easily typed by a right-handed +programmer using a US QWERTY keyboard layout, ending with the +right-ring-finger in a postition to type the `-` key for flags and +other command-line arguments. That command-line utility is always +lower-case, though it starts most sentences it is a part of. + +## How do I list installed packages? + +`npm ls` + +## How do I search for packages? + +`npm search` + +Arguments are greps. `npm search jsdom` shows jsdom packages. + +## How do I update npm? + + npm update npm -g + +You can also update all outdated local packages by doing `npm update` without +any arguments, or global packages by doing `npm update -g`. + +Occasionally, the version of npm will progress such that the current +version cannot be properly installed with the version that you have +installed already. (Consider, if there is ever a bug in the `update` +command.) + +In those cases, you can do this: + + curl http://npmjs.org/install.sh | sh + +## What is a `package`? + +A package is: + +* a) a folder containing a program described by a package.json file +* b) a gzipped tarball containing (a) +* c) a url that resolves to (b) +* d) a `@` that is published on the registry with (c) +* e) a `@` that points to (d) +* f) a `` that has a "latest" tag satisfying (e) +* g) a `git` url that, when cloned, results in (a). + +Even if you never publish your package, you can still get a lot of +benefits of using npm if you just want to write a node program (a), and +perhaps if you also want to be able to easily install it elsewhere +after packing it up into a tarball (b). + +Git urls can be of the form: + + git://github.com/user/project.git#commit-ish + git+ssh://user@hostname:project.git#commit-ish + git+http://user@hostname/project/blah.git#commit-ish + git+https://user@hostname/project/blah.git#commit-ish + +The `commit-ish` can be any tag, sha, or branch which can be supplied as +an argument to `git checkout`. The default is `master`. + +## How do I install node with npm? + +You don't. Try one of these: + +* +* +* + +## How can I use npm for development? + +See `npm-developers(1)` and `npm-json(1)`. + +You'll most likely want to `npm link` your development folder. That's +awesomely handy. + +To set up your own private registry, check out `npm-registry(1)`. + +## Can I list a url as a dependency? + +Yes. It should be a url to a gzipped tarball containing a single folder +that has a package.json in its root, or a git url. +(See "what is a package?" above.) + +## How do I symlink to a dev folder so I don't have to keep re-installing? + +See `npm-link(1)` + +## The package registry website. What is that exactly? + +See `npm-registry(1)`. + +## What's up with the insecure channel warnings? + +Until node 0.4.10, there were problems sending big files over HTTPS. That +means that publishes go over HTTP by default in those versions of node. + +## I forgot my password, and can't publish. How do I reset it? + +Go to . + +## I get ECONNREFUSED a lot. What's up? + +Either the registry is down, or node's DNS isn't able to reach out. +This happens a lot if you don't follow *all* the steps in the Cygwin +setup doc. + +To check if the registry is down, open up + +in a web browser. This will also tell you if you are just unable to +access the internet for some reason. + +If the registry IS down, let me know by emailing or posting an issue. +We'll have someone kick it or something. + +## Who does npm? + +`npm view npm author` + +`npm view npm contributors` + +## I have a question or request not addressed here. Where should I put it? + +Discuss it on the mailing list, or post an issue. + +* +* + +## Why does npm hate me? + +npm is not capable of hatred. It loves everyone, especially you. + +## SEE ALSO + +* npm(1) +* npm-developers(1) +* npm-json(1) +* npm-config(1) +* npm-folders(1) diff --git a/deps/npm/doc/cli/find.md b/deps/npm/doc/cli/find.md new file mode 120000 index 0000000000..9b687d1c19 --- /dev/null +++ b/deps/npm/doc/cli/find.md @@ -0,0 +1 @@ +search.md \ No newline at end of file diff --git a/deps/npm/doc/cli/folders.md b/deps/npm/doc/cli/folders.md new file mode 100644 index 0000000000..20358612ad --- /dev/null +++ b/deps/npm/doc/cli/folders.md @@ -0,0 +1,209 @@ +npm-folders(1) -- Folder Structures Used by npm +=============================================== + +## DESCRIPTION + +npm puts various things on your computer. That's its job. + +This document will tell you what it puts where. + +### tl;dr + +* Local install (default): puts stuff in `./node_modules` of the current + package root. +* Global install (with `-g`): puts stuff in /usr/local or wherever node + is installed. +* Install it **locally** if you're going to `require()` it. +* Install it **globally** if you're going to run it on the command line. +* If you need both, then install it in both places, or use `npm link`. + +### prefix Configuration + +The `prefix` config defaults to the location where node is installed. +On most systems, this is `/usr/local`, and most of the time is the same +as node's `process.installPrefix`. + +On windows, this is the exact location of the node.exe binary. On Unix +systems, it's one level up, since node is typically installed at +`{prefix}/bin/node` rather than `{prefix}/node.exe`. + +When the `global` flag is set, npm installs things into this prefix. +When it is not set, it uses the root of the current package, or the +current working directory if not in a package already. + +### Node Modules + +Packages are dropped into the `node_modules` folder under the `prefix`. +When installing locally, this means that you can +`require("packagename")` to load its main module, or +`require("packagename/lib/path/to/sub/module")` to load other modules. + +Global installs on Unix systems go to `{prefix}/lib/node_modules`. +Global installs on Windows go to `{prefix}/node_modules` (that is, no +`lib` folder.) + +If you wish to `require()` a package, then install it locally. + +### Executables + +When in global mode, executables are linked into `{prefix}/bin` on Unix, +or directly into `{prefix}` on Windows. + +When in local mode, executables are linked into +`./node_modules/.bin` so that they can be made available to scripts run +through npm. (For example, so that a test runner will be in the path +when you run `npm test`.) + +### Man Pages + +When in global mode, man pages are linked into `{prefix}/share/man`. + +When in local mode, man pages are not installed. + +Man pages are not installed on Windows systems. + +### Cache + +See `npm-cache(1)`. Cache files are stored in `~/.npm` on Posix, or +`~/npm-cache` on Windows. + +This is controlled by the `cache` configuration param. + +### Temp Files + +Temporary files are stored by default in the folder specified by the +`tmp` config, which defaults to the TMPDIR, TMP, or TEMP environment +variables, or `/tmp` on Unix and `c:\windows\temp` on Windows. + +Temp files are given a unique folder under this root for each run of the +program, and are deleted upon successful exit. + +## More Information + +When installing locally, npm first tries to find an appropriate +`prefix` folder. This is so that `npm install foo@1.2.3` will install +to the sensible root of your package, even if you happen to have `cd`ed +into some other folder. + +Starting at the $PWD, npm will walk up the folder tree checking for a +folder that contains either a `package.json` file, or a `node_modules` +folder. If such a thing is found, then that is treated as the effective +"current directory" for the purpose of running npm commands. (This +behavior is inspired by and similar to git's .git-folder seeking +logic when running git commands in a working dir.) + +If no package root is found, then the current folder is used. + +When you run `npm install foo@1.2.3`, then the package is loaded into +the cache, and then unpacked into `./node_modules/foo`. Then, any of +foo's dependencies are similarly unpacked into +`./node_modules/foo/node_modules/...`. + +Any bin files are symlinked to `./node_modules/.bin/`, so that they may +be found by npm scripts when necessary. + +### Global Installation + +If the `global` configuration is set to true, then npm will +install packages "globally". + +For global installation, packages are installed roughly the same way, +but using the folders described above. + +### Cycles, Conflicts, and Folder Parsimony + +Cycles are handled using the property of node's module system that it +walks up the directories looking for `node_modules` folders. So, at every +stage, if a package is already installed in an ancestor `node_modules` +folder, then it is not installed at the current location. + +Consider the case above, where `foo -> bar -> baz`. Imagine if, in +addition to that, baz depended on bar, so you'd have: +`foo -> bar -> baz -> bar -> baz ...`. However, since the folder +structure is: `foo/node_modules/bar/node_modules/baz`, there's no need to +put another copy of bar into `.../baz/node_modules`, since when it calls +require("bar"), it will get the copy that is installed in +`foo/node_modules/bar`. + +This shortcut is only used if the exact same +version would be installed in multiple nested `node_modules` folders. It +is still possible to have `a/node_modules/b/node_modules/a` if the two +"a" packages are different versions. However, without repeating the +exact same package multiple times, an infinite regress will always be +prevented. + +Another optimization can be made by installing dependencies at the +highest level possible, below the localized "target" folder. + +#### Example + +Consider this dependency graph: + + foo + +-- blerg@1.2.5 + +-- bar@1.2.3 + | +-- blerg@1.x (latest=1.3.7) + | +-- baz@2.x + | | `-- quux@3.x + | | `-- bar@1.2.3 (cycle) + | `-- asdf@* + `-- baz@1.2.3 + `-- quux@3.x + `-- bar + +In this case, we might expect a folder structure like this: + + foo + +-- node_modules + +-- blerg (1.2.5) <---[A] + +-- bar (1.2.3) <---[B] + | +-- node_modules + | | `-- baz (2.0.2) <---[C] + | | `-- node_modules + | | `-- quux (3.2.0) + | `-- asdf (2.3.4) + `-- baz (1.2.3) <---[D] + `-- node_modules + `-- quux (3.2.0) <---[E] + +Since foo depends directly on bar@1.2.3 and baz@1.2.3, those are +installed in foo's `node_modules` folder. + +Even though the latest copy of blerg is 1.3.7, foo has a specific +dependency on version 1.2.5. So, that gets installed at [A]. Since the +parent installation of blerg satisfie's bar's dependency on blerg@1.x, +it does not install another copy under [B]. + +Bar [B] also has dependencies on baz and asdf, so those are installed in +bar's `node_modules` folder. Because it depends on `baz@2.x`, it cannot +re-use the `baz@1.2.3` installed in the parent `node_modules` folder [D], +and must install its own copy [C]. + +Underneath bar, the `baz->quux->bar` dependency creates a cycle. +However, because `bar` is already in `quux`'s ancestry [B], it does not +unpack another copy of bar into that folder. + +Underneath `foo->baz` [D], quux's [E] folder tree is empty, because its +dependency on bar is satisfied by the parent folder copy installed at [B]. + +For a graphical breakdown of what is installed where, use `npm ls`. + +### Publishing + +Upon publishing, npm will look in the `node_modules` folder. If any of +the items there are not in the `bundledDependencies` array, then they will +not be included in the package tarball. + +This allows a package maintainer to install all of their dependencies +(and dev dependencies) locally, but only re-publish those items that +cannot be found elsewhere. See `npm-json(1)` for more information. + +## SEE ALSO + +* npm-faq(1) +* npm-json(1) +* npm-install(1) +* npm-pack(1) +* npm-cache(1) +* npm-config(1) +* npm-publish(1) diff --git a/deps/npm/doc/cli/get.md b/deps/npm/doc/cli/get.md new file mode 120000 index 0000000000..3dc8737366 --- /dev/null +++ b/deps/npm/doc/cli/get.md @@ -0,0 +1 @@ +config.md \ No newline at end of file diff --git a/deps/npm/doc/cli/global.md b/deps/npm/doc/cli/global.md new file mode 120000 index 0000000000..c3598dd7df --- /dev/null +++ b/deps/npm/doc/cli/global.md @@ -0,0 +1 @@ +folders.md \ No newline at end of file diff --git a/deps/npm/doc/cli/help-search.md b/deps/npm/doc/cli/help-search.md new file mode 100644 index 0000000000..9c16901ebb --- /dev/null +++ b/deps/npm/doc/cli/help-search.md @@ -0,0 +1,35 @@ +npm-help-search(1) -- Search npm help documentation +=================================================== + +## SYNOPSIS + + npm help-search some search terms + +## DESCRIPTION + +This command will search the npm markdown documentation files for the +terms provided, and then list the results, sorted by relevance. + +If only one result is found, then it will show that help topic. + +If the argument to `npm help` is not a known help topic, then it will +call `help-search`. It is rarely if ever necessary to call this +command directly. + +## CONFIGURATION + +### long + +* Type: Boolean +* Default false + +If true, the "long" flag will cause help-search to output context around +where the terms were found in the documentation. + +If false, then help-search will just list out the help topics found. + +## SEE ALSO + +* npm(1) +* npm-faq(1) +* npm-help(1) diff --git a/deps/npm/doc/cli/help.md b/deps/npm/doc/cli/help.md new file mode 100644 index 0000000000..b51b0f1649 --- /dev/null +++ b/deps/npm/doc/cli/help.md @@ -0,0 +1,38 @@ +npm-help(1) -- Get help on npm +============================== + +## SYNOPSIS + + npm help + npm help some search terms + +## DESCRIPTION + +If supplied a topic, then show the appropriate documentation page. + +If the topic does not exist, or if multiple terms are provided, then run +the `help-search` command to find a match. Note that, if `help-search` +finds a single subject, then it will run `help` on that topic, so unique +matches are equivalent to specifying a topic name. + +## CONFIGURATION + +### viewer + +* Default: "man" on Posix, "browser" on Windows +* Type: path + +The program to use to view help content. + +Set to `"browser"` to view html help content in the default web browser. + +## SEE ALSO + +* npm(1) +* README +* npm-faq(1) +* npm-folders(1) +* npm-config(1) +* npm-json(1) +* npm-help-search(1) +* npm-index(1) diff --git a/deps/npm/doc/cli/home.md b/deps/npm/doc/cli/home.md new file mode 120000 index 0000000000..8828313f5b --- /dev/null +++ b/deps/npm/doc/cli/home.md @@ -0,0 +1 @@ +docs.md \ No newline at end of file diff --git a/deps/npm/doc/cli/init.md b/deps/npm/doc/cli/init.md new file mode 100644 index 0000000000..39297b4c4d --- /dev/null +++ b/deps/npm/doc/cli/init.md @@ -0,0 +1,24 @@ +npm-init(1) -- Interactively create a package.json file +======================================================= + +## SYNOPSIS + + npm init + +## DESCRIPTION + +This will ask you a bunch of questions, and then write a package.json for you. + +It attempts to make reasonable guesses about what you want things to be set to, +and then writes a package.json file with the options you've selected. + +If you already have a package.json file, it'll read that first, and default to +the options in there. + +It is strictly additive, so it does not delete options from your package.json +without a really good reason to do so. + +## SEE ALSO + +* npm-json(1) +* npm-version(1) diff --git a/deps/npm/doc/cli/install.md b/deps/npm/doc/cli/install.md new file mode 100644 index 0000000000..22eb8234e7 --- /dev/null +++ b/deps/npm/doc/cli/install.md @@ -0,0 +1,201 @@ +npm-install(1) -- Install a package +=================================== + +## SYNOPSIS + + npm install (with no args in a package dir) + npm install + npm install + npm install + npm install + npm install @ + npm install @ + npm install @ + +## DESCRIPTION + +This command installs a package, and any packages that it depends on. + +A `package` is: + +* a) a folder containing a program described by a package.json file +* b) a gzipped tarball containing (a) +* c) a url that resolves to (b) +* d) a `@` that is published on the registry with (c) +* e) a `@` that points to (d) +* f) a `` that has a "latest" tag satisfying (e) +* g) a `` that resolves to (b) + +Even if you never publish your package, you can still get a lot of +benefits of using npm if you just want to write a node program (a), and +perhaps if you also want to be able to easily install it elsewhere +after packing it up into a tarball (b). + + +* `npm install` (in package directory, no arguments): + Install the dependencies in the local node_modules folder. + + In global mode (ie, with `-g` or `--global` appended to the command), + it installs the current package context (ie, the current working + directory) as a global package. + +* `npm install `: + Install a package that is sitting in a folder on the filesystem. + +* `npm install `: + Install a package that is sitting on the filesystem. Note: if you just want + to link a dev directory into your npm root, you can do this more easily by + using `npm link`. + + Example: + + npm install ./package.tgz + +* `npm install `: + Fetch the tarball url, and then install it. In order to distinguish between + this and other options, the argument must start with "http://" or "https://" + + Example: + + npm install https://github.com/indexzero/forever/tarball/v0.5.6 + +* `npm install `: + Do a `@` install, where `` is the "tag" config. (See + `npm-config(1)`) + + Example: + + npm install sax + + **Note**: If there is a file or folder named `` in the current + working directory, then it will try to install that, and only try to + fetch the package by name if it is not valid. + +* `npm install @`: + Install the version of the package that is referenced by the specified tag. + If the tag does not exist in the registry data for that package, then this + will fail. + + Example: + + npm install sax@latest + +* `npm install @`: + Install the specified version of the package. This will fail if the version + has not been published to the registry. + + Example: + + npm install sax@0.1.1 + +* `npm install @`: + Install a version of the package matching the specified version range. This + will follow the same rules for resolving dependencies described in `npm-json(1)`. + + Note that most version ranges must be put in quotes so that your shell will + treat it as a single argument. + + Example: + + npm install sax@">=0.1.0 <0.2.0" + +* `npm install `: + + Install a package by cloning a git remote url. The format of the git + url is: + + ://[@][#] + + `` is one of `git`, `git+ssh`, `git+http`, or + `git+https`. If no `` is specified, then `master` is + used. + + Examples: + + git+ssh://git@github.com:isaacs/npm.git#v1.0.27 + git+https://isaacs@github.com/isaacs/npm.git + git://github.com/isaacs/npm.git#v1.0.27 + +You may combine multiple arguments, and even multiple types of arguments. +For example: + + npm install sax@">=0.1.0 <0.2.0" bench supervisor + +The `--tag` argument will apply to all of the specified install targets. + +The `--force` argument will force npm to fetch remote resources even if a +local copy exists on disk. + + npm install sax --force + +The `--global` argument will cause npm to install the package globally +rather than locally. See `npm-global(1)`. + +The `--link` argument will cause npm to link global installs into the +local space in some cases. + +See `npm-config(1)`. Many of the configuration params have some +effect on installation, since that's most of what npm does. + +## ALGORITHM + +To install a package, npm uses the following algorithm: + + install(where, what, family, ancestors) + fetch what, unpack to /node_modules/ + for each dep in what.dependencies + resolve dep to precise version + for each dep@version in what.dependencies + not in /node_modules//node_modules/* + and not in + add precise version deps to + install(/node_modules/, dep, family) + +For this `package{dep}` structure: `A{B,C}, B{C}, C{D}`, +this algorithm produces: + + A + +-- B + `-- C + `-- D + +That is, the dependency from B to C is satisfied by the fact that A +already caused C to be installed at a higher level. + +See npm-folders(1) for a more detailed description of the specific +folder structures that npm creates. + +### Limitations of npm's Install Algorithm + +There are some very rare and pathological edge-cases where a cycle can +cause npm to try to install a never-ending tree of packages. Here is +the simplest case: + + A -> B -> A' -> B' -> A -> B -> A' -> B' -> A -> ... + +where `A` is some version of a package, and `A'` is a different version +of the same package. Because `B` depends on a different version of `A` +than the one that is already in the tree, it must install a separate +copy. The same is true of `A'`, which must install `B'`. Because `B'` +depends on the original version of `A`, which has been overridden, the +cycle falls into infinite regress. + +To avoid this situation, npm flat-out refuses to install any +`name@version` that is already present anywhere in the tree of package +folder ancestors. A more correct, but more complex, solution would be +to symlink the existing version into the new location. If this ever +affects a real use-case, it will be investigated. + +## SEE ALSO + +* npm-folders(1) +* npm-update(1) +* npm-link(1) +* npm-rebuild(1) +* npm-scripts(1) +* npm-build(1) +* npm-config(1) +* npm-registry(1) +* npm-folders(1) +* npm-tag(1) +* npm-rm(1) diff --git a/deps/npm/doc/cli/json.md b/deps/npm/doc/cli/json.md new file mode 100644 index 0000000000..5f6e7ef621 --- /dev/null +++ b/deps/npm/doc/cli/json.md @@ -0,0 +1,472 @@ +npm-json(1) -- Specifics of npm's package.json handling +======================================================= + +## DESCRIPTION + +This document is all you need to know about what's required in your package.json +file. It must be actual JSON, not just a JavaScript object literal. + +A lot of the behavior described in this document is affected by the config +settings described in `npm-config(1)`. + +## DEFAULT VALUES + +npm will default some values based on package contents. + +* `"scripts": {"start": "node server.js"}` + + If there is a `server.js` file in the root of your package, then npm + will default the `start` command to `node server.js`. + +* `"scripts":{"preinstall": "node-waf clean || true; node-waf configure build"}` + + If there is a `wscript` file in the root of your package, npm will + default the `preinstall` command to compile using node-waf. + +* `"contributors": [...]` + + If there is an `AUTHORS` file in the root of your package, npm will + treat each line as a `Name (url)` format, where email and url + are optional. Lines which start with a `#` or are blank, will be + ignored. + +## name + +The *most* important things in your package.json are the name and version fields. +Those are actually required, and your package won't install without +them. The name and version together form an identifier that is assumed +to be completely unique. Changes to the package should come along with +changes to the version. + +The name is what your thing is called. Some tips: + +* Don't put "js" or "node" in the name. It's assumed that it's js, since you're + writing a package.json file, and you can specify the engine using the "engines" + field. (See below.) +* The name ends up being part of a URL, an argument on the command line, and a + folder name. Any name with non-url-safe characters will be rejected. + Also, it can't start with a dot or an underscore. +* The name will probably be passed as an argument to require(), so it should + be something short, but also reasonably descriptive. +* You may want to check the npm registry to see if there's something by that name + already, before you get too attached to it. http://registry.npmjs.org/ + +## version + +The *most* important things in your package.json are the name and version fields. +Those are actually required, and your package won't install without +them. The name and version together form an identifier that is assumed +to be completely unique. Changes to the package should come along with +changes to the version. + +Version must be parseable by +[node-semver](https://github.com/isaacs/node-semver), which is bundled +with npm as a dependency. (`npm install semver` to use it yourself.) + +Here's how npm's semver implementation deviates from what's on semver.org: + +* Versions can start with "v" +* A numeric item separated from the main three-number version by a hyphen + will be interpreted as a "build" number, and will *increase* the version. + But, if the tag is not a number separated by a hyphen, then it's treated + as a pre-release tag, and is *less than* the version without a tag. + So, `0.1.2-7 > 0.1.2-7-beta > 0.1.2-6 > 0.1.2 > 0.1.2beta` + +This is a little bit confusing to explain, but matches what you see in practice +when people create tags in git like "v1.2.3" and then do "git describe" to generate +a patch version. + +## description + +Put a description in it. It's a string. This helps people discover your +package, as it's listed in `npm search`. + +## keywords + +Put keywords in it. It's an array of strings. This helps people +discover your package as it's listed in `npm search`. + +## homepage + +The url to the project homepage. + +**NOTE**: This is *not* the same as "url". If you put a "url" field, +then the registry will think it's a redirection to your package that has +been published somewhere else, and spit at you. + +Literally. Spit. I'm so not kidding. + +## bugs + +The url to your project's issue tracker and / or the email address to which +issues should be reported. These are helpful for people who encounter issues +with your package. + +It should look like this: + + { "url" : "http://github.com/owner/project/issues" + , "email" : "project@hostname.com" + } + +You can specify either one or both values. If you want to provide only a url, +you can specify the value for "bugs" as a simple string instead of an object. + +If a url is provided, it will be used by the `npm bugs` command. + +## people fields: author, contributors + +The "author" is one person. "contributors" is an array of people. A "person" +is an object with a "name" field and optionally "url" and "email", like this: + + { "name" : "Barney Rubble" + , "email" : "b@rubble.com" + , "url" : "http://barnyrubble.tumblr.com/" + } + +Or you can shorten that all into a single string, and npm will parse it for you: + + "Barney Rubble (http://barnyrubble.tumblr.com/) + +Both email and url are optional either way. + +npm also sets a top-level "maintainers" field with your npm user info. + +## files + +The "files" field is an array of files to include in your project. If +you name a folder in the array, then it will also include the files +inside that folder. (Unless they would be ignored by another rule.) + +You can also provide a ".npmignore" file in the root of your package, +which will keep files from being included, even if they would be picked +up by the files array. The ".npmignore" file works just like a +".gitignore". + +## main + +The main field is a module ID that is the primary entry point to your program. +That is, if your package is named `foo`, and a user installs it, and then does +`require("foo")`, then your main module's exports object will be returned. + +This should be a module ID relative to the root of your package folder. + +For most modules, it makes the most sense to have a main script and often not +much else. + +## bin + +A lot of packages have one or more executable files that they'd like to +install into the PATH. npm makes this pretty easy (in fact, it uses this +feature to install the "npm" executable.) + +To use this, supply a `bin` field in your package.json which is a map of +command name to local file name. On install, npm will symlink that file into +`prefix/bin` for global installs, or `./node_modules/.bin/` for local +installs. + + +For example, npm has this: + + { "bin" : { "npm" : "./cli.js" } } + +So, when you install npm, it'll create a symlink from the `cli.js` script to +`/usr/local/bin/npm`. + +If you have a single executable, and its name should be the name +of the package, then you can just supply it as a string. For example: + + { "name": "my-program" + , "version": "1.2.5" + , "bin": "./path/to/program" } + +would be the same as this: + + { "name": "my-program" + , "version": "1.2.5" + , "bin" : { "my-program" : "./path/to/program" } } + +## man + +Specify either a single file or an array of filenames to put in place for the +`man` program to find. + +If only a single file is provided, then it's installed such that it is the +result from `man `, regardless of its actual filename. For example: + + { "name" : "foo" + , "version" : "1.2.3" + , "description" : "A packaged foo fooer for fooing foos" + , "main" : "foo.js" + , "man" : "./man/doc.1" + } + +would link the `./man/doc.1` file in such that it is the target for `man foo` + +If the filename doesn't start with the package name, then it's prefixed. +So, this: + + { "name" : "foo" + , "version" : "1.2.3" + , "description" : "A packaged foo fooer for fooing foos" + , "main" : "foo.js" + , "man" : [ "./man/foo.1", "./man/bar.1" ] + } + +will create files to do `man foo` and `man foo-bar`. + +Man files must end with a number, and optionally a `.gz` suffix if they are +compressed. The number dictates which man section the file is installed into. + + { "name" : "foo" + , "version" : "1.2.3" + , "description" : "A packaged foo fooer for fooing foos" + , "main" : "foo.js" + , "man" : [ "./man/foo.1", "./man/foo.2" ] + } + +will create entries for `man foo` and `man 2 foo` + +## directories + +The CommonJS [Packages](http://wiki.commonjs.org/wiki/Packages/1.0) spec details a +few ways that you can indicate the structure of your package using a `directories` +hash. If you look at [npm's package.json](http://registry.npmjs.org/npm/latest), +you'll see that it has directories for doc, lib, and man. + +In the future, this information may be used in other creative ways. + +### directories.lib + +Tell people where the bulk of your library is. Nothing special is done +with the lib folder in any way, but it's useful meta info. + +### directories.bin + +If you specify a "bin" directory, then all the files in that folder will +be used as the "bin" hash. + +If you have a "bin" hash already, then this has no effect. + +### directories.man + +A folder that is full of man pages. Sugar to generate a "man" array by +walking the folder. + +### directories.doc + +Put markdown files in here. Eventually, these will be displayed nicely, +maybe, someday. + +### directories.example + +Put example scripts in here. Someday, it might be exposed in some clever way. + +## repository + +Specify the place where your code lives. This is helpful for people who +want to contribute. If the git repo is on github, then the `npm docs` +command will be able to find you. + +Do it like this: + + "repository" : + { "type" : "git" + , "url" : "http://github.com/isaacs/npm.git" + } + + "repository" : + { "type" : "svn" + , "url" : "http://v8.googlecode.com/svn/trunk/" + } + +The URL should be a publicly available (perhaps read-only) url that can be handed +directly to a VCS program without any modification. It should not be a url to an +html project page that you put in your browser. It's for computers. + +## scripts + +The "scripts" member is an object hash of script commands that are run +at various times in the lifecycle of your package. The key is the lifecycle +event, and the value is the command to run at that point. + +See `npm-scripts(1)` to find out more about writing package scripts. + +## config + +A "config" hash can be used to set configuration +parameters used in package scripts that persist across upgrades. For +instance, if a package had the following: + + { "name" : "foo" + , "config" : { "port" : "8080" } } + +and then had a "start" command that then referenced the +`npm_package_config_port` environment variable, then the user could +override that by doing `npm config set foo:port 8001`. + +See `npm-config(1)` and `npm-scripts(1)` for more on package +configs. + +## dependencies + +Dependencies are specified with a simple hash of package name to version +range. The version range is EITHER a string which has one or more +space-separated descriptors, OR a range like "fromVersion - toVersion" + +**Please do not put test harnesses in your `dependencies` hash.** See +`devDependencies`, below. + +Version range descriptors may be any of the following styles, where "version" +is a semver compatible version identifier. + +* `version` Must match `version` exactly +* `=version` Same as just `version` +* `>version` Must be greater than `version` +* `>=version` etc +* `=version1 <=version2`. +* `range1 || range2` Passes if either range1 or range2 are satisfied. + +For example, these are all valid: + + { "dependencies" : + { "foo" : "1.0.0 - 2.9999.9999" + , "bar" : ">=1.0.2 <2.1.2" + , "baz" : ">1.0.2 <=2.3.4" + , "boo" : "2.0.1" + , "qux" : "<1.0.0 || >=2.3.1 <2.4.5 || >=2.5.2 <3.0.0" + , "asd" : "http://asdf.com/asdf.tar.gz" + , "til" : "~1.2" + , "elf" : "~1.2.3" + , "two" : "2.x" + , "thr" : "3.3.x" + } + } + +### Tilde Version Ranges + +A range specifier starting with a tilde `~` character is matched against +a version in the following fashion. + +* The version must be at least as high as the range. +* The version must be less than the next major revision above the range. + +For example, the following are equivalent: + +* `"~1.2.3" = ">=1.2.3 <1.3.0"` +* `"~1.2" = ">=1.2.0 <2.0.0"` +* `"~1" = ">=1.0.0 <2.0.0"` + +### X Version Ranges + +An "x" in a version range specifies that the version number must start +with the supplied digits, but any digit may be used in place of the x. + +The following are equivalent: + +* `"1.2.x" = ">=1.2.0 <1.3.0"` +* `"1.x.x" = ">=1.0.0 <2.0.0"` +* `"1.2" = "1.2.x"` +* `"1.x" = "1.x.x"` +* `"1" = "1.x.x"` + +You may not supply a comparator with a version containing an x. Any +digits after the first "x" are ignored. + +### URLs as Dependencies + +Starting with npm version 0.2.14, you may specify a tarball URL in place +of a version range. + +This tarball will be downloaded and installed locally to your package at +install time. + +## devDependencies + +If someone is planning on downloading and using your module in their +program, then they probably don't want or need to download and build +the external test or documentation framework that you use. + +In this case, it's best to list these additional items in a +`devDependencies` hash. + +These things will be installed whenever the `--dev` configuration flag +is set. This flag is set automatically when doing `npm link`, and can +be managed like any other npm configuration param. See `npm-config(1)` +for more on the topic. + +## bundledDependencies + +Array of package names that will be bundled when publishing the package. + +If this is spelled `"bundleDependencies"`, then that is also honorable. + +## engines + +You can specify the version of +node that your stuff works on: + + { "engines" : { "node" : ">=0.1.27 <0.1.30" } } + +And, like with dependencies, if you don't specify the version (or if you +specify "*" as the version), then any version of node will do. + +If you specify an "engines" field, then npm will require that "node" be +somewhere on that list. If "engines" is omitted, then npm will just assume +that it works on node. + +You can also use the "engines" field to specify which versions of npm +are capable of properly installing your program. For example: + + { "engines" : { "npm" : "~1.0.20" } } + +## preferGlobal + +If your package is primarily a command-line application that should be +installed globally, then set this value to `true` to provide a warning +if it is installed locally. + +It doesn't actually prevent users from installing it locally, but it +does help prevent some confusion if it doesn't work as expected. + +## private + +If you set `"private": true` in your package.json, then npm will refuse +to publish it. + +This is a way to prevent accidental publication of private repositories. +If you would like to ensure that a given package is only ever published +to a speciic registry (for example, an internal registry), +then use the `publishConfig` hash described below +to override the `registry` config param at publish-time. + +## publishConfig + +This is a set of config values that will be used at publish-time. It's +especially handy if you want to set the tag or registry, so that you can +ensure that a given package is not tagged with "latest" or published to +the global public registry by default. + +Any config values can be overridden, but of course only "tag" and +"registry" probably matter for the purposes of publishing. + +See `npm-config(1)` to see the list of config options that can be +overridden. + +## SEE ALSO + +* npm-semver(1) +* npm-init(1) +* npm-version(1) +* npm-config(1) +* npm-help(1) +* npm-faq(1) +* npm-install(1) +* npm-publish(1) +* npm-rm(1) diff --git a/deps/npm/doc/cli/link.md b/deps/npm/doc/cli/link.md new file mode 100644 index 0000000000..dd54792e29 --- /dev/null +++ b/deps/npm/doc/cli/link.md @@ -0,0 +1,57 @@ +npm-link(1) -- Symlink a package folder +======================================= + +## SYNOPSIS + + npm link (in package folder) + npm link + +## DESCRIPTION + +Package linking is a two-step process. + +First, `npm link` in a package folder will create a globally-installed +symbolic link from `prefix/package-name` to the current folder. + +Next, in some other location, `npm link package-name` will create a +symlink from the local `node_modules` folder to the global symlink. + +When creating tarballs for `npm publish`, the linked packages are +"snapshotted" to their current state by resolving the symbolic links. + +This is +handy for installing your own stuff, so that you can work on it and test it +iteratively without having to continually rebuild. + +For example: + + cd ~/projects/node-redis # go into the package directory + npm link # creates global link + cd ~/projects/node-bloggy # go into some other package directory. + npm link redis # link-install the package + +Now, any changes to ~/projects/node-redis will be reflected in +~/projects/node-bloggy/node_modules/redis/ + +You may also shortcut the two steps in one. For example, to do the +above use-case in a shorter way: + + cd ~/projects/node-bloggy # go into the dir of your main project + npm link ../node-redis # link the dir of your dependency + +The second line is the equivalent of doing: + + (cd ../node-redis; npm link) + npm link redis + +That is, it first creates a global link, and then links the global +installation target into your project's `node_modules` folder. + +## SEE ALSO + +* npm-developers(1) +* npm-faq(1) +* npm-json(1) +* npm-install(1) +* npm-folders(1) +* npm-config(1) diff --git a/deps/npm/doc/cli/list.md b/deps/npm/doc/cli/list.md new file mode 100644 index 0000000000..596349a815 --- /dev/null +++ b/deps/npm/doc/cli/list.md @@ -0,0 +1,55 @@ +npm-ls(1) -- List installed packages +====================================== + +## SYNOPSIS + + npm list + npm ls + npm la + npm ll + +## DESCRIPTION + +This command will print to stdout all the versions of packages that are +installed, as well as their dependencies, in a tree-structure. + +It does not take positional arguments, though you may set config flags +like with any other command, such as `-g` to list global packages. + +It will print out extraneous, missing, and invalid packages. + +When run as `ll` or `la`, it shows extended information by default. + +## CONFIGURATION + +### long + +* Default: false +* Type: Boolean + +Show extended information. + +### parseable + +* Default: false +* Type: Boolean + +Show parseable output instead of tree view. + +### global + +* Default: false +* Type: Boolean + +List packages in the global install prefix instead of in the current +project. + +## SEE ALSO + +* npm-config(1) +* npm-folders(1) +* npm-install(1) +* npm-link(1) +* npm-prune(1) +* npm-outdated(1) +* npm-update(1) diff --git a/deps/npm/doc/cli/ln.md b/deps/npm/doc/cli/ln.md new file mode 120000 index 0000000000..243f994145 --- /dev/null +++ b/deps/npm/doc/cli/ln.md @@ -0,0 +1 @@ +link.md \ No newline at end of file diff --git a/deps/npm/doc/cli/ls.md b/deps/npm/doc/cli/ls.md new file mode 120000 index 0000000000..eaad7acae2 --- /dev/null +++ b/deps/npm/doc/cli/ls.md @@ -0,0 +1 @@ +list.md \ No newline at end of file diff --git a/deps/npm/doc/cli/npm.md b/deps/npm/doc/cli/npm.md new file mode 100644 index 0000000000..cd3360d7c5 --- /dev/null +++ b/deps/npm/doc/cli/npm.md @@ -0,0 +1,155 @@ +npm(1) -- node package manager +============================== + +## SYNOPSIS + + npm [args] + +## VERSION + +@VERSION@ + +## DESCRIPTION + +npm is the package manager for the Node JavaScript platform. It puts +modules in place so that node can find them, and manages dependency +conflicts intelligently. + +It is extremely configurable to support a wide variety of use cases. +Most commonly, it is used to publish, discover, install, and develop node +programs. + +Run `npm help` to get a list of available commands. + +## INTRODUCTION + +You probably got npm because you want to install stuff. + +Use `npm install blerg` to install the latest version of "blerg". Check out +`npm-install(1)` for more info. It can do a lot of stuff. + +Use the `npm search` command to show everything that's available. +Use `npm ls` to show everything you've installed. + +## DIRECTORIES + +See `npm-folders(1)` to learn about where npm puts stuff. + +In particular, npm has two modes of operation: + +* global mode: + npm installs packages into the install prefix at + `prefix/lib/node_modules` and bins are installed in `prefix/bin`. +* local mode: + npm installs packages into the current project directory, which + defaults to the current working directory. Packages are installed to + `./node_modules`, and bins are installed to `./node_modules/.bin`. + +Local mode is the default. Use `--global` or `-g` on any command to +operate in global mode instead. + +## DEVELOPER USAGE + +If you're using npm to develop and publish your code, check out the +following help topics: + +* json: + Make a package.json file. See `npm-json(1)`. +* link: + For linking your current working code into Node's path, so that you + don't have to reinstall every time you make a change. Use + `npm link` to do this. +* install: + It's a good idea to install things if you don't need the symbolic link. + Especially, installing other peoples code from the registry is done via + `npm install` +* adduser: + Create an account or log in. Creditials are stored in the + user config file. +* publish: + Use the `npm publish` command to upload your code to the registry. + +## CONFIGURATION + +npm is extremely configurable. It reads its configuration options from +5 places. + +* Command line switches: + Set a config with `--key val`. All keys take a value, even if they + are booleans (the config parser doesn't know what the options are at + the time of parsing.) If no value is provided, then the option is set + to boolean `true`. +* Environment Variables: + Set any config by prefixing the name in an environment variable with + `npm_config_`. For example, `export npm_config_key=val`. +* User Configs: + The file at $HOME/.npmrc is an ini-formatted list of configs. If + present, it is parsed. If the `userconfig` option is set in the cli + or env, then that will be used instead. +* Global Configs: + The file found at ../etc/npmrc (from the node executable, by default + this resolves to /usr/local/etc/npmrc) will be parsed if it is found. + If the `globalconfig` option is set in the cli, env, or user config, + then that file is parsed instead. +* Defaults: + npm's default configuration options are defined in + lib/utils/config-defs.js. These must not be changed. + +See `npm-config(1)` for much much more information. + +## CONTRIBUTIONS + +Patches welcome! + +* code: + Read through `npm-coding-style(1)` if you plan to submit code. + You don't have to agree with it, but you do have to follow it. +* docs: + If you find an error in the documentation, edit the appropriate markdown + file in the "doc" folder. (Don't worry about generating the man page.) + +Contributors are listed in npm's `package.json` file. You can view them +easily by doing `npm view npm contributors`. + +If you would like to contribute, but don't know what to work on, check +the issues list or ask on the mailing list. + +* +* + +## BUGS + +When you find issues, please report them: + +* web: + +* email: + + +Be sure to include *all* of the output from the npm command that didn't work +as expected. The `npm-debug.log` file is also helpful to provide. + +You can also look for isaacs in #node.js on irc://irc.freenode.net. He +will no doubt tell you to put the output in a gist or email. + +## HISTORY + +See npm-changelog(1) + +## AUTHOR + +[Isaac Z. Schlueter](http://blog.izs.me/) :: +[isaacs](https://github.com/isaacs/) :: +[@izs](http://twitter.com/izs) :: + + +## SEE ALSO + +* npm-help(1) +* npm-faq(1) +* README +* npm-json(1) +* npm-install(1) +* npm-config(1) +* npm-index(1) +* npm(3) diff --git a/deps/npm/doc/cli/outdated.md b/deps/npm/doc/cli/outdated.md new file mode 100644 index 0000000000..78df4a8b30 --- /dev/null +++ b/deps/npm/doc/cli/outdated.md @@ -0,0 +1,17 @@ +npm-outdated(1) -- Check for outdated packages +============================================== + +## SYNOPSIS + + npm outdated [ [ ...]] + +## DESCRIPTION + +This command will check the registry to see if any (or, specific) installed +packages are currently outdated. + +## SEE ALSO + +* npm-update(1) +* npm-registry(1) +* npm-folders(1) diff --git a/deps/npm/doc/cli/owner.md b/deps/npm/doc/cli/owner.md new file mode 100644 index 0000000000..8365da379e --- /dev/null +++ b/deps/npm/doc/cli/owner.md @@ -0,0 +1,32 @@ +npm-owner(1) -- Manage package owners +===================================== + +## SYNOPSIS + + npm owner ls + npm owner add + npm owner rm + +## DESCRIPTION + +Manage ownership of published packages. + +* ls: + List all the users who have access to modify a package and push new versions. + Handy when you need to know who to bug for help. +* add: + Add a new user as a maintainer of a package. This user is enabled to modify + metadata, publish new versions, and add other owners. +* rm: + Remove a user from the package owner list. This immediately revokes their + privileges. + +Note that there is only one level of access. Either you can modify a package, +or you can't. Future versions may contain more fine-grained access levels, but +that is not implemented at this time. + +## SEE ALSO + +* npm-publish(1) +* npm-registry(1) +* npm-adduser(1) diff --git a/deps/npm/doc/cli/pack.md b/deps/npm/doc/cli/pack.md new file mode 100644 index 0000000000..98d8c81c50 --- /dev/null +++ b/deps/npm/doc/cli/pack.md @@ -0,0 +1,25 @@ +npm-pack(1) -- Create a tarball from a package +============================================== + +## SYNOPSIS + + npm pack [ [ ...]] + +## DESCRIPTION + +For anything that's installable (that is, a package folder, tarball, +tarball url, name@tag, name@version, or name), this command will fetch +it to the cache, and then copy the tarball to the current working +directory as `-.tgz`, and then write the filenames out to +stdout. + +If the same package is specified multiple times, then the file will be +overwritten the second time. + +If no arguments are supplied, then npm packs the current package folder. + +## SEE ALSO + +* npm-cache(1) +* npm-publish(1) +* npm-config(1) diff --git a/deps/npm/doc/cli/prefix.md b/deps/npm/doc/cli/prefix.md new file mode 100644 index 0000000000..f6247cab12 --- /dev/null +++ b/deps/npm/doc/cli/prefix.md @@ -0,0 +1,17 @@ +npm-prefix(1) -- Display prefix +=============================== + +## SYNOPSIS + + npm prefix + +## DESCRIPTION + +Print the prefix to standard out. + +## SEE ALSO + +* npm-root(1) +* npm-bin(1) +* npm-folders(1) +* npm-config(1) diff --git a/deps/npm/doc/cli/prune.md b/deps/npm/doc/cli/prune.md new file mode 100644 index 0000000000..8c4b957e6c --- /dev/null +++ b/deps/npm/doc/cli/prune.md @@ -0,0 +1,21 @@ +npm-prune(1) -- Remove extraneous packages +========================================== + +## SYNOPSIS + + npm prune [ [ + npm publish + +## DESCRIPTION + +Publishes a package to the registry so that it can be installed by name. + +* ``: + A folder containing a package.json file + +* ``: + A url or file path to a gzipped tar archive containing a single folder + with a package.json file inside. + +Fails if the package name and version combination already exists in +the registry. Overwrites when the "--force" flag is set. + +## SEE ALSO + +* npm-registry(1) +* npm-adduser(1) +* npm-owner(1) +* npm-deprecate(1) +* npm-tag(1) diff --git a/deps/npm/doc/cli/rebuild.md b/deps/npm/doc/cli/rebuild.md new file mode 100644 index 0000000000..6985a7bdd7 --- /dev/null +++ b/deps/npm/doc/cli/rebuild.md @@ -0,0 +1,20 @@ +npm-rebuild(1) -- Rebuild a package +=================================== + +## SYNOPSIS + + npm rebuild [ [ ...]] + +* ``: + The package to rebuild + +## DESCRIPTION + +This command runs the `npm build` command on the matched folders. This is useful +when you install a new version of node, and must recompile all your C++ addons with +the new binary. + +## SEE ALSO + +* npm-build(1) +* npm-install(1) diff --git a/deps/npm/doc/cli/registry.md b/deps/npm/doc/cli/registry.md new file mode 100644 index 0000000000..8073ea0820 --- /dev/null +++ b/deps/npm/doc/cli/registry.md @@ -0,0 +1,92 @@ +npm-registry(1) -- The JavaScript Package Registry +================================================== + +## DESCRIPTION + +To resolve packages by name and version, npm talks to a registry website +that implements the CommonJS Package Registry specification for reading +package info. + +Additionally, npm's package registry implementation supports several +write APIs as well, to allow for publishing packages and managing user +account information. + +The official public npm registry is at . It +is powered by a CouchDB database at +. The code for the couchapp is +available at . npm user accounts +are CouchDB users, stored in the +database. + +The registry URL is supplied by the `registry` config parameter. See +`npm-config(1)` for more on managing npm's configuration. + +## Can I run my own private registry? + +Yes! + +The easiest way is to replicate the couch database, and use the same (or +similar) design doc to implement the APIs. + +If you set up continuous replication from the official CouchDB, and then +set your internal CouchDB as the registry config, then you'll be able +to read any published packages, in addition to your private ones, and by +default will only publish internally. If you then want to publish a +package for the whole world to see, you can simply override the +`--registry` config for that command. + +## I don't want my package published in the official registry. It's private. + +Set `"private": true` in your package.json to prevent it from being +published at all, or +`"publishConfig":{"registry":"http://my-internal-registry.local"}` +to force it to be published only to your internal registry. + +See `npm-json(1)` for more info on what goes in the package.json file. + +## Will you replicate from my registry into the public one? + +No. If you want things to be public, then publish them into the public +registry using npm. What little security there is would be for nought +otherwise. + +## Do I have to use couchdb to build a registry that npm can talk to? + +No, but it's way easier. + +## I published something elsewhere, and want to tell the npm registry about it. + +That is supported, but not using the npm client. You'll have to get +your hands dirty and do some HTTP. The request looks something like +this: + + PUT /my-foreign-package + content-type:application/json + accept:application/json + authorization:Basic $base_64_encoded + + { "name":"my-foreign-package" + , "maintainers":["owner","usernames"] + , "description":"A package that is hosted elsewhere" + , "keywords":["nih","my cheese smells the best"] + , "url":"http://my-different-registry.com/blerg/my-local-package" + } + +(Keywords and description are optional, but recommended. Name, +maintainers, and url are required.) + +Then, when a user tries to install "my-foreign-package", it'll redirect +to your registry. If that doesn't resolve to a valid package entry, +then it'll fail, so please make sure that you understand the spec, and +ask for help on the mailing list. + +## Is there a website or something to see package docs and such? + +No, but such a thing is planned, and a tiny bit developed. + +Stay tuned! + +## SEE ALSO + +* npm-config(1) +* npm-developers(1) diff --git a/deps/npm/doc/cli/removing-npm.md b/deps/npm/doc/cli/removing-npm.md new file mode 100644 index 0000000000..bedd28a2fa --- /dev/null +++ b/deps/npm/doc/cli/removing-npm.md @@ -0,0 +1,54 @@ +npm-removal(1) -- Cleaning the Slate +==================================== + +## SYNOPSIS + +So sad to see you go. + + sudo npm uninstall npm -g + +Or, if that fails, get the npm source code, and do: + + sudo make uninstall + +## More Severe Uninstalling + +Usually, the above instructions are sufficient. That will remove +npm, but leave behind anything you've installed. + +If that doesn't work, or if you require more drastic measures, +continue reading. + +Note that this is only necessary for globally-installed packages. Local +installs are completely contained within a project's `node_modules` +folder. Delete that folder, and everything is gone (unless a package's +install script is particularly ill-behaved). + +This assumes that you installed node and npm in the default place. If +you configured node with a different `--prefix`, or installed npm with a +different prefix setting, then adjust the paths accordingly, replacing +`/usr/local` with your install prefix. + +To remove everything npm-related manually: + + rm -rf /usr/local/{lib/node{,/.npm,_modules},bin,share/man}/npm* + +If you installed things *with* npm, then your best bet is to uninstall +them with npm first, and then install them again once you have a +proper install. This can help find any symlinks that are lying +around: + + ls -laF /usr/local/{lib/node{,/.npm},bin,share/man} | grep npm + +Prior to version 0.3, npm used shim files for executables and node +modules. To track those down, you can do the following: + + find /usr/local/{lib/node,bin} -exec grep -l npm \{\} \; ; + +(This is also in the README file.) + +## SEE ALSO + +* README +* npm-rm(1) +* npm-prune(1) diff --git a/deps/npm/doc/cli/restart.md b/deps/npm/doc/cli/restart.md new file mode 100644 index 0000000000..6139dbeefb --- /dev/null +++ b/deps/npm/doc/cli/restart.md @@ -0,0 +1,22 @@ +npm-restart(1) -- Start a package +================================= + +## SYNOPSIS + + npm restart + +## DESCRIPTION + +This runs a package's "restart" script, if one was provided. +Otherwise it runs package's "stop" script, if one was provided, and then +the "start" script. + +If no version is specified, then it restarts the "active" version. + +## SEE ALSO + +* npm-run-script(1) +* npm-scripts(1) +* npm-test(1) +* npm-start(1) +* npm-stop(1) diff --git a/deps/npm/doc/cli/rm.md b/deps/npm/doc/cli/rm.md new file mode 120000 index 0000000000..32d3b511f9 --- /dev/null +++ b/deps/npm/doc/cli/rm.md @@ -0,0 +1 @@ +uninstall.md \ No newline at end of file diff --git a/deps/npm/doc/cli/root.md b/deps/npm/doc/cli/root.md new file mode 100644 index 0000000000..3e4199541e --- /dev/null +++ b/deps/npm/doc/cli/root.md @@ -0,0 +1,17 @@ +npm-root(1) -- Display npm root +=============================== + +## SYNOPSIS + + npm root + +## DESCRIPTION + +Print the effective `node_modules` folder to standard out. + +## SEE ALSO + +* npm-prefix(1) +* npm-bin(1) +* npm-folders(1) +* npm-config(1) diff --git a/deps/npm/doc/cli/run-script.md b/deps/npm/doc/cli/run-script.md new file mode 100644 index 0000000000..41ef5e7872 --- /dev/null +++ b/deps/npm/doc/cli/run-script.md @@ -0,0 +1,21 @@ +npm-run-script(1) -- Run arbitrary package scripts +================================================== + +## SYNOPSIS + + npm run-script + diff --git a/deps/npm/html/dochead.html b/deps/npm/html/dochead.html new file mode 100644 index 0000000000..c96b0d733f --- /dev/null +++ b/deps/npm/html/dochead.html @@ -0,0 +1,8 @@ + + + @NAME@ + + + + +
diff --git a/deps/npm/html/favicon.ico b/deps/npm/html/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..9e0d4eef78c9c01027fb8a88af38c17628e34889 GIT binary patch literal 7094 zcmeHLI}XAy40Q@fj1>bj6Y33M%n3LL$KVKv%fQIaDVRA#u@eWiRY0^ARe)zH@h@)N zm$Z^iWGEwR$x$&UTC$YqUf)iLHf$?cwLEq_3{81vnt9fo%d(`^`a*~Ol}^W7(+wR6 z00AHX1b_e#00JLIAgfXP&1&IU>hbk#RWqJOQRf@RVdHf-!-ng@@UuiiKmZ5; z0U#gXT)GJTKs4hWlUmH+?% literal 0 HcmV?d00001 diff --git a/deps/npm/html/index.html b/deps/npm/html/index.html new file mode 100644 index 0000000000..9ccf28e57c --- /dev/null +++ b/deps/npm/html/index.html @@ -0,0 +1,86 @@ + + + + + + npm - Node Package Manager + +

npm

+ +

npm is a package manager for node. You can use it to install + and publish your node programs. It manages dependencies and does other cool stuff.

+ +

One Line Install

+ +curl http://npmjs.org/install.sh | sh + +

More Than One Line Install

+ +
    +
  1. Get the code. +
  2. Do what the README + says to do. +
+ +

Other Cool Stuff

+ + + + + diff --git a/deps/npm/html/n-64.png b/deps/npm/html/n-64.png new file mode 100644 index 0000000000000000000000000000000000000000..d4145efae63e6dce185fff60faf42c903d46430b GIT binary patch literal 679 zcmeAS@N?(olHy`uVBq!ia0vp^4Is?O3?zSk_}l@cSkfJR9T^xl_H+M9WCijWi-X*q z7}lMWc?skwBzpw;GB8xB0X4NSF#G}vHN0S8C^cYUc$L7wU^Rn*LA+qju0R{01Y3Yl zi0f$!^Z)<cps|0B<&^0vDH82Y?G_o=> zumZ6yfEt!Gw#)=-Fap^C3^psU&yzDs^RgAn6N?qn@=Nnl6f}xcQx%*uit=+4OEQx) zODc5~5|i@FQZ?BxdfWo))`aN>2Wom^Zf>Hjfu4z}iK(fAMlevXJ4gtoNk~X%IZ%@b zOcN-I^pY}5;!{!!N-`KMG+XNefbya!^2zx*`9<+1l?AB`W^7)|wSfwxQ50n6m82Hs zBqpcE=cblq#kNKk#-=(7l}PN;yv&kfTf-*JB489D^ngmdKI;Vst02iUamH+?% literal 0 HcmV?d00001 diff --git a/deps/npm/html/n-large.png b/deps/npm/html/n-large.png new file mode 100644 index 0000000000000000000000000000000000000000..9e1525f05aae2a5658a70842ca99ca039a0834bc GIT binary patch literal 699 zcmeAS@N?(olHy`uVBq!ia0y~yU<5K585o&?RN5XZRUpNZ?&#~tz_78O`%fY(NVYi0 z-HBn{IhmJ0jzY3WkS_y6l^O#>Lkk1LFQ8Dv3kHT#0|tgy2@DKYGZ+}e3+C(!v;j)6 z1^9%xp0+Um|Np;Iw9{9h2xF4By9?ugCf$2L4rhT!WHAE+w=f7ZGR&GI0Ti_Gba4#H zxcBzdMotC?jsp(gNoI0pNu`cLVp4utswVqIk6S?9nlRnqfKE@$%}um5&@(YL zF*Q}t2nOnP2MNJ62?^;e2Wk?5X#z!?UQ%XBd`fCTNd|+3W@}vlP+k;8J~=-pzbL+> zvLKbgjLmDgHc){yih|6%lGLJ{#N^cY+|-hc{1gU*-`4CsKxHy0%E~iSN-}gZQZv&t zN*D~ybQBEDEUvQg0@Z22)PZBMBEBfK*w)Cx*i=WM5{X@!mswJ5YuKb&WQVK=oT5NZ n$SbimfcQBmH7_N#C^Iiz0V1yXNl*loWEebM{an^LB{Ts5qI1dt literal 0 HcmV?d00001 diff --git a/deps/npm/html/npm-16.png b/deps/npm/html/npm-16.png new file mode 100644 index 0000000000000000000000000000000000000000..c3c9d05253f46deded5235e9276a3e9c21b1227f GIT binary patch literal 145 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!61|;P_|4#%`jKx9jP7LeL$-D$|I14-?iy0WW zg+Z8+Vb&Z8prDzji(`nz>13PS#DDV}nGPIrP+V|>K`f>3_Bh)cGcpwH;F`MpIFAU>1O`u6KbLh*2~7ZBUL|+{ literal 0 HcmV?d00001 diff --git a/deps/npm/html/npm-256-square.png b/deps/npm/html/npm-256-square.png new file mode 100644 index 0000000000000000000000000000000000000000..f7f18b560eebe76c23162dd2ac9fdd7fae13851b GIT binary patch literal 3290 zcmb`Jc{J4T9>>4dX6)ImdVao z8bw0cO|pz7#@@~Ech0@{+;jiC_c`bNJbygz^ZvZf`}yZdgqs?3vWc((0KjPiy^Q#C zt{n=*`ezTh{kRwa*bF=k4B#dP22%b3e(s*gTL2I`l5Y`Wn@$(h9;e^pw$>kBd6kWf z24rLjpiUmUBA!Qp%RL@m-L|*{Hfw%?qj^xP)|f#yXnw*};R&LL-h+heNme&BmLzE?ABBBsC#|res0WBUiq}SA}`qwaga*(w^L$Vh4i z3JJP0rdq&s>Y0%1{NSV#TLu0>kMFU`ieb(lm;VUAi>wmBn4wCJTuTVQRd_y zM}#r}BywhN_SJ#2{KU+UA|Y!l6I0St&B#~!E|8Nv?%)}jyzE?AJ}8w5*2e(=BHNS? z2O2t9>fFraoI4ds`_)!-;JxA}qbv9mA9xjt<;Go7UU`|PCOL{jADpWB1c zuod7=jxMf(gpb|7l|;l3Hk^zD&1WF#qanSn8h%XT0b(2ZPIdMLu42Jm>O zsiNrsklwVNs;j^N^kx@N13>)`p&O+c5Li1104^6spL(fxgyRE0fy9;fLA;&>VRT`= ztjlqXbcsiwbp|8G_EacKPuP;AIn4a#C6h?u(VR)1nhz=(oYV>48_^5?5LRcQf@Y!m z`(S3hP7t4qv}VFl*OXzIJ`e<#!YtzgReOXfmYPrCH?YpOQK(X}MX~k(k7piCegW&#^?e5P8bgrEPOHUo zW6d7zcD2mZ@@uvwOu@iK#}LICyjRq3>C7@ed!nk3D|}(zTUPKH^ddp?$#oN-!eF?v zMSK~bT2PK$Z?2#LxmXE5_%X-R!;^knc^*{<{uV1?nEf$(!28FMdFp)ZuMiH-`<$^I zP}dW}-okf|gl}}(kJ)kK ze0@L8sPH}%Z|PRsE}Sea*sgkO)mDEfz?x}6(fjc=SNqr7i?CppUD06WyD$rbz!&ZW z_cZtXxjVfKTD7=|g?mM0#rcu2k;W0%rGO>DrG*h% zkbk~zzGD7{rjI5)2wZor&b%(JZZxR>r}ihe1@t_4qkDsE{u{-3!{--%1M-tLM59Qa zKcfaZhZRPuXt=HPVBLOrKlUV2R_;xf9wpzdOKZK2rT)KQUK= z+V@*y-j`flJyl1GTZ`Lf8c+6Sc;~p+QEEP+dU`q)#|XoOuEmrKgL&QsE{$L7NB5HB zo*Cw~T}e8Z)R8rk6;5W!x|T&JJhs-hMq6)Ke<7F*-B?gu@ENi}!bNRzIdU~~$!|+5 zaw}ScbA877r}$G(4xHSu@EuWE+*@Q@%voezy0Gc8DYxmf*)QHB#w$K_5p^*<02R~| zjP$QxD4tYnDQhijnYZx}^>`YXzpf5>eicjR%?{3nSooAYDDiy~M$ViSS*{y+^JxR0 z`#yK9;;YZ+Z#`OV5#_trPUnhtIkafe^nR_-ZKv+^Oy`&6mNc(psmhMQj&_c_4H9P6 zdGZBAXM8$GTkR(r$akGJwY6^T^vyGAw`(ŔRX_`FeOT&qzwawrlPd1S9(_u>wD zKXt2V61^t4YRclw@PT2Lp&?o%x|+$GNs<}I{OpK2OESxQR%H%t0eucIr=4U#x*rL$ zCc5o*cFaRDha%7o$``mUx^t#S9wsq#H06FjXq+d~@A)=tzUM6{VZs=hK$P?p#zD3t z_)nfveR2Ayd5s0?4vMZgM3}ML^=L?0O)Y!)?&7G|+kjvXf)^5jbX&|NVu&w@(V_8M zhO61UD86IhrNo)m;a080=|Dy*GUsgGOpaw1%B^yCtmqwNZoZFP5OT>*-3}YB5%-uk z`?bz&tdu@nS;%a*#P^yw-`c(0I5We>qU0j;SGB||mmbaKTW4-+e7hK7uygyqnNd*t zS+E&G0i7oN=$-8PHS5bm-wFL+?|fCdkdc>>hevRFX4NQsIa7P0*34Z&A)RvuJ?GOc znjYUxpOM`TTptxc#=$7~8bl?WOAw}9st{@~WFJa|HbAd`K0EGy@N-+1Jw`CR0s9h8 zd|5qA47FRGkZCe}FDRR20SP&+;((~T8S1$*5JA38ex0=spRz?*t$Crx)>06$1J9lr zD8$&PWtL@hk$o$FU4Q2$;F5XMisqF)Xm1^L3XBURr=I6fgJ^=ypw2R~o$Z20z|k;j9o{O`a8X^11RI^)$?DBc->dx4V}~-9>%L zzUM7rbnWRm{AO5wlmpp%zoNXh@vHpEP}8@NyX#!B;sO`H(=rz8-zbFIcsCeEZ1@f= z$9zi_aTTp=nv48KG!8LqD@eWj?Cz0GL)wBD$`3hK)ztJF?1MO$ECJQ@pUuVijpQ!n zguL;;eu}GEZ}!9S@#o=pr{O=I_O*<)^p4}AYBxX&P_)8;W>XMB3GY*=^|XQZ;~_p@ z+)>=PvG#W)O=a3Gs6#!~rl2tYgx2czOicQ8$<302d{Z5*&$ux(9kz$_b})Bn@G$5v z(1D5uJ!NKREL#%|6ek6*HO9Js})+6TE>z7`;*Wrd%uNSbn=qTrApMVYYP`C zv`kBIrq2NanrcK@dXDNzdyxK4jA_s56T+JCa300N{VudG_2k1OP?%x zUBcJj9cld=y%1PV9p9hYz1nZPH@w2Woz&k{i7s0Ypp}<_gI1ORrvLoon3 zc|ridsAY0l&oXpmX;kG(3tWseY11_$t@%SBW2ijgy+($F-X)m%Dq|h$&9j&ZsnZXn zT(SI?mp^g^p-N*DAJwf_AvDj@NdX}8CJ}W8Nnr&B9&`)wPHzp|AGHcARYLR zlEm_*`~ktO$S8=blbhW@3w}jS%;>%L?i8JhyH+wAw-*VP~rDo_82! MVrY7~Qr|h|ANy+UtN;K2 literal 0 HcmV?d00001 diff --git a/deps/npm/html/npm-256w.png b/deps/npm/html/npm-256w.png new file mode 100644 index 0000000000000000000000000000000000000000..dac32c8a02c3bf77aa3166914b84bd179ef243af GIT binary patch literal 3169 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 z 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%0D`po*cm$Hpv3s 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 +8wDKN_nR-36 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:"*"} + // asyncMap over deps: + // shouldHave = cache.add(dep, req).version + // if has === shouldHave then + // return outdated(args, dir/node_modules/dep, parentHas + has) + // else if dep in args or args is empty + // return [dir, dep, has, shouldHave] + + var deps = null + readJson(path.resolve(dir, "package.json"), function (er, d) { + deps = (er) ? true : d.dependencies + return next() + }) + + var has = null + fs.readdir(path.resolve(dir, "node_modules"), function (er, pkgs) { + if (er) { + has = Object.create(parentHas) + return next() + } + asyncMap(pkgs, function (pkg, cb) { + readJson( path.resolve(dir, "node_modules", pkg, "package.json") + , function (er, d) { + cb(null, er ? [] : [[d.name, d.version]]) + }) + }, function (er, pvs) { + if (er) return cb(er) + has = Object.create(parentHas) + pvs.forEach(function (pv) { + has[pv[0]] = pv[1] + }) + next() + }) + }) + + function next () { + if (!has || !deps) return + if (deps === true) { + deps = Object.keys(has).reduce(function (l, r) { + l[r] = "*" + return l + }, {}) + } + // now get what we should have, based on the dep. + // if has[dep] !== shouldHave[dep], then cb with the data + // otherwise dive into the folder + asyncMap(Object.keys(deps), function (dep, cb) { + shouldUpdate(args, dir, dep, has, deps[dep], cb) + }, cb) + } +} + +function shouldUpdate (args, dir, dep, has, req, cb) { + // look up the most recent version. + // if that's what we already have, or if it's not on the args list, + // then dive into it. Otherwise, cb() with the data. + + function skip () { + outdated_( args + , path.resolve(dir, "node_modules", dep) + , has + , cb ) + } + + function doIt (shouldHave) { + cb(null, [[ dir, dep, has[dep], shouldHave ]]) + } + + if (args.length && args.indexOf(dep) === -1) { + return skip() + } + + // so, we can conceivably update this. find out if we need to. + cache.add(dep, req, function (er, d) { + // if this fails, then it means we can't update this thing. + // it's probably a thing that isn't published. + return (er || d.version === has[dep]) ? skip() : doIt(d.version) + }) +} diff --git a/deps/npm/lib/owner.js b/deps/npm/lib/owner.js new file mode 100644 index 0000000000..6025e58a75 --- /dev/null +++ b/deps/npm/lib/owner.js @@ -0,0 +1,188 @@ + +module.exports = owner + +owner.usage = "npm owner add " + + "\nnpm owner rm " + + "\nnpm owner ls " + +owner.completion = function (opts, cb) { + var argv = opts.conf.argv.remain + if (argv.length > 4) return cb() + if (argv.length <= 2) { + var subs = ["add", "rm"] + if (opts.partialWord === "l") subs.push("ls") + else subs.push("ls", "list") + return cb(null, subs) + } + var un = encodeURIComponent(npm.config.get("username")) + switch (argv[2]) { + case "ls": + if (argv.length > 3) return cb() + else return registry.get("/-/short", cb) + + case "rm": + if (argv.length > 3) { + var theUser = encodeURIComponent(argv[3]) + , uri = "/-/by-user/"+theUser+"|"+un + console.error(uri) + return registry.get(uri, function (er, d) { + if (er) return cb(er) + // return the intersection + return cb(null, d[theUser].filter(function (p) { + // kludge for server adminery. + return un === "isaacs" || d[un].indexOf(p) === -1 + })) + }) + } + // else fallthrough + case "add": + if (argv.length > 3) { + var theUser = encodeURIComponent(argv[3]) + , uri = "/-/by-user/"+theUser+"|"+un + console.error(uri) + return registry.get(uri, function (er, d) { + console.error(uri, er || d) + // return mine that they're not already on. + if (er) return cb(er) + var mine = d[un] || [] + , theirs = d[theUser] || [] + return cb(null, mine.filter(function (p) { + return theirs.indexOf(p) === -1 + })) + }) + } + // just list all users who aren't me. + return registry.get("/-/users", function (er, list) { + if (er) return cb() + return cb(null, Object.keys(list).filter(function (n) { + return n !== un + })) + }) + + default: + return cb() + } +} + +var registry = require("./utils/npm-registry-client/index.js") + , get = registry.request.GET + , put = registry.request.PUT + , log = require("./utils/log.js") + , output + , npm = require("./npm.js") + +function owner (args, cb) { + var action = args.shift() + switch (action) { + case "ls": case "list": return ls(args[0], cb) + case "add": return add(args[0], args[1], cb) + case "rm": case "remove": return rm(args[0], args[1], cb) + default: return unknown(action, cb) + } +} + +function ls (pkg, cb) { + if (!pkg) return cb(owner.usage) + get(pkg, function (er, data) { + var msg = "" + if (er) return log.er(cb, "Couldn't get owner data for "+pkg)(er) + var owners = data.maintainers + if (!owners || !owners.length) msg = "admin party!" + else msg = owners.map(function (o) { return o.name +" <"+o.email+">" }).join("\n") + output = output || require("./utils/output.js") + output.write(msg, function (er) { cb(er, owners) }) + }) +} + +function add (user, pkg, cb) { + if (!pkg) readLocalPkg(function (er, pkg) { + if (er) return cb(er) + if (!pkg) return cb(new Error(owner.usage)) + add(user, pkg, cb) + }) + + log.verbose(user+" to "+pkg, "owner add") + mutate(pkg, user, function (u, owners) { + if (!owners) owners = [] + for (var i = 0, l = owners.length; i < l; i ++) { + var o = owners[i] + if (o.name === u.name) { + log( "Already a package owner: "+o.name+" <"+o.email+">" + , "owner add" + ) + return false + } + } + owners.push(u) + return owners + }, cb) +} + +function rm (user, pkg, cb) { + if (!pkg) readLocalPkg(function (er, pkg) { + if (er) return cb(er) + if (!pkg) return cb(new Error(owner.usage)) + rm(user, pkg, cb) + }) + + log.verbose(user+" from "+pkg, "owner rm") + mutate(pkg, null, function (u, owners) { + var found = false + , m = owners.filter(function (o) { + var match = (o.name === user) + found = found || match + return !match + }) + if (!found) { + log("Not a package owner: "+user, "owner rm") + return false + } + if (!m.length) return new Error( + "Cannot remove all owners of a package. Add someone else first.") + return m + }, cb) +} + +function mutate (pkg, user, mutation, cb) { + if (user) { + get("/-/user/org.couchdb.user:"+user, mutate_) + } else { + mutate_(null, null) + } + + function mutate_ (er, u) { + if (er) return log.er(cb, "Error getting user data for "+user)(er) + if (user && (!u || u.error)) return cb(new Error( + "Couldn't get user data for "+user+": "+JSON.stringify(u))) + if (u) u = { "name" : u.name, "email" : u.email } + get("/"+pkg, function (er, data) { + if (er) return log.er(cb, "Couldn't get package data for "+pkg)(er) + var m = mutation(u, data.maintainers) + if (!m) return cb() // handled + if (m instanceof Error) return cb(m) // error + data = { _id : data._id + , _rev : data._rev + , maintainers : m + } + put("/"+pkg+"/-rev/"+data._rev, data, function (er, data) { + if (er) return log.er(cb, "Failed to update package metadata")(er) + if (data.error) return cb(new Error( + "Failed to update pacakge metadata: "+JSON.stringify(data))) + cb(null, data) + }) + }) + } +} + +function readLocalPkg (cb) { + if (npm.config.get("global")) return cb() + var path = require("path") + , readJson = require("./utils/read-json.js") + readJson(path.resolve(npm.prefix, "package.json"), function (er, d) { + return cb(er, d && d.name) + }) +} + +function unknown (action, cb) { + cb("Usage: \n"+owner.usage) +} diff --git a/deps/npm/lib/pack.js b/deps/npm/lib/pack.js new file mode 100644 index 0000000000..3475c4a3ef --- /dev/null +++ b/deps/npm/lib/pack.js @@ -0,0 +1,66 @@ +// npm pack +// Packs the specified package into a .tgz file, which can then +// be installed. + +module.exports = pack + +var npm = require("./npm.js") + , install = require("./install.js") + , cache = require("./cache.js") + , output = require("./utils/output.js") + , fs = require("graceful-fs") + , chain = require("slide").chain + , path = require("path") + , relativize = require("./utils/relativize.js") + , cwd = process.cwd() + +pack.usage = "npm pack " + +// if it can be installed, it can be packed. +pack.completion = install.completion + +function pack (args, silent, cb) { + if (typeof cb !== "function") cb = silent, silent = false + + if (args.length === 0) args = ["."] + + chain(args.map(function (arg) { return function (cb) { + pack_(arg, cb) + }}), function (er, files) { + if (er || silent) return cb(er, files) + printFiles(files, cb) + }) +} + +function printFiles (files, cb) { + files = files.map(function (file) { + return relativize(file, cwd) + }) + output.write(files.join("\n"), cb) +} + +// add to cache, then cp to the cwd +function pack_ (pkg, cb) { + cache.add(pkg, function (er, data) { + if (er) return cb(er) + var fname = path.resolve(data._id.replace(/@/g, "-") + ".tgz") + , cached = path.resolve( npm.cache + , data.name + , data.version + , "package.tgz" ) + , from = fs.createReadStream(cached) + , to = fs.createWriteStream(fname) + , errState = null + + from.on("error", cb_) + to.on("error", cb_) + to.on("close", cb_) + from.pipe(to) + + function cb_ (er) { + if (errState) return + if (er) return cb(errState = er) + cb(null, fname) + } + }) +} diff --git a/deps/npm/lib/prefix.js b/deps/npm/lib/prefix.js new file mode 100644 index 0000000000..f37aacf2c5 --- /dev/null +++ b/deps/npm/lib/prefix.js @@ -0,0 +1,10 @@ +module.exports = prefix + +var npm = require("./npm.js") + , output = require("./utils/output.js") + +prefix.usage = "npm prefix\nnpm prefix -g\n(just prints the prefix folder)" + +function prefix (args, cb) { + output.write(npm.prefix, function (er) { cb(er, npm.prefix) }) +} diff --git a/deps/npm/lib/prune.js b/deps/npm/lib/prune.js new file mode 100644 index 0000000000..4a02dc1f8a --- /dev/null +++ b/deps/npm/lib/prune.js @@ -0,0 +1,40 @@ +// prune extraneous packages. + +module.exports = prune + +prune.usage = "npm prune" + +var readInstalled = require("./utils/read-installed.js") + , npm = require("./npm.js") + +prune.completion = require("./utils/completion/installed-deep.js") + +function prune (args, cb) { + readInstalled(npm.prefix, function (er, data) { + if (er) return cb(er) + prune_(args, data, cb) + }) +} + +function prune_ (args, data, cb) { + npm.commands.unbuild(prunables(args, data, []), cb) +} + +function prunables (args, data, seen) { + var deps = data.dependencies || {} + return Object.keys(deps).map(function (d) { + if (typeof deps[d] !== "object" + || seen.indexOf(deps[d]) !== -1) return null + seen.push(deps[d]) + if (deps[d].extraneous + && (args.length === 0 || args.indexOf(d) !== -1)) { + var extra = deps[d] + delete deps[d] + return extra.path + } + return prunables(args, deps[d], seen) + }).filter(function (d) { return d !== null }) + .reduce(function FLAT (l, r) { + return l.concat(Array.isArray(r) ? r.reduce(FLAT,[]) : r) + }, []) +} diff --git a/deps/npm/lib/publish.js b/deps/npm/lib/publish.js new file mode 100644 index 0000000000..a5855ffdee --- /dev/null +++ b/deps/npm/lib/publish.js @@ -0,0 +1,166 @@ + +module.exports = publish + +var npm = require("./npm.js") + , registry = require("./utils/npm-registry-client/index.js") + , log = require("./utils/log.js") + , tar = require("./utils/tar.js") + , sha = require("./utils/sha.js") + , path = require("path") + , readJson = require("./utils/read-json.js") + , fs = require("graceful-fs") + , lifecycle = require("./utils/lifecycle.js") + , chain = require("slide").chain + , output = require("./utils/output.js") + +publish.usage = "npm publish " + + "\nnpm publish " + + "\n\nPublishes '.' if no argument supplied" + +publish.completion = function (opts, cb) { + // publish can complete to a folder with a package.json + // or a tarball, or a tarball url. + // for now, not yet implemented. + return cb() +} + +function publish (args, isRetry, cb) { + if (typeof cb !== "function") cb = isRetry, isRetry = false + if (args.length === 0) args = ["."] + if (args.length !== 1) return cb(publish.usage) + + log.verbose(args, "publish") + var arg = args[0] + // if it's a local folder, then run the prepublish there, first. + readJson(path.resolve(arg, "package.json"), function (er, data) { + // error is ok. could be publishing a url or tarball + if (er) return cacheAddPublish(arg, false, isRetry, cb) + lifecycle(data, "prepublish", arg, function (er) { + if (er) return cb(er) + cacheAddPublish(arg, true, isRetry, cb) + }) + }) +} + +function cacheAddPublish (arg, didPre, isRetry, cb) { + npm.commands.cache.add(arg, function (er, data) { + if (er) return cb(er) + log.silly(data, "publish") + var cachedir = path.resolve( npm.cache + , data.name + , data.version + , "package" ) + chain + ( [ !didPre && [lifecycle, data, "prepublish", cachedir] + , [publish_, arg, data, isRetry, cachedir] + , [lifecycle, data, "publish", cachedir] + , [lifecycle, data, "postpublish", cachedir] ] + , cb ) + }) +} + +function publish_ (arg, data, isRetry, cachedir, cb) { + if (!data) return cb(new Error("no package.json file found")) + + // check for publishConfig hash + if (data.publishConfig) { + Object.keys(data.publishConfig).forEach(function (k) { + log.info(k + "=" + data.publishConfig[k], "publishConfig") + npm.config.set(k, data.publishConfig[k]) + }) + } + + delete data.modules + if (data.private) return cb(new Error + ("This package has been marked as private\n" + +"Remove the 'private' field from the package.json to publish it.")) + + // pre-build + var bd = data.scripts + && ( data.scripts.preinstall + || data.scripts.install + || data.scripts.postinstall ) + && npm.config.get("bindist") + && npm.config.get("bin-publish") + preBuild(data, bd, function (er, tb) { + if (er) return cb(er) + return regPublish(data, tb, isRetry, arg, cachedir, cb) + }) +} + + +function preBuild (data, bd, cb) { + if (!bd) return cb() + // unpack to cache/n/v/build + // build there + // pack to cache/package-.tgz + var cf = path.resolve(npm.cache, data.name, data.version) + var pb = path.resolve(cf, "build") + , buildTarget = path.resolve(pb, "node_modules", data.name) + , tb = path.resolve(cf, "package-"+bd+".tgz") + , sourceBall = path.resolve(cf, "package.tgz") + + log.verbose("about to cache unpack") + log.verbose(sourceBall, "the tarball") + npm.commands.install(pb, sourceBall, function (er) { + log.info(data._id, "prebuild done") + // build failure just means that we can't prebuild + if (er) { + log.warn(er.message, "prebuild failed "+bd) + return cb() + } + // now strip the preinstall/install scripts + // they've already been run. + var pbj = path.resolve(buildTarget, "package.json") + readJson(pbj, function (er, pbo) { + if (er) return cb(er) + if (pbo.scripts) { + delete pbo.scripts.preinstall + delete pbo.scripts.install + delete pbo.scripts.postinstall + } + pbo.prebuilt = bd + pbo.files = pbo.files || [] + pbo.files.push("build") + pbo.files.push("build/") + pbo.files.push("*.node") + pbo.files.push("*.js") + fs.writeFile(pbj, JSON.stringify(pbo, null, 2), function (er) { + if (er) return cb(er) + tar.pack(tb, buildTarget, pbo, true, function (er) { + if (er) return cb(er) + // try to validate the shasum, too + sha.get(tb, function (er, shasum) { + if (er) return cb(er) + // binary distribution requires shasum checking. + if (!shasum) return cb() + data.dist.bin = data.dist.bin || {} + data.dist.bin[bd] = data.dist.bin[bd] || {} + data.dist.bin[bd].shasum = shasum + return cb(null, tb) + }) + }) + }) + }) + }) +} + +function regPublish (data, prebuilt, isRetry, arg, cachedir, cb) { + // check to see if there's a README.md in there. + var readme = path.resolve(cachedir, "README.md") + fs.readFile(readme, function (er, readme) { + // ignore error. it's an optional feature + registry.publish(data, prebuilt, readme, function (er) { + if (er && er.errno === npm.EPUBLISHCONFLICT + && npm.config.get("force") && !isRetry) { + log.warn("Forced publish over "+data._id, "publish") + return npm.commands.unpublish([data._id], function (er) { + // ignore errors. Use the force. Reach out with your feelings. + publish([arg], true, cb) + }) + } + if (er) return cb(er) + output.write("+ " + data._id, cb) + }) + }) +} diff --git a/deps/npm/lib/rebuild.js b/deps/npm/lib/rebuild.js new file mode 100644 index 0000000000..f3f7596204 --- /dev/null +++ b/deps/npm/lib/rebuild.js @@ -0,0 +1,68 @@ + +module.exports = rebuild + +var readInstalled = require("./utils/read-installed.js") + , semver = require("semver") + , log = require("./utils/log.js") + , path = require("path") + , npm = require("./npm.js") + , output = require("./utils/output.js") + +rebuild.usage = "npm rebuild [[@] [name[@] ...]]" + +rebuild.completion = require("./utils/completion/installed-deep.js") + +function rebuild (args, cb) { + readInstalled(npm.prefix, function (er, data) { + log(typeof data, "read Installed") + if (er) return cb(er) + var set = filter(data, args) + , folders = Object.keys(set).filter(function (f) { + return f !== npm.prefix + }) + if (!folders.length) return cb() + log.silly(folders, "rebuild set") + npm.commands.build(folders, function (er) { + if (er) return cb(er) + output.write(folders.map(function (f) { + return set[f] + " " + f + }).join("\n"), cb) + }) + }) +} + +function filter (data, args, set, seen) { + if (!set) set = {} + if (!seen) seen = {} + if (set.hasOwnProperty(data.path)) return set + if (seen.hasOwnProperty(data.path)) return set + seen[data.path] = true + var pass + if (!args.length) pass = true // rebuild everything + else if (data.name && data._id) { + for (var i = 0, l = args.length; i < l; i ++) { + var arg = args[i] + , nv = arg.split("@") + , n = nv.shift() + , v = nv.join("@") + if (n !== data.name) continue + if (!semver.satisfies(data.version, v)) continue + pass = true + break + } + } + if (pass && data._id) { + log.verbose([data.path, data._id], "path id") + set[data.path] = data._id + } + // need to also dive through kids, always. + // since this isn't an install these won't get auto-built unless + // they're not dependencies. + Object.keys(data.dependencies || {}).forEach(function (d) { + // return + var dep = data.dependencies[d] + if (typeof dep === "string") return + filter(dep, args, set, seen) + }) + return set +} diff --git a/deps/npm/lib/restart.js b/deps/npm/lib/restart.js new file mode 100644 index 0000000000..69c4b913db --- /dev/null +++ b/deps/npm/lib/restart.js @@ -0,0 +1 @@ +module.exports = require("./utils/lifecycle.js").cmd("restart") diff --git a/deps/npm/lib/root.js b/deps/npm/lib/root.js new file mode 100644 index 0000000000..6062ec2202 --- /dev/null +++ b/deps/npm/lib/root.js @@ -0,0 +1,11 @@ +module.exports = root + +var npm = require("./npm.js") + , output = require("./utils/output.js") + , log = require("./utils/log.js") + +root.usage = "npm root\nnpm root -g\n(just prints the root folder)" + +function root (args, cb) { + output.write(npm.dir, function (er) { cb(er, npm.dir) }) +} diff --git a/deps/npm/lib/run-script.js b/deps/npm/lib/run-script.js new file mode 100644 index 0000000000..4a4d2dc610 --- /dev/null +++ b/deps/npm/lib/run-script.js @@ -0,0 +1,100 @@ + +module.exports = runScript + +var lifecycle = require("./utils/lifecycle.js") + , npm = require("./npm.js") + , path = require("path") + , readJson = require("./utils/read-json.js") + , log = require("./utils/log.js") + , chain = require("slide").chain + , fs = require("graceful-fs") + , asyncMap = require("slide").asyncMap + +runScript.usage = "npm run-script [] " + +runScript.completion = function (opts, cb) { + + // see if there's already a package specified. + var argv = opts.conf.argv.remain + , installedShallow = require("./utils/completion/installed-shallow.js") + + if (argv.length >= 4) return cb() + + if (argv.length === 3) { + // either specified a script locally, in which case, done, + // or a package, in which case, complete against its scripts + var json = path.join(npm.prefix, "package.json") + return readJson(json, function (er, d) { + if (er) d = {} + var scripts = Object.keys(d.scripts || {}) + console.error("local scripts", scripts) + if (scripts.indexOf(argv[2]) !== -1) return cb() + // ok, try to find out which package it was, then + var pref = npm.config.get("global") ? npm.config.get("prefix") + : npm.prefix + var pkgDir = path.resolve( pref, "node_modules" + , argv[2], "package.json" ) + console.error("global?", npm.config.get("global")) + console.error(pkgDir, "package dir") + readJson(pkgDir, function (er, d) { + if (er) d = {} + var scripts = Object.keys(d.scripts || {}) + return cb(null, scripts) + }) + }) + } + + // complete against the installed-shallow, and the pwd's scripts. + // but only packages that have scripts + var installed + , scripts + installedShallow(opts, function (d) { + return d.scripts + }, function (er, inst) { + installed = inst + next() + }) + + if (npm.config.get("global")) scripts = [], next() + else readJson(path.join(npm.prefix, "package.json"), function (er, d) { + d = d || {} + scripts = Object.keys(d.scripts || {}) + next() + }) + + function next () { + if (!installed || !scripts) return + return cb(null, scripts.concat(installed)) + } +} + +function runScript (args, cb) { + var pkgdir = args.length === 1 ? process.cwd() + : path.resolve(npm.dir, args[0]) + , cmd = args.pop() + + readJson(path.resolve(pkgdir, "package.json"), function (er, d) { + if (er) return cb(er) + run(d, pkgdir, cmd, cb) + }) +} + +function run (pkg, wd, cmd, cb) { + var cmds = [] + if (!pkg.scripts) pkg.scripts = {} + if (cmd === "restart") { + cmds = ["prestop","stop","poststop" + ,"restart" + ,"prestart","start","poststart"] + } else { + cmds = [cmd] + } + if (!cmd.match(/^(pre|post)/)) { + cmds = ["pre"+cmd].concat(cmds).concat("post"+cmd) + } + log.verbose(cmds, "run-script") + chain(cmds.map(function (c) { + // when running scripts explicitly, assume that they're trusted. + return [lifecycle, pkg, c, wd, true] + }), cb) +} diff --git a/deps/npm/lib/search.js b/deps/npm/lib/search.js new file mode 100644 index 0000000000..45e436a0c1 --- /dev/null +++ b/deps/npm/lib/search.js @@ -0,0 +1,222 @@ + +module.exports = exports = search + +var npm = require("./npm.js") + , registry = require("./utils/npm-registry-client/index.js") + , semver = require("semver") + , output + , log = require("./utils/log.js") + +search.usage = "npm search [some search terms ...]" + +search.completion = function (opts, cb) { + var compl = {} + , partial = opts.partialWord + , ipartial = partial.toLowerCase() + , plen = partial.length + + // get the batch of data that matches so far. + // this is an example of using npm.commands.search programmatically + // to fetch data that has been filtered by a set of arguments. + search(opts.conf.argv.remain.slice(2), true, function (er, data) { + if (er) return cb(er) + Object.keys(data).forEach(function (name) { + data[name].words.split(" ").forEach(function (w) { + if (w.toLowerCase().indexOf(ipartial) === 0) { + compl[partial + w.substr(plen)] = true + } + }) + }) + cb(null, Object.keys(compl)) + }) +} + +function search (args, silent, staleness, cb_) { + if (typeof cb_ !== "function") cb_ = staleness, staleness = 600 + if (typeof cb_ !== "function") cb_ = silent, silent = false + + var searchopts = npm.config.get("searchopts") + , searchexclude = npm.config.get("searchexclude") + if (typeof searchopts !== "string") searchopts = "" + searchopts = searchopts.split(/\s+/) + if (typeof searchexclude === "string") { + searchexclude = searchexclude.split(/\s+/) + } else searchexclude = [] + var opts = searchopts.concat(args).map(function (s) { + return s.toLowerCase() + }).filter(function (s) { return s }) + searchexclude = searchexclude.map(function (s) { + return s.toLowerCase() + }) + getFilteredData( staleness, opts, searchexclude, function (er, data) { + // now data is the list of data that we want to show. + // prettify and print it, and then provide the raw + // data to the cb. + if (er || silent) return cb_(er, data) + function cb (er) { return cb_(er, data) } + output = output || require("./utils/output.js") + output.write(prettify(data, args), cb) + }) +} + +function getFilteredData (staleness, args, notArgs, cb) { + registry.get( "/-/all", null, staleness, false + , true, function (er, data) { + if (er) return cb(er) + return cb(null, filter(data, args, notArgs)) + }) +} + +function filter (data, args, notArgs) { + // data={:{package data}} + return Object.keys(data).map(function (d) { + return data[d] + }).filter(function (d) { + return typeof d === "object" + }).map(stripData).map(getWords).filter(function (data) { + return filterWords(data, args, notArgs) + }).reduce(function (l, r) { + l[r.name] = r + return l + }, {}) +} + +function stripData (data) { + return { name:data.name + , description:npm.config.get("description") ? data.description : "" + , maintainers:(data.maintainers || []).map(function (m) { + return "=" + m.name + }) + , url:!Object.keys(data.versions || {}).length ? data.url : null + , keywords:data.keywords || [] + } +} + +function getWords (data) { + data.words = [ data.name ] + .concat(data.description) + .concat(data.maintainers) + .concat(data.url && ("<" + data.url + ">")) + .concat(data.keywords) + .map(function (f) { return f && f.trim && f.trim() }) + .filter(function (f) { return f }) + .join(" ") + .toLowerCase() + return data +} + +function filterWords (data, args, notArgs) { + var words = data.words + for (var i = 0, l = args.length; i < l; i ++) { + if (words.indexOf(args[i]) === -1) { + return false + } + } + for (var i = 0, l = notArgs.length; i < l; i ++) { + if (words.indexOf(notArgs[i]) !== -1) return false + } + return true +} + +function prettify (data, args) { + try { + var tty = require("tty") + , stdout = process.stdout + , cols = !tty.isatty(stdout.fd) ? Infinity + : stdout._handle ? stdout._handle.getWindowSize()[0] + : tty.getWindowSize()[1] + } catch (ex) { cols = Infinity } + + // name, desc, author, keywords + var longest = [] + , spaces + , maxLen = [20, 60, 20, Infinity] + , headings = ["NAME", "DESCRIPTION", "AUTHOR", "KEYWORDS"] + , lines + + lines = Object.keys(data).map(function (d) { + return data[d] + }).filter(function (data) { + return data.name + }).map(function (data) { + // turn a pkg data into a string + // [name,who,desc,targets,keywords] tuple + // also set longest to the longest name + if (typeof data.keywords === "string") { + data.keywords = data.keywords.split(/[,\s]+/) + } + if (!Array.isArray(data.keywords)) data.keywords = [] + var l = [ data.name + , data.description || "" + , data.maintainers.join(" ") + , (data.keywords || []).join(" ") + ] + l.forEach(function (s, i) { + var len = s.length + longest[i] = Math.min(maxLen[i] || Infinity + ,Math.max(longest[i] || 0, len)) + if (len > longest[i]) { + l._undent = l._undent || [] + l._undent[i] = len - longest[i] + } + l[i] = l[i].replace(/\s+/g, " ") + }) + return l + }).map(function (line) { + return line.map(function (s, i) { + spaces = spaces || longest.map(function (n) { + return new Array(n + 2).join(" ") + }) + var len = s.length + if (line._undent && line._undent[i - 1]) { + len += line._undent[i - 1] - 1 + } + return s + spaces[i].substr(len) + }).join(" ").substr(0, cols).trim() + }).sort(function (a, b) { + return a === b ? 0 : a > b ? 1 : -1 + }).map(function (line) { + // colorize! + args.forEach(function (arg, i) { + line = addColorMarker(line, arg, i) + }) + return colorize(line).trim() + }) + + if (lines.length === 0) { + return "No match found for "+(args.map(JSON.stringify).join(" ")) + } + + // build the heading padded to the longest in each field + return headings.map(function (h, i) { + var space = Math.max(2, 3 + (longest[i] || 0) - h.length) + return h + (new Array(space).join(" ")) + }).join("").substr(0, cols).trim() + "\n" + lines.join("\n") + +} + +var colors = [31, 33, 32, 36, 34, 35 ] + , cl = colors.length +function addColorMarker (str, arg, i) { + var m = i % cl + 1 + , markStart = String.fromCharCode(m) + , markEnd = String.fromCharCode(0) + , pieces = str.toLowerCase().split(arg.toLowerCase()) + , p = 0 + return pieces.map(function (piece, i) { + piece = str.substr(p, piece.length) + var mark = markStart + + str.substr(p+piece.length, arg.length) + + markEnd + p += piece.length + arg.length + return piece + mark + }).join("") + return str.split(arg).join(mark) +} +function colorize (line) { + for (var i = 0; i < cl; i ++) { + var m = i + 1 + line = line.split(String.fromCharCode(m)).join("\033["+colors[i]+"m") + } + return line.split("\u0000").join("\033[0m") +} diff --git a/deps/npm/lib/set.js b/deps/npm/lib/set.js new file mode 100644 index 0000000000..d821095157 --- /dev/null +++ b/deps/npm/lib/set.js @@ -0,0 +1,12 @@ + +module.exports = set + +set.usage = "npm set (See `npm config`)" + +var npm = require("./npm.js") + +set.completion = npm.commands.config.completion + +function set (args, cb) { + npm.commands.config(["set"].concat(args), cb) +} diff --git a/deps/npm/lib/star.js b/deps/npm/lib/star.js new file mode 100644 index 0000000000..d84fa02ed4 --- /dev/null +++ b/deps/npm/lib/star.js @@ -0,0 +1,34 @@ + +module.exports = star + +var npm = require("./npm.js") + , registry = require("./utils/npm-registry-client/index.js") + , log = require("./utils/log.js") + , asyncMap = require("slide").asyncMap + , output = require("./utils/output.js") + +star.usage = "npm star [pkg, pkg, ...]\n" + + "npm unstar [pkg, pkg, ...]" + +star.completion = function (opts, cb) { + registry.get("/-/short", null, 60000, function (er, list) { + return cb(null, list || []) + }) +} + +function star (args, cb) { + if (!args.length) return cb(star.usage) + var s = npm.config.get("unicode") ? "\u2605 " : "(*)" + , u = npm.config.get("unicode") ? "\u2606 " : "( )" + , using = !(npm.command.match(/^un/)) + if (!using) s = u + asyncMap(args, function (pkg, cb) { + registry.star(pkg, using, function (er, data, raw, req) { + if (!er) { + output.write(s + " "+pkg, npm.config.get("outfd")) + log.verbose(data, "back from star/unstar") + } + cb(er, data, raw, req) + }) + }, cb) +} diff --git a/deps/npm/lib/start.js b/deps/npm/lib/start.js new file mode 100644 index 0000000000..98823825bb --- /dev/null +++ b/deps/npm/lib/start.js @@ -0,0 +1 @@ +module.exports = require("./utils/lifecycle.js").cmd("start") diff --git a/deps/npm/lib/stop.js b/deps/npm/lib/stop.js new file mode 100644 index 0000000000..8ea5ba6aa6 --- /dev/null +++ b/deps/npm/lib/stop.js @@ -0,0 +1 @@ +module.exports = require("./utils/lifecycle.js").cmd("stop") diff --git a/deps/npm/lib/submodule.js b/deps/npm/lib/submodule.js new file mode 100644 index 0000000000..92fb41f883 --- /dev/null +++ b/deps/npm/lib/submodule.js @@ -0,0 +1,93 @@ +// npm submodule +// Check the package contents for a git repository url. +// If there is one, then create a git submodule in the node_modules folder. + +module.exports = submodule + +var npm = require("./npm.js") + , readJson = require("./utils/read-json.js") + , exec = require("./utils/exec.js") + , output = require("./utils/output.js") + , cache = require("./cache.js") + , asyncMap = require("slide").asyncMap + , chain = require("slide").chain + +submodule.usage = "npm submodule " + +submodule.completion = require("./docs.js").completion + +function submodule (args, cb) { + if (npm.config.get("global")) { + return cb(new Error("Cannot use submodule command in global mode.")) + } + + if (args.length === 0) return cb(submodule.usage) + + asyncMap(args, function (arg, cb) { + cache.add(arg, cb) + }, function (er, pkgs) { + if (er) return cb(er) + chain(pkgs.map(function (pkg) { return function (cb) { + submodule_(pkg, cb) + }}), cb) + }) + +} + +function submodule_ (pkg, cb) { + if (!pkg.repository + || pkg.repository.type !== "git" + || !pkg.repository.url) { + return cb(new Error(pkg._id + ": No git repository listed")) + } + + // prefer https:// github urls + pkg.repository.url = pkg.repository.url + .replace(/^(git:\/\/)?(git@)?github.com[:\/]/, "https://github.com/") + + // first get the list of submodules, and update if it's already there. + getSubmodules(function (er, modules) { + if (er) return cb(er) + // if there's already a submodule, then just update it. + if (modules.indexOf(pkg.name) !== -1) { + return updateSubmodule(pkg.name, cb) + } + addSubmodule(pkg.name, pkg.repository.url, cb) + }) +} + +function updateSubmodule (name, cb) { + exec( "git", [ "submodule", "update", "--init" + , "node_modules/" + name ] + , null, true, npm.prefix, cb) +} + +function addSubmodule (name, url, cb) { + exec( "git", [ "submodule", "add", url + , "node_modules/" + name ] + , null, true, npm.prefix, function (er) { + if (er) return cb(er) + updateSubmodule(name, cb) + }) +} + + +var getSubmodules = function getSubmodules (cb) { + exec( "git", ["submodule", "status"], null, false + , npm.prefix, function (er, code, stdout, stderr) { + if (er) return cb(er) + res = stdout.trim().split(/\n/).map(function (line) { + return line.trim().split(/\s+/)[1] + }).filter(function (line) { + // only care about submodules in the node_modules folder. + return line && line.match(/^node_modules\//) + }).map(function (line) { + return line.replace(/^node_modules\//g, "") + }) + + // memoize. + getSubmodules = function (cb) { return cb(null, res) } + + cb(null, res) + }) +} diff --git a/deps/npm/lib/tag.js b/deps/npm/lib/tag.js new file mode 100644 index 0000000000..06aa70c5e0 --- /dev/null +++ b/deps/npm/lib/tag.js @@ -0,0 +1,17 @@ +// turns out tagging isn't very complicated +// all the smarts are in the couch. +module.exports = tag +tag.usage = "npm tag @ []" + +tag.completion = require("./unpublish.js").completion + +var npm = require("./npm.js") + +function tag (args, cb) { + var thing = (args.shift() || "").split("@") + , project = thing.shift() + , version = thing.join("@") + , t = args.shift() || npm.config.get("tag") + if (!project || !version || !t) return cb("Usage:\n"+tag.usage) + require("./utils/npm-registry-client/index.js").tag(project, version, t, cb) +} diff --git a/deps/npm/lib/test.js b/deps/npm/lib/test.js new file mode 100644 index 0000000000..1d2ee30a32 --- /dev/null +++ b/deps/npm/lib/test.js @@ -0,0 +1 @@ +module.exports = require("./utils/lifecycle.js").cmd("test") diff --git a/deps/npm/lib/unbuild.js b/deps/npm/lib/unbuild.js new file mode 100644 index 0000000000..f4bd008a66 --- /dev/null +++ b/deps/npm/lib/unbuild.js @@ -0,0 +1,85 @@ +module.exports = unbuild +unbuild.usage = "npm unbuild \n(this is plumbing)" + +var readJson = require("./utils/read-json.js") + , rm = require("rimraf") + , npm = require("./npm.js") + , path = require("path") + , fs = require("graceful-fs") + , lifecycle = require("./utils/lifecycle.js") + , asyncMap = require("slide").asyncMap + , chain = require("slide").chain + , log = require("./utils/log.js") + , build = require("./build.js") + +// args is a list of folders. +// remove any bins/etc, and then delete the folder. +function unbuild (args, cb) { asyncMap(args, unbuild_, cb) } + +function unbuild_ (folder, cb) { + folder = path.resolve(folder) + delete build._didBuild[folder] + log.info(folder, "unbuild") + readJson(path.resolve(folder, "package.json"), function (er, pkg) { + // if no json, then just trash it, but no scripts or whatever. + if (er) return rm(folder, cb) + readJson.clearCache(folder) + chain + ( [ [lifecycle, pkg, "preuninstall", folder, false, true] + , [lifecycle, pkg, "uninstall", folder, false, true] + , [rmStuff, pkg, folder] + , [lifecycle, pkg, "postuninstall", folder, false, true] + , [rm, folder] ] + , cb ) + }) +} + +function rmStuff (pkg, folder, cb) { + // if it's global, and folder is in {prefix}/node_modules, + // then bins are in {prefix}/bin + // otherwise, then bins are in folder/../.bin + var parent = path.dirname(folder) + , gnm = npm.dir + , top = gnm === parent + + log.verbose([top, gnm, parent], "unbuild "+pkg._id) + asyncMap([rmBins, rmMans], function (fn, cb) { + fn(pkg, folder, parent, top, cb) + }, cb) +} + +function rmBins (pkg, folder, parent, top, cb) { + if (!pkg.bin) return cb() + var binRoot = top ? npm.bin : path.resolve(parent, ".bin") + log.verbose([binRoot, pkg.bin], "binRoot") + asyncMap(Object.keys(pkg.bin), function (b, cb) { + if (process.platform === "win32") { + rm(path.resolve(binRoot, b) + ".cmd", cb) + } else { + rm( path.resolve(binRoot, b) + , { gently: !npm.config.get("force") && folder } + , cb ) + } + }, cb) +} + +function rmMans (pkg, folder, parent, top, cb) { + if (!pkg.man || !top || process.platform === "win32") return cb() + var manRoot = path.resolve(npm.config.get("prefix"), "share", "man") + asyncMap(pkg.man, function (man, cb) { + var parseMan = man.match(/(.*)\.([0-9]+)(\.gz)?$/) + , stem = parseMan[1] + , sxn = parseMan[2] + , gz = parseMan[3] || "" + , bn = path.basename(stem) + , manDest = path.join( manRoot + , "man"+sxn + , (bn.indexOf(pkg.name) === 0 ? bn + : pkg.name + "-" + bn) + + "." + sxn + gz + ) + rm( manDest + , { gently: !npm.config.get("force") && folder } + , cb ) + }, cb) +} diff --git a/deps/npm/lib/uninstall.js b/deps/npm/lib/uninstall.js new file mode 100644 index 0000000000..1b43607d6d --- /dev/null +++ b/deps/npm/lib/uninstall.js @@ -0,0 +1,55 @@ + +// remove a package. + +module.exports = uninstall + +uninstall.usage = "npm uninstall [@ [[@] ...]" + + "\nnpm rm [@ [[@] ...]" + +uninstall.completion = require("./utils/completion/installed-shallow.js") + +var fs = require("graceful-fs") + , log = require("./utils/log.js") + , readJson = require("./utils/read-json.js") + , path = require("path") + , npm = require("./npm.js") + , semver = require("semver") + , asyncMap = require("slide").asyncMap + +function uninstall (args, cb) { + // this is super easy + // get the list of args that correspond to package names in either + // the global npm.dir, + // then call unbuild on all those folders to pull out their bins + // and mans and whatnot, and then delete the folder. + + var nm = npm.dir + if (args.length === 1 && args[0] === ".") args = [] + if (args.length) return uninstall_(args, nm, cb) + + // remove this package from the global space, if it's installed there + if (npm.config.get("global")) return cb(uninstall.usage) + readJson(path.resolve(npm.prefix, "package.json"), function (er, pkg) { + if (er) return cb(uninstall.usage) + uninstall_( [pkg.name] + , npm.dir + , cb ) + }) + +} + +function uninstall_ (args, nm, cb) { + asyncMap(args, function (arg, cb) { + var p = path.resolve(nm, arg) + fs.lstat(p, function (er) { + if (er) { + log.warn(arg, "Not installed in "+nm) + return cb(null, []) + } + cb(null, p) + }) + }, function (er, folders) { + if (er) return cb(er) + asyncMap(folders, npm.commands.unbuild, cb) + }) +} diff --git a/deps/npm/lib/unpublish.js b/deps/npm/lib/unpublish.js new file mode 100644 index 0000000000..2945a7887f --- /dev/null +++ b/deps/npm/lib/unpublish.js @@ -0,0 +1,69 @@ + +module.exports = unpublish + +var registry = require("./utils/npm-registry-client/index.js") + , log = require("./utils/log.js") + , npm = require("./npm.js") + , readJson = require("./utils/read-json.js") + , path = require("path") + , output = require("./utils/output.js") + +unpublish.usage = "npm unpublish [@]" + +unpublish.completion = function (opts, cb) { + if (opts.conf.argv.remain.length >= 3) return cb() + var un = encodeURIComponent(npm.config.get("username")) + if (!un) return cb() + registry.get("/-/by-user/"+un, function (er, pkgs) { + // do a bit of filtering at this point, so that we don't need + // to fetch versions for more than one thing, but also don't + // accidentally a whole project. + pkgs = pkgs[un] + if (!pkgs || !pkgs.length) return cb() + var partial = opts.partialWord.split("@") + , pp = partial.shift() + , pv = partial.join("@") + pkgs = pkgs.filter(function (p) { + return p.indexOf(pp) === 0 + }) + if (pkgs.length > 1) return cb(null, pkgs) + registry.get(pkgs[0], function (er, d) { + if (er) return cb(er) + var vers = Object.keys(d.versions) + if (!vers.length) return cb(null, pkgs) + return cb(null, vers.map(function (v) { + return pkgs[0]+"@"+v + })) + }) + }) +} + +function unpublish (args, cb) { + var thing = args.length ? args.shift().split("@") : [] + , project = thing.shift() + , version = thing.join("@") + + if (!project || path.resolve(project) === npm.prefix) { + // if there's a package.json in the current folder, then + // read the package name and version out of that. + var cwdJson = path.join(process.cwd(), "package.json") + return readJson(cwdJson, function (er, data) { + if (er) return cb("Usage:\n"+unpublish.usage) + gotProject(data.name, data.version, cb) + }) + } + return gotProject(project, version, cb) +} + +function gotProject (project, version, cb_) { + function cb (er) { + if (er) return cb_(er) + output.write("- " + project + (version ? "@" + version : ""), cb_) + } + + // remove from the cache first + npm.commands.cache(["clean", project, version], function (er) { + if (er) return log.er(cb, "Failed to clean cache")(er) + registry.unpublish(project, version, cb) + }) +} diff --git a/deps/npm/lib/update.js b/deps/npm/lib/update.js new file mode 100644 index 0000000000..69b9f98e83 --- /dev/null +++ b/deps/npm/lib/update.js @@ -0,0 +1,39 @@ +/* +for each pkg in prefix that isn't a git repo + look for a new version of pkg that satisfies dep + if so, install it. + if not, then update it +*/ + +module.exports = update + +update.usage = "npm update [pkg]" + +var npm = require("./npm.js") + , lifecycle = require("./utils/lifecycle.js") + , asyncMap = require("slide").asyncMap + , log = require("./utils/log.js") + + // load these, just so that we know that they'll be available, in case + // npm itself is getting overwritten. + , install = require("./install.js") + , build = require("./build.js") + +update.completion = npm.commands.outdated.completion + +function update (args, cb) { + npm.commands.outdated(args, true, function (er, outdated) { + log(outdated, "outdated updating") + if (er) return cb(er) + + asyncMap(outdated, function (ww, cb) { + // [[ dir, dep, has, want ]] + var where = ww[0] + , dep = ww[1] + , want = ww[3] + , what = dep + "@" + want + + npm.commands.install(where, what, cb) + }, cb) + }) +} diff --git a/deps/npm/lib/utils/cmd-shim.js b/deps/npm/lib/utils/cmd-shim.js new file mode 100644 index 0000000000..802fdb8fc3 --- /dev/null +++ b/deps/npm/lib/utils/cmd-shim.js @@ -0,0 +1,98 @@ +// XXX Todo: +// On windows, create a .cmd file. +// Read the #! in the file to see what it uses. The vast majority +// of the time, this will be either: +// "#!/usr/bin/env " +// or: +// "#! " +// +// Write a binroot/pkg.bin + ".cmd" file that has this line in it: +// @ %~dp0 %* + +module.exports = cmdShim +cmdShim.ifExists = cmdShimIfExists + +var fs = require("graceful-fs") + , chain = require("slide").chain + , mkdir = require("./mkdir-p.js") + , rm = require("rimraf") + , log = require("./log.js") + , path = require("path") + , relativize = require("./relativize.js") + , npm = require("../npm.js") + , shebangExpr = /^#\!(?:\/usr\/bin\/env )?([^ \t]+)(.*)$/ + +function cmdShimIfExists (from, to, cb) { + fs.stat(from, function (er) { + if (er) return cb() + cmdShim(from, to, cb) + }) +} + +function cmdShim (from, to, cb) { + if (process.platform !== "win32") { + return cb(new Error(".cmd shims only should be used on windows")) + } + + chain + ( [ [fs, "stat", from] + , [rm, to + ".cmd"] + , [mkdir, path.dirname(to)] + , [writeShim, from, to] ] + , cb ) +} + +function writeShim (from, to, cb) { + // make a cmd file + // First, check if the bin is a #! of some sort. + // If not, then assume it's something that'll be compiled, or some other + // sort of script, and just call it directly. + fs.readFile(from, "utf8", function (er, data) { + if (er) return writeShim_(from, to, null, null, cb) + var firstLine = data.trim().split(/\r*\n/)[0] + , shebang = firstLine.match(shebangExpr) + if (!shebang) return writeShim_(from, to, null, null, cb) + var prog = shebang[1] + , args = shebang[2] || "" + return writeShim_(from, to, prog, args, cb) + }) +} + +function writeShim_ (from, to, prog, args, cb) { + var target = relativize(from, to).split("/").join("\\") + , longProg + args = args || "" + if (!prog) { + prog = "\"%~dp0\\" + target + "\"" + args = "" + target = "" + } else { + longProg = "\"%~dp0\"\\\"" + prog + ".exe\"" + target = "\"%~dp0\\" + target + "\"" + } + + // @IF EXIST "%~dp0"\"node.exe" ( + // "%~dp0\node.exe" "%~dp0\.\node_modules\npm\bin\npm-cli.js" %* + // ) ELSE ( + // node "%~dp0\.\node_modules\npm\bin\npm-cli.js" %* + // ) + var cmd + if (longProg) { + cmd = "@IF EXIST " + longProg + " (\r\n" + + " " + longProg + " " + args + " " + target + " %*\r\n" + + ") ELSE (\r\n" + + " " + prog + " " + args + " " + target + " %*\r\n" + + ")" + } else { + cmd = prog + " " + args + " " + target + " %*\r\n" + } + + cmd = ":: Created by npm, please don't edit manually.\r\n" + cmd + + fs.writeFile(to + ".cmd", cmd, "utf8", function (er) { + if (er) { + log.warn("Could not write "+to+".cmd", "cmdShim") + } + cb(er) + }) +} diff --git a/deps/npm/lib/utils/completion.sh b/deps/npm/lib/utils/completion.sh new file mode 100755 index 0000000000..f19046b198 --- /dev/null +++ b/deps/npm/lib/utils/completion.sh @@ -0,0 +1,43 @@ +#!/bin/bash +###-begin-npm-completion-### +# +# npm command completion script +# +# Installation: npm completion >> ~/.bashrc (or ~/.zshrc) +# Or, maybe: npm completion > /usr/local/etc/bash_completion.d/npm +# + +COMP_WORDBREAKS=${COMP_WORDBREAKS/=/} +COMP_WORDBREAKS=${COMP_WORDBREAKS/@/} +export COMP_WORDBREAKS + +if complete &>/dev/null; then + _npm_completion () { + local si="$IFS" + IFS=$'\n' COMPREPLY=($(COMP_CWORD="$COMP_CWORD" \ + COMP_LINE="$COMP_LINE" \ + COMP_POINT="$COMP_POINT" \ + npm completion -- "${COMP_WORDS[@]}" \ + 2>/dev/null)) || return $? + IFS="$si" + } + complete -F _npm_completion npm +elif compctl &>/dev/null; then + _npm_completion () { + local cword line point words si + read -Ac words + read -cn cword + let cword-=1 + read -l line + read -ln point + si="$IFS" + IFS=$'\n' reply=($(COMP_CWORD="$cword" \ + COMP_LINE="$line" \ + COMP_POINT="$point" \ + npm completion -- "${words[@]}" \ + 2>/dev/null)) || return $? + IFS="$si" + } + compctl -K _npm_completion npm +fi +###-end-npm-completion-### diff --git a/deps/npm/lib/utils/completion/file-completion.js b/deps/npm/lib/utils/completion/file-completion.js new file mode 100644 index 0000000000..427efefb44 --- /dev/null +++ b/deps/npm/lib/utils/completion/file-completion.js @@ -0,0 +1,29 @@ +module.exports = fileCompletion + +var find = require("../find.js") + , mkdir = require("../mkdir-p.js") + , path = require("path") + +function fileCompletion (root, req, depth, cb) { + if (typeof cb !== "function") cb = depth, depth = Infinity + mkdir(root, function (er) { + if (er) return cb(er) + function dirFilter (f, type) { + // return anything that is a file, + // or not exactly the req. + return type !== "dir" || + ( f && f !== path.join(root, req) + && f !== path.join(root, req) + "/" ) + } + find(path.join(root, req), dirFilter, depth, function (er, files) { + if (er) return cb(er) + return cb(null, (files || []).map(function (f) { + return path.join(req, f.substr(root.length + 1) + .substr((f === req ? path.dirname(req) + : req).length) + .replace(/^\//, "")) + })) + }) + }) +} + diff --git a/deps/npm/lib/utils/completion/installed-deep.js b/deps/npm/lib/utils/completion/installed-deep.js new file mode 100644 index 0000000000..1188f408fa --- /dev/null +++ b/deps/npm/lib/utils/completion/installed-deep.js @@ -0,0 +1,46 @@ +module.exports = installedDeep + +var npm = require("../../npm.js") + , readInstalled = require("../read-installed.js") + +function installedDeep (opts, cb) { + var local + , global + if (npm.config.get("global")) local = [], next() + else readInstalled(npm.prefix, function (er, data) { + local = getNames(data || {}) + next() + }) + readInstalled(npm.config.get("prefix"), function (er, data) { + global = getNames(data || {}) + next() + }) + + function getNames_ (d, n) { + if (d.realName && n) { + if (n[d.realName]) return n + n[d.realName] = true + } + if (!n) n = {} + Object.keys(d.dependencies || {}).forEach(function (dep) { + getNames_(d.dependencies[dep], n) + }) + return n + } + function getNames (d) { + return Object.keys(getNames_(d)) + } + + function next () { + if (!local || !global) return + if (!npm.config.get("global")) { + global = global.map(function (g) { + return [g, "-g"] + }) + } + var names = local.concat(global) + return cb(null, names) + } + +} + diff --git a/deps/npm/lib/utils/completion/installed-shallow.js b/deps/npm/lib/utils/completion/installed-shallow.js new file mode 100644 index 0000000000..1ee3cce66c --- /dev/null +++ b/deps/npm/lib/utils/completion/installed-shallow.js @@ -0,0 +1,79 @@ + +module.exports = installedShallow + +var npm = require("../../npm.js") + , fs = require("graceful-fs") + , path = require("path") + , readJson = require("../read-json.js") + , asyncMap = require("slide").asyncMap + +function installedShallow (opts, filter, cb) { + if (typeof cb !== "function") cb = filter, filter = null + var conf = opts.conf + , args = conf.argv.remain + if (args.length > 3) return cb() + var local + , global + , localDir = npm.dir + , globalDir = npm.globalDir + if (npm.config.get("global")) local = [], next() + else fs.readdir(localDir, function (er, pkgs) { + local = (pkgs || []).filter(function (p) { + return p.charAt(0) !== "." + }) + next() + }) + fs.readdir(globalDir, function (er, pkgs) { + global = (pkgs || []).filter(function (p) { + return p.charAt(0) !== "." + }) + next() + }) + function next () { + if (!local || !global) return + filterInstalled(local, global, filter, cb) + } +} + +function filterInstalled (local, global, filter, cb) { + var fl + , fg + + if (!filter) { + fl = local + fg = global + return next() + } + + asyncMap(local, function (p, cb) { + readJson(path.join(npm.dir, p, "package.json"), function (er, d) { + if (!d || !filter(d)) return cb(null, []) + return cb(null, d.name) + }) + }, function (er, local) { + fl = local || [] + next() + }) + + var globalDir = npm.globalDir + asyncMap(global, function (p, cb) { + readJson(path.join(globalDir, p, "package.json"), function (er, d) { + if (!d || !filter(d)) return cb(null, []) + return cb(null, d.name) + }) + }, function (er, global) { + fg = global || [] + next() + }) + + function next () { + if (!fg || !fl) return + if (!npm.config.get("global")) { + fg = fg.map(function (g) { + return [g, "-g"] + }) + } + console.error("filtered", fl, fg) + return cb(null, fl.concat(fg)) + } +} diff --git a/deps/npm/lib/utils/completion/remote-packages.js b/deps/npm/lib/utils/completion/remote-packages.js new file mode 100644 index 0000000000..4bf82d070a --- /dev/null +++ b/deps/npm/lib/utils/completion/remote-packages.js @@ -0,0 +1,57 @@ + +module.exports = remotePackages + +var registry = require("../npm-registry-client/index.js") + , containsSingleMatch = require("./contains-single-match.js") + , getCompletions = require("./get-completions.js") + +/* + Looks up remote packages for CLI tab-completion. + + NOTE: If doVersion is true, versions in the form @ + will be completed. + + If doTag is true, tags in the form @ will be + completed. + + If recurring in true, sequences of multiple packages can be + completed. i.e. for schemes such as: + npm [@ [[@] ...] +*/ +function remotePackages (args, index, doVersion, doTag + , recurring, cb) { + if (recurring || index < 3) { + var name = (args.length + 1 === index) ? args[args.length - 1] : "" + if (name === undefined) name = "" + if (name.indexOf("/") !== -1) return cb(null, []) + // use up-to 1 hour stale cache. not super urgent. + registry.get("/", null, 3600, function (er, d) { + if (er) return cb(er) + var remoteList = Object.keys(d) + , found = remoteList.indexOf(name) + , unique = found && containsSingleMatch(name, remoteList) + , simpleMatches = getCompletions(name, remoteList) + , uniqueMatch = unique && simpleMatches[0] + , addTag = doTag && (unique || found || name.indexOf("@") !== -1) + , addVer = doVersion && (unique || found || name.indexOf("@") !== -1) + , list = [] + , pieces = (uniqueMatch || name).split("@") + , pkgname = pieces[0] + , extras = [] + if (unique && !addTag && !addVer) return cb(null, [uniqueMatch]) + if (d[pkgname] && (addTag || addVer)) { + if (d[pkgname].versions && addVer) { + extras = extras.concat(Object.keys(d[pkgname].versions)) + } + if (d[pkgname]["dist-tags"] && addTag) { + extras = extras.concat(Object.keys(d[pkgname]["dist-tags"])) + } + list = getCompletions(name, list.concat(extras.map(function (e) { + return pkgname + "@" + e + }))) + } + if (!unique) list = list.concat(getCompletions(name, remoteList)) + return cb(null, list) + }) + } +} diff --git a/deps/npm/lib/utils/completion/users.js b/deps/npm/lib/utils/completion/users.js new file mode 100644 index 0000000000..f77312c94e --- /dev/null +++ b/deps/npm/lib/utils/completion/users.js @@ -0,0 +1,22 @@ + +module.exports = users + +var registry = require("../npm-registry-client/index.js") + , containsSingleMatch = require("./contains-single-match.js") + , getCompletions = require("./get-completions.js") + , log = require("../log.js") + +function users (args, index, cb) { + var name = (args.length + 1 === index) ? args[args.length - 1] : "" + if (name === undefined) name = "" + // use up-to 1 day stale cache. doesn't change much + log.warn("About to fetch", "users completion") + registry.get("/-/users", null, 24*60*60, function (er, d) { + log.warn(d, "userdata") + log.warn(name, "name") + if (er) return cb(er) + var remoteList = Object.keys(d) + , simpleMatches = getCompletions(name, remoteList) + return cb(null, simpleMatches) + }) +} diff --git a/deps/npm/lib/utils/config-defs.js b/deps/npm/lib/utils/config-defs.js new file mode 100644 index 0000000000..55d5c887de --- /dev/null +++ b/deps/npm/lib/utils/config-defs.js @@ -0,0 +1,299 @@ +// defaults, types, and shorthands. + + +var path = require("path") + , url = require("url") + , Stream = require("stream").Stream + , semver = require("semver") + , stableFamily = semver.parse(process.version) + , os = require("os") + , nopt = require("nopt") + , log = require("./log.js") + +function Octal () {} +function validateOctal (data, k, val) { + // must be either an integer or an octal string. + if (typeof val === "number") { + data[k] = "0" + val.toString(8) + } + if (typeof val === "string") { + if (val.charAt(0) !== "0" || isNaN(val)) return false + data[k] = "0" + parseInt(val, 8).toString(8) + } +} + +function validateSemver (data, k, val) { + if (!semver.valid(val)) return false + data[k] = semver.valid(val) +} + +nopt.typeDefs.semver = { type: semver, validate: validateSemver } +nopt.typeDefs.Octal = { type: Octal, validate: validateOctal } + +nopt.invalidHandler = function (k, val, type, data) { + log.warn(k + "=" + JSON.stringify(val), "invalid config") + + if (Array.isArray(type)) { + if (type.indexOf(url) !== -1) type = url + else if (type.indexOf(path) !== -1) type = path + } + + switch (type) { + case Octal: + log.warn("Must be octal number, starting with 0", "invalid config") + break + case url: + log.warn("Must be a full url with 'http://'", "invalid config") + break + case path: + log.warn("Must be a valid filesystem path", "invalid config") + break + case Number: + log.warn("Must be a numeric value", "invalid config") + break + } +} + +if (!stableFamily || (+stableFamily[2] % 2)) stableFamily = null +else stableFamily = stableFamily[1] + "." + stableFamily[2] + +var httpsOk = semver.satisfies(process.version, ">=0.4.9") +var winColor = semver.satisfies(process.version, ">=0.5.9") + +var defaults + +var temp = process.env.TMPDIR + || process.env.TMP + || process.env.TEMP + || ( process.platform === "win32" + ? "c:\\windows\\temp" + : "/tmp" ) + +var home = ( process.platform === "win32" + ? process.env.USERPROFILE + : process.env.HOME ) || temp + +var globalPrefix +Object.defineProperty(exports, "defaults", {get: function () { + if (defaults) return defaults + + if (process.env.PREFIX) { + globalPrefix = process.env.PREFIX + } else if (process.platform === "win32") { + // c:\node\node.exe --> prefix=c:\node\ + globalPrefix = path.join(process.execPath, "..") + } else { + // /usr/local/bin/node --> prefix=/usr/local + globalPrefix = path.join(process.execPath, "..", "..") + + // destdir only is respected on Unix + if (process.env.DESTDIR) { + globalPrefix = process.env.DESTDIR + "/" + globalPrefix + } + } + + return defaults = + { "always-auth" : false + + // Disable bindist publishing for now. Too problematic. + // Revisit when we have a less crappy approach, or just make + // bindist be a thing that only dedicated build-farms will enable. + , "bin-publish" : false + + , bindist : stableFamily + && ( stableFamily + "-" + + "ares" + process.versions.ares + "-" + + "ev" + process.versions.ev + "-" + + "openssl" + process.versions.openssl + "-" + + "v8" + process.versions.v8 + "-" + + process.platform + "-" + + (process.arch ? process.arch + "-" : "") + + os.release() ) + + // are there others? + , browser : process.platform === "darwin" ? "open" + : process.platform === "win32" ? "start" + : "google-chrome" + + , ca : // the npm CA certificate. + "-----BEGIN CERTIFICATE-----\n"+ + "MIIChzCCAfACCQDauvz/KHp8ejANBgkqhkiG9w0BAQUFADCBhzELMAkGA1UEBhMC\n"+ + "VVMxCzAJBgNVBAgTAkNBMRAwDgYDVQQHEwdPYWtsYW5kMQwwCgYDVQQKEwNucG0x\n"+ + "IjAgBgNVBAsTGW5wbSBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkxDjAMBgNVBAMTBW5w\n"+ + "bUNBMRcwFQYJKoZIhvcNAQkBFghpQGl6cy5tZTAeFw0xMTA5MDUwMTQ3MTdaFw0y\n"+ + "MTA5MDIwMTQ3MTdaMIGHMQswCQYDVQQGEwJVUzELMAkGA1UECBMCQ0ExEDAOBgNV\n"+ + "BAcTB09ha2xhbmQxDDAKBgNVBAoTA25wbTEiMCAGA1UECxMZbnBtIENlcnRpZmlj\n"+ + "YXRlIEF1dGhvcml0eTEOMAwGA1UEAxMFbnBtQ0ExFzAVBgkqhkiG9w0BCQEWCGlA\n"+ + "aXpzLm1lMIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDLI4tIqPpRW+ACw9GE\n"+ + "OgBlJZwK5f8nnKCLK629Pv5yJpQKs3DENExAyOgDcyaF0HD0zk8zTp+ZsLaNdKOz\n"+ + "Gn2U181KGprGKAXP6DU6ByOJDWmTlY6+Ad1laYT0m64fERSpHw/hjD3D+iX4aMOl\n"+ + "y0HdbT5m1ZGh6SJz3ZqxavhHLQIDAQABMA0GCSqGSIb3DQEBBQUAA4GBAC4ySDbC\n"+ + "l7W1WpLmtLGEQ/yuMLUf6Jy/vr+CRp4h+UzL+IQpCv8FfxsYE7dhf/bmWTEupBkv\n"+ + "yNL18lipt2jSvR3v6oAHAReotvdjqhxddpe5Holns6EQd1/xEZ7sB1YhQKJtvUrl\n"+ + "ZNufy1Jf1r0ldEGeA+0ISck7s+xSh9rQD2Op\n"+ + "-----END CERTIFICATE-----\n" + + , cache : path.resolve( home || temp + , process.platform === "win32" + ? "npm-cache" : ".npm") + , color : process.platform !== "win32" || winColor + , depth: Infinity + , description : true + , dev : false + , editor : process.env.EDITOR || + ( process.platform === "win32" ? "notepad" : "vi" ) + , force : false + , global : false + , globalconfig : path.resolve(process.execPath, "..", "..", "etc", "npmrc") + , globalignorefile : path.resolve( process.execPath + , "..", "..", "etc", "npmignore") + , group : process.platform === "win32" ? 0 + : process.env.SUDO_GID || (process.getgid && process.getgid()) + , ignore: "" + , "init.version" : "0.0.0" + , "init.author.name" : "" + , "init.author.email" : "" + , "init.author.url" : "" + , link: false + , logfd : 2 + , loglevel : "warn" + , logprefix : process.platform !== "win32" || winColor + , long : false + , message : "%s" + , "node-version" : process.version + , npaturl : "http://npat.npmjs.org/" + , npat : false + , "onload-script" : false + , outfd : 1 + , parseable : false + , pre: false + , prefix : globalPrefix + , production: false + , proxy : process.env.HTTP_PROXY || process.env.http_proxy || null + , "https-proxy" : process.env.HTTPS_PROXY || process.env.https_proxy || + process.env.HTTP_PROXY || process.env.http_proxy || null + , "rebuild-bundle" : true + , registry : "http" + (httpsOk ? "s" : "") + "://registry.npmjs.org/" + , rollback : true + , save : false + , searchopts: "" + , searchexclude: null + , shell : process.env.platform === "win32" + ? process.env.ComSpec || "cmd" + : process.env.SHELL || "bash" + , "strict-ssl": true + , tag : "latest" + , tmp : temp + , unicode : true + , "unsafe-perm" : process.platform === "win32" + || process.platform === "cygwin" + || !( process.getuid && process.setuid + && process.getgid && process.setgid ) + || process.getuid() !== 0 + , usage : false + , user : process.platform === "win32" ? 0 : "nobody" + , username : "" + , userconfig : path.resolve(home, ".npmrc") + , userignorefile : path.resolve(home, ".npmignore") + , umask: 022 + , version : false + , viewer: process.platform === "win32" ? "browser" : "man" + , yes: null + + , _exit : true + } +}}) + +exports.types = + { "always-auth" : Boolean + , "bin-publish" : Boolean + , bindist : [null, String] + , browser : String + , ca: [null, String] + , cache : path + , color : ["always", Boolean] + , depth : Number + , description : Boolean + , dev : Boolean + , editor : String + , force : Boolean + , global : Boolean + , globalconfig : path + , globalignorefile: path + , group : [Number, String] + , "https-proxy" : [null, url] + , ignore : String + , "init.version" : [null, semver] + , "init.author.name" : String + , "init.author.email" : String + , "init.author.url" : ["", url] + , link: Boolean + , logfd : [Number, Stream] + , loglevel : ["silent","win","error","warn","info","verbose","silly"] + , logprefix : Boolean + , long : Boolean + , message: String + , "node-version" : [null, semver] + , npaturl : url + , npat : Boolean + , "onload-script" : [null, String] + , outfd : [Number, Stream] + , parseable : Boolean + , pre: Boolean + , prefix: path + , production: Boolean + , proxy : [null, url] + , "rebuild-bundle" : Boolean + , registry : [null, url] + , rollback : Boolean + , save : Boolean + , searchopts : String + , searchexclude: [null, String] + , shell : String + , "strict-ssl": Boolean + , tag : String + , tmp : path + , unicode : Boolean + , "unsafe-perm" : Boolean + , usage : Boolean + , user : [Number, String] + , username : String + , userconfig : path + , userignorefile : path + , umask: Octal + , version : Boolean + , viewer: String + , yes: [false, null, Boolean] + , _exit : Boolean + } + +exports.shorthands = + { s : ["--loglevel", "silent"] + , d : ["--loglevel", "info"] + , dd : ["--loglevel", "verbose"] + , ddd : ["--loglevel", "silly"] + , noreg : ["--no-registry"] + , reg : ["--registry"] + , "no-reg" : ["--no-registry"] + , silent : ["--loglevel", "silent"] + , verbose : ["--loglevel", "verbose"] + , h : ["--usage"] + , H : ["--usage"] + , "?" : ["--usage"] + , help : ["--usage"] + , v : ["--version"] + , f : ["--force"] + , gangster : ["--force"] + , gangsta : ["--force"] + , desc : ["--description"] + , "no-desc" : ["--no-description"] + , "local" : ["--no-global"] + , l : ["--long"] + , m : ["--message"] + , p : ["--parseable"] + , porcelain : ["--parseable"] + , g : ["--global"] + , S : ["--save"] + , y : ["--yes"] + , n : ["--no-yes"] + } diff --git a/deps/npm/lib/utils/error-handler.js b/deps/npm/lib/utils/error-handler.js new file mode 100644 index 0000000000..af52e61697 --- /dev/null +++ b/deps/npm/lib/utils/error-handler.js @@ -0,0 +1,264 @@ + +module.exports = errorHandler + +var cbCalled = false + , log = require("./log.js") + , npm = require("../npm.js") + , rm = require("rimraf") + , constants = require("constants") + , itWorked = false + , path = require("path") + , ini = require("./ini.js") + , wroteLogFile = false + + +process.on("exit", function (code) { + // console.error("exit", code) + if (!ini.resolved) return + if (code) itWorked = false + if (itWorked) log("ok") + else { + if (!cbCalled) { + log.error("cb() never called!\n ") + } + if (wroteLogFile) { + log.error(["" + ,"Additional logging details can be found in:" + ," " + path.resolve("npm-debug.log") + ].join("\n")) + wroteLogFile = false + } + log.win("not ok") + } + itWorked = false // ready for next exit +}) + +function errorHandler (er) { + // console.error("errorHandler", er) + if (!ini.resolved) { + // logging won't work unless we pretend that it's ready + er = er || new Error("Exit prior to config file resolving.") + console.error(er.stack || er.message) + } + + if (cbCalled) { + er = er || new Error("Callback called more than once.") + } + + cbCalled = true + if (!er) return exit(0) + if (!(er instanceof Error)) { + log.error(er) + return exit(1, true) + } + + var m = er.code || er.message.match(/^(?:Error: )?(E[A-Z]+)/) + if (m) { + m = m[1] + if (!constants[m] && !npm[m]) constants[m] = {} + er.errno = npm[m] || constants[m] + } + + switch (er.errno) { + case "ECONNREFUSED": + case constants.ECONNREFUSED: + log.error(er) + log.error(["If you are using Cygwin, please set up your /etc/resolv.conf" + ,"See step 4 in this wiki page:" + ," http://github.com/ry/node/wiki/Building-node.js-on-Cygwin-%28Windows%29" + ,"If you are not using Cygwin, please report this" + ,"at " + ,"or email it to " + ].join("\n")) + break + + case "EACCES": + case "EPERM": + case constants.EACCES: + case constants.EPERM: + log.error(er) + log.error(["", + "Please use 'sudo' or log in as root to run this command." + ,"" + ," sudo npm " + +npm.config.get("argv").original.map(JSON.stringify).join(" ") + ,"" + ,"or set the 'unsafe-perm' config var to true." + ,"" + ," npm config set unsafe-perm true" + ].join("\n")) + break + + case npm.ELIFECYCLE: + er.code = "ELIFECYCLE" + log.error(er.message) + log.error(["","Failed at the "+er.pkgid+" "+er.stage+" script." + ,"This is most likely a problem with the "+er.pkgname+" package," + ,"not with npm itself." + ,"Tell the author that this fails on your system:" + ," "+er.script + ,"You can get their info via:" + ," npm owner ls "+er.pkgname + ,"There is likely additional logging output above." + ].join("\n")) + break + + case npm.EJSONPARSE: + er.code = "EJSONPARSE" + log.error(er.message) + log.error("File: "+er.file) + log.error(["Failed to parse package.json data." + ,"package.json must be actual JSON, not just JavaScript." + ,"","This is not a bug in npm." + ,"Tell the package author to fix their package.json file." + ].join("\n"), "JSON.parse") + break + + case npm.E404: + er.code = "E404" + if (er.pkgid && er.pkgid !== "-") { + var msg = ["'"+er.pkgid+"' is not in the npm registry." + ,"You could maybe bug the author to publish it"] + if (er.pkgid.match(/^node[\.\-]|[\.\-]js$/)) { + var s = er.pkgid.replace(/^node[\.\-]|[\.\-]js$/g, "") + if (s !== er.pkgid) { + s = s.replace(/[^a-z0-9]/g, ' ') + msg.push("Maybe try 'npm search " + s + "'") + } + } + msg.push("Note that you can also install from a tarball or folder.") + log.error(msg.join("\n"), "404") + } + break + + case npm.EPUBLISHCONFLICT: + er.code = "EPUBLISHCONFLICT" + log.error(["Cannot publish over existing version." + ,"Bump the 'version' field, set the --force flag, or" + ," npm unpublish '"+er.pkgid+"'" + ,"and try again" + ].join("\n"), "publish fail" ) + break + + case npm.EISGIT: + er.code = "EISGIT" + log.error([er.message + ," "+er.path + ,"Refusing to remove it. Update manually," + ,"or move it out of the way first." + ].join("\n"), "git" ) + break + + case npm.ECYCLE: + er.code = "ECYCLE" + log.error([er.message + ,"While installing: "+er.pkgid + ,"Found a pathological dependency case that npm cannot solve." + ,"Please report this to the package author." + ].join("\n")) + break + + case npm.ENOTSUP: + er.code = "ENOTSUP" + log.error([er.message + ,"Not compatible with your version of node/npm: "+er.pkgid + ,"Required: "+JSON.stringify(er.required) + ,"Actual: " + +JSON.stringify({npm:npm.version + ,node:npm.config.get("node-version")}) + ].join("\n")) + break + + case "EEXIST": + case constants.EEXIST: + log.error([er.message + ,"File exists: "+er.path + ,"Move it away, and try again."].join("\n")) + break + + default: + log.error(er) + log.error(["Report this *entire* log at:" + ," " + ,"or email it to:" + ," " + ].join("\n")) + break + } + + var os = require("os") + log.error("") + log.error(os.type() + " " + os.release(), "System") + log.error(process.argv + .map(JSON.stringify).join(" "), "command") + log.error(process.cwd(), "cwd") + log.error(process.version, "node -v") + log.error(npm.version, "npm -v") + + ; [ "file" + , "path" + , "type" + , "syscall" + , "fstream_path" + , "fstream_unc_path" + , "fstream_type" + , "fstream_class" + , "fstream_finish_call" + , "fstream_linkpath" + , "arguments" + , "code" + , "message" + ].forEach(function (k) { + if (er[k]) log.error(er[k], k) + }) + + if (er.fstream_stack) { + log.error(er.fstream_stack.join("\n"), "fstream_stack") + } + + if (er.errno && typeof er.errno !== "object") log.error(er.errno, "errno") + exit(typeof er.errno === "number" ? er.errno : 1) +} + +function exit (code, noLog) { + var doExit = npm.config.get("_exit") + log.verbose([code, doExit], "exit") + if (log.level === log.LEVEL.silent) noLog = true + + if (code && !noLog) writeLogFile(reallyExit) + else rm("npm-debug.log", function () { rm(npm.tmp, reallyExit) }) + + function reallyExit() { + itWorked = !code + //if (!itWorked) { + if (!doExit) process.emit("exit", code) + else process.exit(code) + //} + } +} + +var writingLogFile = false +function writeLogFile (cb) { + if (writingLogFile) return cb() + writingLogFile = true + wroteLogFile = true + + var fs = require("graceful-fs") + , fstr = fs.createWriteStream("npm-debug.log") + , util = require("util") + + log.history.forEach(function (m) { + var lvl = log.LEVEL[m.level] + , pref = m.pref ? " " + m.pref : "" + , b = lvl + pref + " " + , eol = process.platform === "win32" ? "\r\n" : "\n" + , msg = typeof m.msg === "string" ? m.msg + : msg instanceof Error ? msg.stack || msg.message + : util.inspect(m.msg, 0, 4) + fstr.write(new Buffer(b + +(msg.split(/\r?\n+/).join(eol+b)) + + eol)) + }) + fstr.end() + fstr.on("close", cb) +} diff --git a/deps/npm/lib/utils/excludes.js b/deps/npm/lib/utils/excludes.js new file mode 100644 index 0000000000..6df89ccaf4 --- /dev/null +++ b/deps/npm/lib/utils/excludes.js @@ -0,0 +1,145 @@ +// build up a set of exclude lists in order of precedence: +// [ ["!foo", "bar"] +// , ["foo", "!bar"] ] +// being *included* will override a previous exclusion, +// and being excluded will override a previous inclusion. +// +// Each time the tar file-list generator thingie enters a new directory, +// it calls "addIgnoreFile(dir, list, cb)". If an ignore file is found, +// then it is added to the list and the cb() is called with an +// child of the original list, so that we don't have +// to worry about popping it off at the right time, since other +// directories will continue to use the original parent list. +// +// If no ignore file is found, then the original list is returned. +// +// To start off with, ~/.{npm,git}ignore is added, as is +// prefix/{npm,git}ignore, effectively treated as if they were in the +// base package directory. + +exports.addIgnoreFile = addIgnoreFile +exports.readIgnoreFile = readIgnoreFile +exports.parseIgnoreFile = parseIgnoreFile +exports.test = test +exports.filter = filter + +var path = require("path") + , fs = require("graceful-fs") + , minimatch = require("minimatch") + , relativize = require("./relativize.js") + , log = require("./log.js") + +// todo: memoize + +// read an ignore file, or fall back to the +// "gitBase" file in the same directory. +function readIgnoreFile (file, gitBase, cb) { + //log.warn(file, "ignoreFile") + if (!file) return cb(null, "") + fs.readFile(file, function (er, data) { + if (!er || !gitBase) return cb(null, data || "") + var gitFile = path.resolve(path.dirname(file), gitBase) + fs.readFile(gitFile, function (er, data) { + return cb(null, data || "") + }) + }) +} + +// read a file, and then return the list of patterns +function parseIgnoreFile (file, gitBase, dir, cb) { + readIgnoreFile(file, gitBase, function (er, data) { + data = data ? data.toString("utf8") : "" + + data = data.split(/[\r\n]+/).map(function (p) { + return p.trim() + }).filter(function (p) { + return p.length && p.charAt(0) !== "#" + }) + data.dir = dir + return cb(er, data) + }) +} + +// add an ignore file to an existing list which can +// then be passed to the test() function. If the ignore +// file doesn't exist, then the list is unmodified. If +// it is, then a concat-child of the original is returned, +// so that this is suitable for walking a directory tree. +function addIgnoreFile (file, gitBase, list, dir, cb) { + if (typeof cb !== "function") cb = dir, dir = path.dirname(file) + if (typeof cb !== "function") cb = list, list = [] + parseIgnoreFile(file, gitBase, dir, function (er, data) { + if (!er && data) { + // package.json "files" array trumps everything + // Make sure it's always last. + if (list.length && list[list.length-1].packageFiles) { + list = list.concat([data, list.pop()]) + } else { + list = list.concat([data]) + } + } + cb(er, list) + }) +} + + +// no IO +// loop through the lists created in the functions above, and test to +// see if a file should be included or not, given those exclude lists. +function test (file, excludeList) { + if (path.basename(file) === "package.json") return true + //log.warn(file, "test file") + //log.warn(excludeList, "test list") + var incRe = /^\!(\!\!)*/ + , excluded = false + for (var i = 0, l = excludeList.length; i < l; i ++) { + var excludes = excludeList[i] + , dir = excludes.dir + + // chop the filename down to be relative to excludeDir + var rf = relativize(file, dir, true) + rf = rf.replace(/^\.\//, "") + + for (var ii = 0, ll = excludes.length; ii < ll; ii ++) { + //log.warn(JSON.stringify(excludes[ii]), "ex") + var ex = excludes[ii].replace(/^(!*)\.\//, "$1") + , inc = !!ex.match(incRe) + + // excluding/including a dir excludes/includes all the files in it. + if (ex.slice(-1) === "/") ex += "**" + + // if this is not an inclusion attempt, and someone else + // excluded it, then just continue, because there's nothing + // that can be done here to change the exclusion. + if (!inc && excluded) continue + + // if it's an inclusion attempt, and the file has not been + // excluded, then skip it, because there's no need to try again. + if (inc && !excluded) continue + + // if it matches the pattern, then it should be excluded. + excluded = !!minimatch(rf, ex, { baseMatch: true }) + //if (inc) excluded = !excluded + + //if (excluded) { + // console.error("excluded %s %s", rf, ex) + //} + + // if you include foo, then it also includes foo/bar.js + if (inc && excluded && ex.slice(-3) !== "/**") { + excluded = minimatch(rf, ex + "/**", { baseMatch: true }) + // console.error(rf, ex + "/**", inc, excluded) + } + } + //log.warn([rf, excluded, excludes], "file, excluded, excludes") + } + // true if it *should* be included + // log.warn([file, excludeList, excluded], "file, excluded") + return !excluded +} + +// returns a function suitable for Array#filter +function filter (dir, list) { return function (file) { + file = file.trim() + return file && test(path.resolve(dir, file), list) +}} diff --git a/deps/npm/lib/utils/exec.js b/deps/npm/lib/utils/exec.js new file mode 100644 index 0000000000..0fa0371b2e --- /dev/null +++ b/deps/npm/lib/utils/exec.js @@ -0,0 +1,106 @@ + +module.exports = exec +exec.spawn = spawn +exec.pipe = pipe + +var log = require("./log.js") + , child_process = require("child_process") + , util = require("util") + , npm = require("../npm.js") + , myUID = process.getuid ? process.getuid() : null + , myGID = process.getgid ? process.getgid() : null + , isRoot = process.getuid && myUID === 0 + , constants = require("constants") + +function exec (cmd, args, env, takeOver, cwd, uid, gid, cb) { + if (typeof cb !== "function") cb = gid, gid = null + if (typeof cb !== "function") cb = uid, uid = null + if (typeof cb !== "function") cb = cwd, cwd = null + if (typeof cb !== "function") cb = takeOver, takeOver = true + if (typeof cb !== "function") cb = env, env = process.env + gid = gid == null ? myGID : gid + uid = uid == null ? myUID : uid + if (!isRoot) { + if (npm.config.get("unsafe-perm")) { + uid = myUID + gid = myGID + } else if (uid !== myUID || gid !== myGID) { + var e = new Error("EPERM: setuid() operation not permitted") + e.errno = constants.EPERM + return cb(e) + } + } + if (uid !== myUID) { + log.verbose(uid, "Setting uid from "+myUID) + log.verbose(new Error().stack, "stack at uid setting") + } + log.silly(cmd+" "+args.map(JSON.stringify).join(" "), "exec") + var stdout = "" + , stderr = "" + , cp = spawn(cmd, args, env, takeOver, cwd, uid, gid) + cp.stdout && cp.stdout.on("data", function (chunk) { + if (chunk) stdout += chunk + }) + cp.stderr && cp.stderr.on("data", function (chunk) { + if (chunk) stderr += chunk + }) + cp.on("exit", function (code) { + var er = null + if (code) er = new Error("`"+cmd + +(args.length ? " " + + args.map(JSON.stringify).join(" ") + : "") + +"` failed with "+code) + cb(er, code, stdout, stderr) + }) + return cp +} + +function logger (d) { if (d) process.stderr.write(d+"") } +function pipe (cp1, cp2, cb) { + util.pump(cp1.stdout, cp2.stdin) + var errState = null + , buff1 = "" + , buff2 = "" + if (log.level <= log.LEVEL.silly) { + cp1.stderr.on("data", logger) + cp2.stderr.on("data", logger) + } else { + cp1.stderr.on("data", function (d) { buff1 += d }) + cp2.stderr.on("data", function (d) { buff2 += d }) + } + + cp1.on("exit", function (code) { + if (!code) return log.verbose(cp1.name || "", "success") + if (!cp2._exited) cp2.kill() + log.error(buff1, cp1.name || "") + cb(errState = new Error( + "Failed "+(cp1.name || "")+"\nexited with "+code)) + }) + cp2.on("exit", function (code) { + cp2._exited = true + if (errState) return + if (!code) return log.verbose(cp2.name || "", "success", cb) + log.error(buff2, cp2.name || "") + cb(new Error( "Failed " + + (cp2.name || "") + + "\nexited with " + code )) + }) +} + +function spawn (c, a, env, takeOver, cwd, uid, gid) { + var fds = [ 0, 1, 2 ] + , opts = { customFds : takeOver ? fds : [-1,-1,-1] + , env : env || process.env + , cwd : cwd || null } + , cp + if (uid != null) opts.uid = uid + if (gid != null) opts.gid = gid + if (!isNaN(opts.uid)) opts.uid = +opts.uid + if (!isNaN(opts.gid)) opts.gid = +opts.gid + var name = c +" "+ a.map(JSON.stringify).join(" ") + log.silly([c, a, opts.cwd], "spawning") + cp = child_process.spawn(c, a, opts) + cp.name = name + return cp +} diff --git a/deps/npm/lib/utils/fetch.js b/deps/npm/lib/utils/fetch.js new file mode 100644 index 0000000000..726be8848b --- /dev/null +++ b/deps/npm/lib/utils/fetch.js @@ -0,0 +1,62 @@ +/** + * Fetch an HTTP url to a local file. + **/ + +var request = require("request") + , fs = require("graceful-fs") + , npm = require("../npm.js") + , url = require("url") + , log = require("./log.js") + , path = require("path") + , mkdir = require("./mkdir-p.js") + , regHost + , getAgent = require("./get-agent.js") + +module.exports = fetch + +function fetch (remote, local, headers, cb) { + if (typeof cb !== "function") cb = headers, headers = {} + log.info(remote, "fetch") + log.verbose(local, "fetch to") + mkdir(path.dirname(local), function (er) { + if (er) return cb(er) + fetch_(remote, local, headers, cb) + }) +} + +function fetch_ (remote, local, headers, cb) { + var fstr = fs.createWriteStream(local, { mode : npm.modes.file }) + fstr.on("error", function (er) { + fs.close(fstr.fd, function () {}) + if (fstr._ERROR) return + cb(fstr._ERROR = er) + }) + fstr.on("open", function () { + makeRequest(remote, fstr, headers) + }) + fstr.on("close", function () { + if (fstr._ERROR) return + cb() + }) +} + +function makeRequest (remote, fstr, headers) { + remote = url.parse(remote) + regHost = regHost || url.parse(npm.config.get("registry")).host + + if (remote.host === regHost && npm.config.get("always-auth")) { + remote.auth = new Buffer( npm.config.get("_auth") + , "base64" ).toString("utf8") + if (!remote.auth) return fstr.emit("error", new Error( + "Auth required and none provided. Please run 'npm adduser'")) + } + + var proxy = npm.config.get( remote.protocol === "https:" + ? "https-proxy" + : "proxy") + + request({ url: remote + , proxy: proxy + , agent: getAgent(remote) + , strictSSL: npm.config.get("strict-ssl") }).pipe(fstr) +} diff --git a/deps/npm/lib/utils/find-prefix.js b/deps/npm/lib/utils/find-prefix.js new file mode 100644 index 0000000000..320456c7ea --- /dev/null +++ b/deps/npm/lib/utils/find-prefix.js @@ -0,0 +1,50 @@ +// try to find the most reasonable prefix to use + +module.exports = findPrefix + +var fs = require("graceful-fs") + , path = require("path") + , npm = require("../npm.js") + +function findPrefix (p, cb_) { + function cb (er, p) { + process.nextTick(function () { + cb_(er, p) + }) + } + + p = path.resolve(p) + if (npm.config.get("global")) return cb(null, p) + // if there's no node_modules folder, then + // walk up until we hopefully find one. + // if none anywhere, then use cwd. + var walkedUp = false + while (path.basename(p) === "node_modules") { + p = path.dirname(p) + walkedUp = true + } + if (walkedUp) return cb(null, p) + + findPrefix_(p, p, cb) +} + +function findPrefix_ (p, original, cb) { + if (p === "/" + || (process.platform === "win32" && p.match(/^[a-zA-Z]:(\\|\/)?$/))) { + return cb(null, original) + } + fs.readdir(p, function (er, files) { + // an error right away is a bad sign. + if (er && p === original) return cb(er) + + // walked up too high or something. + if (er) return cb(null, original) + + if (files.indexOf("node_modules") !== -1 + || files.indexOf("package.json") !== -1) { + return cb(null, p) + } + + return findPrefix_(path.dirname(p), original, cb) + }) +} diff --git a/deps/npm/lib/utils/find.js b/deps/npm/lib/utils/find.js new file mode 100644 index 0000000000..89d1c2bc79 --- /dev/null +++ b/deps/npm/lib/utils/find.js @@ -0,0 +1,53 @@ + +// walks a set of directories recursively, and returns +// the list of files that match the filter, if one is +// provided. + +module.exports = find +var fs = require("graceful-fs") + , asyncMap = require("slide").asyncMap + , path = require("path") + +function find (dir, filter, depth, cb) { + if (typeof cb !== "function") cb = depth, depth = Infinity + if (typeof cb !== "function") cb = filter, filter = null + if (filter instanceof RegExp) filter = reFilter(filter) + if (typeof filter === "string") filter = strFilter(filter) + if (!Array.isArray(dir)) dir = [dir] + if (!filter) filter = nullFilter + asyncMap(dir, findDir(filter, depth), cb) +} +function findDir (filter, depth) { return function (dir, cb) { + fs.lstat(dir, function (er, stats) { + // don't include missing files, but don't abort either + if (er) return cb() + if (!stats.isDirectory()) return findFile(dir, filter, depth)("", cb) + var found = [] + if (!filter || filter(dir, "dir")) found.push(dir+"/") + if (depth <= 0) return cb(null, found) + cb = (function (cb) { return function (er, f) { + cb(er, found.concat(f)) + }})(cb) + fs.readdir(dir, function (er, files) { + if (er) return cb(er) + asyncMap(files, findFile(dir, filter, depth - 1), cb) + }) + }) +}} +function findFile (dir, filter, depth) { return function (f, cb) { + f = path.join(dir, f) + fs.lstat(f, function (er, s) { + // don't include missing files, but don't abort either + if (er) return cb(null, []) + if (s.isDirectory()) return find(f, filter, depth, cb) + if (!filter || filter(f, "file")) cb(null, f) + else cb(null, []) + }) +}} +function reFilter (re) { return function (f, type) { + return nullFilter(f, type) && f.match(re) +}} +function strFilter (s) { return function (f, type) { + return nullFilter(f, type) && f.indexOf(s) === 0 +}} +function nullFilter (f, type) { return type === "file" && f } diff --git a/deps/npm/lib/utils/get-agent.js b/deps/npm/lib/utils/get-agent.js new file mode 100644 index 0000000000..b4a5738d2d --- /dev/null +++ b/deps/npm/lib/utils/get-agent.js @@ -0,0 +1,62 @@ +// get an http/https agent +// This is necessary for the custom CA certs in http2, +// especially while juggling multiple different registries. +// +// When using http2, the agent key is just the CA setting, +// since it can manage socket pooling across different host:port +// options. When using the older implementation, the +// key is ca:host:port combination. + +module.exports = getAgent + +var npm = require("../npm.js") + , url = require("url") + , agents = {} + , isHttp2 = !!require("http").globalAgent + , registry = url.parse(npm.config.get("registry")) + , regCA = npm.config.get("ca") + +function getAgent (remote) { + // If not doing https, then there's no CA cert to manage. + // on http2, this will use the default global agent. + // on http1, this is undefined, so it'll spawn based on + // host:port if necessary. + if (remote.protocol !== "https:") { + return require("http").globalAgent + } + + if (typeof remote === "string") { + remote = url.parse(remote) + } + + var ca + // if this is the registry, then use the configuration ca. + // otherwise, just use the built-in CAs that node has. + // todo: multi-registry support. + if (remote.hostname === registry.hostname + && remote.port === registry.port) { + ca = regCA + } + + // no CA, just use the default agent. + if (!ca) { + return require("https").globalAgent + } + + var hostname = remote.hostname + , port = remote.port + , key = agentKey(hostname, port, ca) + + return agents[key] = agents[key] || getAgent_(hostname, port, ca) +} + +function getAgent_ (hostname, port, ca) { + var Agent = require("https").Agent + return new Agent({ host: hostname + , port: port + , ca: ca }) +} + +function agentKey (hostname, port, ca) { + return JSON.stringify(isHttp2 ? ca : [hostname, port, ca]) +} diff --git a/deps/npm/lib/utils/get.js b/deps/npm/lib/utils/get.js new file mode 100644 index 0000000000..3c5e4f099b --- /dev/null +++ b/deps/npm/lib/utils/get.js @@ -0,0 +1,6 @@ + +module.exports = get +function get (obj, key) { + for (var i in obj) if (i.toLowerCase() === key.toLowerCase()) return obj[i] + return undefined +} diff --git a/deps/npm/lib/utils/ini.js b/deps/npm/lib/utils/ini.js new file mode 100644 index 0000000000..f31bdeb790 --- /dev/null +++ b/deps/npm/lib/utils/ini.js @@ -0,0 +1,333 @@ +// Create a chain of config objects, in this priority order: +// +// CLI - the --foo things in the command line. +// ENV - all the things starting with npm_config_ in the environment +// USER - $HOME/.npmrc +// GLOBAL - $PREFIX/etc/npmrc +// +// If the CLI or ENV specify a userconfig, then that file is used +// as the USER config. +// +// If the CLI or ENV specify a globalconfig, then that file is used +// as the GLOBAL config. +// +// export npm_config_userconfig=/some/other/file +// export npm_config_globalconfig=global +// +// For implementation reasons, "_" in env vars is turned into "-". So, +// export npm_config_node_version + +exports.resolveConfigs = resolveConfigs +exports.save = save +exports.saveConfig = saveConfig +exports.del = del +exports.get = get +exports.set = set +exports.unParseField = unParseField +exports.defaultConfig = null + +Object.defineProperty(exports, "keys", + { get : function () { return configList.keys }}) + +var fs = require("graceful-fs") + , path = require("path") + , nopt = require("nopt") + , ini = require("ini") + , ProtoList = require("proto-list") + + , log = require("./log.js") + , configDefs = require("./config-defs.js") + + , myUid = process.env.SUDO_UID !== undefined + ? process.env.SUDO_UID : (process.getuid && process.getuid()) + , myGid = process.env.SUDO_GID !== undefined + ? process.env.SUDO_GID : (process.getgid && process.getgid()) + , eol = process.platform === "win32" ? "\r\n" : "\n" + , privateKey = null + , defaultConfig + , configList = new ProtoList() + , types = configDefs.types + , TRANS = exports.TRANS = + { "default" : 5 + , "builtin": 4 + , "global" : 3 + , "user" : 2 + , "env" : 1 + , "cli" : 0 + } + +exports.configList = configList + +// just put this here for a moment, so that the logs +// in the config-loading phase don't cause it to blow up. +configList.push({loglevel:"warn"}) + +function resolveConfigs (cli, cb_) { + defaultConfig = defaultConfig || configDefs.defaults + exports.defaultConfig = defaultConfig + configList.pop() + configList.push(defaultConfig) + var cl = configList + , dc = cl.pop() + if (!cb_) cb_ = cli, cli = {} + + function cb (er) { + //console.error("resolving configs") + exports.resolved = true + cb_(er) + } + + cl.list.length = 0 + Object.keys(cli).forEach(function (k) { + cli[k] = parseField(cli[k], k) + }) + cl.push(cli) + cl.push(parseEnv(process.env)) + parseFile(cl.get("userconfig") || dc.userconfig, function (er, conf) { + if (er) return cb(er) + cl.push(conf) + parseFile( cl.get("globalconfig") || dc.globalconfig + , function (er, conf) { + if (er) return cb(er) + cl.push(conf) + // the builtin config file, for distros to use. + parseFile(path.resolve(__dirname, "../../npmrc"), function (er, conf) { + if (er) conf = {} + cl.push(conf) + cl.push(dc) + setUser(cl, dc, thenValidate(cl, cb)) + }) + }) + }) +} + +function thenValidate (cl, cb) { return function (er) { + if (er) return cb(er) + + // warn about invalid configs at every level. + cl.list.forEach(function (conf, level) { + // clean(data, types, typeDefs) + nopt.clean(conf, configDefs.types) + }) + + cb() +}} + +function setUser (cl, dc, cb) { + // If global, leave it as-is. + // If not global, then set the user to the owner of the prefix folder. + // Just set the default, so it can be overridden. + //console.error("setUser "+cl.get("global")+" "+ cb.toString()) + if (cl.get("global")) return cb() + if (process.env.SUDO_UID) { + //console.error("uid="+process.env.SUDO_UID) + dc.user = +(process.env.SUDO_UID) + return cb() + } + //console.error("prefix="+cl.get("prefix")) + fs.stat(path.resolve(cl.get("prefix")), function (er, st) { + if (er) { + return log.er(cb, "prefix directory not found")(er) + } + dc.user = st.uid + return cb() + }) +} + +function parseEnv (env) { + var conf = {} + Object.keys(env) + .filter(function (k) { return k.match(/^npm_config_[^_]/i) }) + .forEach(function (k) { + conf[k.replace(/^npm_config_/i, "") + .toLowerCase() + .replace(/_/g, "-")] = parseField(env[k], k) + }) + return conf +} + +function unParseField (f, k) { + // type can be an array or single thing. + var isPath = -1 !== [].concat(types[k]).indexOf(path) + if (isPath) { + if (typeof process.env.HOME !== 'undefined') { + if (process.env.HOME.substr(-1) === "/") { + process.env.HOME = process.env.HOME(0, process.env.HOME.length-1) + } + if (f.indexOf(process.env.HOME) === 0) { + f = "~"+f.substr(process.env.HOME.length) + } + } + } + return ini.safe(f) +} + +function parseField (f, k, emptyIsFalse) { + if (typeof f !== "string" && !(f instanceof String)) return f + // type can be an array or single thing. + var isPath = -1 !== [].concat(types[k]).indexOf(path) + , isBool = -1 !== [].concat(types[k]).indexOf(Boolean) + , isString = -1 !== [].concat(types[k]).indexOf(String) + f = ini.unsafe((""+f).trim()) + if (isBool && !isString && f === "") return f = true + switch (f) { + case "true": return true + case "false": return false + case "null": return null + case "undefined": return undefined + } + if (isPath) { + if (f.substr(0, 2) === "~/" && process.env.HOME) { + f = path.resolve(process.env.HOME, f.substr(2)) + } + f = path.resolve(f) + } + return f +} + +function parseFile (file, cb) { + if (!file) return cb(null, {}) + log.verbose(file, "config file") + fs.readFile(file, function (er, data) { + // treat all errors as just an empty file + if (er) return cb(null, {}) + var d = ini.parse(""+data) + , f = {} + Object.keys(d).forEach(function (k) { + f[k] = parseField(d[k], k) + }) + cb(null, parseAuth(f)) + }) +} + +function encryptAuth (config, cb) { + if (config.username && config._password) { + var b = new Buffer(config.username+":"+config._password) + config._auth = b.toString("base64") + } + delete config.username + delete config._password + return cb(null, config) +} + +function parseAuth (config) { + //console.error("parsing config %j", config) + if (!config._auth) return config + var b = new Buffer(config._auth, "base64") + , unpw = b.toString().split(":") + , un = unpw.shift() + , pw = unpw.join(":") + config.username = un = (config.username || un) + config._password = pw = (config._password || pw) + b = new Buffer(un + ":" + pw) + config._auth = b.toString("base64") + return config +} + +function save (which, cb) { + if (typeof which === "function") cb = which, which = null + if (!which) which = ["global", "user", "builtin"] + if (!Array.isArray(which)) which = [which] + var errState = null + , done = which.length + , failed = [] + which.forEach(function (c) { + saveConfig(c, function (er) { + if (errState) return + if (er) return cb(errState = er) + if (-- done === 0) return cb() + }) + }) +} + +function saveConfig (which, file, cb) { + if (typeof file === "function") cb = file, file = null + if (!file) { + switch (which) { + case "builtin": + file = path.resolve(__dirname, "../../npmrc") + break + case "global": + file = configList.get("globalconfig") + break + default: + file = configList.get("userconfig") + which = "user" + } + } + + saveConfigfile + ( file + , configList.list[TRANS[which]] + , which + , cb ) +} + +function saveConfigfile (file, config, which, cb) { + encryptAuth(config, function () { // ignore errors + var data = {} + Object.keys(config).forEach(function (k) { + data[k] = unParseField(config[k], k) + }) + data = ini.stringify(data) + return (data.trim()) + ? writeConfigfile(file, data, which, cb) + : rmConfigfile(file, cb) + }) +} +function writeConfigfile (configfile, data, which, cb) { + data = data.split(/\r*\n/).join(eol) + fs.writeFile + ( configfile, data, "utf8" + , function (er) { + if (er) log(er, "Failed saving "+configfile, cb) + else if (which) { + fs.chmod(configfile, which === "user" ? 0600 : 0644, function (e) { + if (e || which !== "user" || typeof myUid !== "number") { + return cb(e) + } + fs.chown(configfile, +myUid, +myGid, cb) + }) + } + else cb() + } + ) +} +function rmConfigfile (configfile, cb) { + fs.stat(configfile, function (e) { + if (e) return cb() + fs.unlink(configfile, function (er) { + if (er) log(er, "Couldn't remove "+configfile) + cb() + }) + }) +} +function snapshot (which) { + var x = (!which) ? configList.snapshot + : configList.list[TRANS[which]] ? configList.list[TRANS[which]] + : undefined + if (!x) return + Object.keys(x).forEach(function (k) { if (k.match(/^_/)) delete x[k] }) + return x +} +function get (key, which) { + return (!key) ? snapshot(which) + : (!which) ? configList.get(key) // resolved + : configList.list[TRANS[which]] ? configList.list[TRANS[which]][key] + : undefined +} +function del (key, which) { + if (!which) configList.list.forEach(function (l) { + delete l[key] + }) + else if (configList.list[TRANS[which]]) { + delete configList.list[TRANS[which]] + } +} +function set (key, value, which) { + which = which || "cli" + if (configList.length === 1) { + return new Error("trying to set before loading") + } + return configList.list[TRANS[which]][key] = value +} diff --git a/deps/npm/lib/utils/lifecycle.js b/deps/npm/lib/utils/lifecycle.js new file mode 100644 index 0000000000..878fc88cf3 --- /dev/null +++ b/deps/npm/lib/utils/lifecycle.js @@ -0,0 +1,278 @@ + +exports = module.exports = lifecycle +exports.cmd = cmd + +var log = require("./log.js") + , exec = require("./exec.js") + , npm = require("../npm.js") + , path = require("path") + , readJson = require("./read-json.js") + , fs = require("graceful-fs") + , chain = require("slide").chain + , constants = require("constants") + , output = require("./output.js") + +function lifecycle (pkg, stage, wd, unsafe, failOk, cb) { + if (typeof cb !== "function") cb = failOk, failOk = false + if (typeof cb !== "function") cb = unsafe, unsafe = false + if (typeof cb !== "function") cb = wd, wd = null + + while (pkg && pkg._data) pkg = pkg._data + if (!pkg) return cb(new Error("Invalid package data")) + + log(pkg._id, stage) + if (!pkg.scripts) pkg.scripts = {} + + validWd(wd || path.resolve(npm.dir, pkg.name), function (er, wd) { + if (er) return cb(er) + + unsafe = unsafe || npm.config.get("unsafe-perm") + + if ((wd.indexOf(npm.dir) !== 0 || path.basename(wd) !== pkg.name) + && !unsafe && pkg.scripts[stage]) { + log.warn(pkg._id+" "+pkg.scripts[stage], "skipping, cannot run in "+wd) + return cb() + } + + // set the env variables, then run scripts as a child process. + var env = makeEnv(pkg) + env.npm_lifecycle_event = stage + + // "nobody" typically doesn't have permission to write to /tmp + // even if it's never used, sh freaks out. + if (!npm.config.get("unsafe-perm")) env.TMPDIR = wd + + lifecycle_(pkg, stage, wd, env, unsafe, failOk, cb) + }) +} + +function checkForLink (pkg, cb) { + var f = path.join(npm.dir, pkg.name) + fs.lstat(f, function (er, s) { + cb(null, !(er || !s.isSymbolicLink())) + }) +} + +function lifecycle_ (pkg, stage, wd, env, unsafe, failOk, cb) { + var PATH = [] + , p = wd.split("node_modules") + , acc = path.resolve(p.shift()) + p.forEach(function (pp) { + PATH.unshift(path.join(acc, "node_modules", ".bin")) + acc = path.join(acc, "node_modules", pp) + }) + PATH.unshift(path.join(acc, "node_modules", ".bin")) + if (env.PATH) PATH.push(env.PATH) + env.PATH = PATH.join(process.platform === "win32" ? ";" : ":") + + var packageLifecycle = pkg.scripts && pkg.scripts.hasOwnProperty(stage) + + if (packageLifecycle) { + // define this here so it's available to all scripts. + env.npm_lifecycle_script = pkg.scripts[stage] + } + + if (failOk) { + cb = (function (cb_) { return function (er) { + if (er) log.warn(er.message, "continuing anyway") + cb_() + }})(cb) + } + + if (npm.config.get("force")) { + cb = (function (cb_) { return function (er) { + if (er) log(er, "forced, continuing") + cb_() + }})(cb) + } + + chain + ( [ packageLifecycle && [runPackageLifecycle, pkg, env, wd, unsafe] + , [runHookLifecycle, pkg, env, wd, unsafe] ] + , cb ) +} + +function validWd (d, cb) { + fs.stat(d, function (er, st) { + if (er || !st.isDirectory()) { + var p = path.dirname(d) + if (p === d) { + return cb(new Error("Could not find suitable wd")) + } + return validWd(p, cb) + } + return cb(null, d) + }) +} + +function runPackageLifecycle (pkg, env, wd, unsafe, cb) { + // run package lifecycle scripts in the package root, or the nearest parent. + var stage = env.npm_lifecycle_event + , user = unsafe ? null : npm.config.get("user") + , group = unsafe ? null : npm.config.get("group") + , cmd = env.npm_lifecycle_script + , sh = "sh" + , shFlag = "-c" + + if (process.platform === "win32") { + sh = "cmd" + shFlag = "/c" + } + + log.verbose(unsafe, "unsafe-perm in lifecycle") + + output.write("\n> "+pkg._id+" " + stage+" "+wd+"\n> "+cmd+"\n", function (er) { + if (er) return cb(er) + + exec( sh, [shFlag, cmd], env, true, wd + , user, group + , function (er, code, stdout, stderr) { + if (er && !npm.ROLLBACK) { + log("Failed to exec "+stage+" script", pkg._id) + er.message = pkg._id + " " + + stage + ": `" + env.npm_lifecycle_script+"`\n" + + er.message + if (er.errno !== constants.EPERM) { + er.errno = npm.ELIFECYCLE + } + er.pkgid = pkg._id + er.stage = stage + er.script = env.npm_lifecycle_script + er.pkgname = pkg.name + return cb(er) + } else if (er) { + log.error(er, pkg._id+"."+stage) + log.error("failed, but continuing anyway", pkg._id+"."+stage) + return cb() + } + cb(er) + }) + }) +} + +function runHookLifecycle (pkg, env, wd, unsafe, cb) { + // check for a hook script, run if present. + var stage = env.npm_lifecycle_event + , hook = path.join(npm.dir, ".hooks", stage) + , user = unsafe ? null : npm.config.get("user") + , group = unsafe ? null : npm.config.get("group") + , cmd = hook + + fs.stat(hook, function (er) { + if (er) return cb() + + exec( "sh", ["-c", cmd], env, true, wd + , user, group + , function (er) { + if (er) { + er.message += "\nFailed to exec "+stage+" hook script" + log(er, pkg._id) + } + if (npm.ROLLBACK) return cb() + cb(er) + }) + }) +} + +function makeEnv (data, prefix, env) { + prefix = prefix || "npm_package_" + if (!env) { + env = {} + for (var i in process.env) if (!i.match(/^npm_/)) { + env[i] = process.env[i] + } + + // npat asks for tap output + if (npm.config.get("npat")) env.TAP = 1 + + // express and others respect the NODE_ENV value. + if (npm.config.get("production")) env.NODE_ENV = "production" + + } else if (!data.hasOwnProperty("_lifecycleEnv")) { + Object.defineProperty(data, "_lifecycleEnv", + { value : env + , enumerable : false + }) + } + + for (var i in data) if (i.charAt(0) !== "_") { + var envKey = (prefix+i).replace(/[^a-zA-Z0-9_]/g, '_') + if (data[i] && typeof(data[i]) === "object") { + try { + // quick and dirty detection for cyclical structures + JSON.stringify(data[i]) + makeEnv(data[i], envKey+"_", env) + } catch (ex) { + // usually these are package objects. + // just get the path and basic details. + var d = data[i] + makeEnv( { name: d.name, version: d.version, path:d.path } + , envKey+"_", env) + } + } else { + env[envKey] = String(data[i]) + env[envKey] = -1 !== env[envKey].indexOf("\n") + ? JSON.stringify(env[envKey]) + : env[envKey] + } + + } + + if (prefix !== "npm_package_") return env + + prefix = "npm_config_" + var pkgConfig = {} + , ini = require("./ini.js") + , keys = ini.keys + , pkgVerConfig = {} + , namePref = data.name + ":" + , verPref = data.name + "@" + data.version + ":" + + keys.forEach(function (i) { + if (i.charAt(0) === "_" && i.indexOf("_"+namePref) !== 0) { + return + } + var value = String(ini.get(i)) + value = -1 !== value.indexOf("\n") + ? JSON.stringify(value) + : value + i = i.replace(/^_+/, "") + if (i.indexOf(namePref) === 0) { + var k = i.substr(namePref.length).replace(/[^a-zA-Z0-9_]/g, "_") + pkgConfig[ k ] = value + } else if (i.indexOf(verPref) === 0) { + var k = i.substr(verPref.length).replace(/[^a-zA-Z0-9_]/g, "_") + pkgVerConfig[ k ] = value + } + var envKey = (prefix+i).replace(/[^a-zA-Z0-9_]/g, "_") + env[envKey] = value + }) + + prefix = "npm_package_config_" + ;[pkgConfig, pkgVerConfig].forEach(function (conf) { + for (var i in conf) { + var envKey = (prefix+i) + env[envKey] = conf[i] + } + }) + + return env +} + +function cmd (stage) { + function CMD (args, cb) { + if (args.length) { + chain(args.map(function (p) { + return [npm.commands, "run-script", [p, stage]] + }), cb) + } else npm.commands["run-script"]([stage], cb) + } + CMD.usage = "npm "+stage+" " + var installedShallow = require("./completion/installed-shallow.js") + CMD.completion = function (opts, cb) { + installedShallow(opts, function (d) { + return d.scripts && d.scripts[stage] + }, cb) + } + return CMD +} diff --git a/deps/npm/lib/utils/link.js b/deps/npm/lib/utils/link.js new file mode 100644 index 0000000000..9be1221f0a --- /dev/null +++ b/deps/npm/lib/utils/link.js @@ -0,0 +1,30 @@ + +module.exports = link +link.ifExists = linkIfExists + +var fs = require("graceful-fs") + , chain = require("slide").chain + , mkdir = require("./mkdir-p.js") + , rm = require("rimraf") + , log = require("./log.js") + , path = require("path") + , relativize = require("./relativize.js") + , npm = require("../npm.js") + +function linkIfExists (from, to, gently, cb) { + fs.stat(from, function (er) { + if (er) return cb() + link(from, to, gently, cb) + }) +} + +function link (from, to, gently, cb) { + if (typeof cb !== "function") cb = gently, gently = null + if (npm.config.get("force")) gently = false + chain + ( [ [fs, "stat", from] + , [rm, to, { gently: gently }] + , [mkdir, path.dirname(to)] + , [fs, "symlink", relativize(from, to), to] ] + , cb) +} diff --git a/deps/npm/lib/utils/load-package-defaults.js b/deps/npm/lib/utils/load-package-defaults.js new file mode 100644 index 0000000000..180507a445 --- /dev/null +++ b/deps/npm/lib/utils/load-package-defaults.js @@ -0,0 +1,75 @@ + +module.exports = loadPackageDefaults + +var path = require("path") + , log = require("./log.js") + , find = require("./find.js") + , asyncMap = require("slide").asyncMap + , npm = require("../npm.js") + , fs = require("graceful-fs") + +function loadPackageDefaults (pkg, pkgDir, cb) { + if (!pkg) return cb(new Error("no package to load defaults from!")) + if (typeof pkgDir === "function") { + cb = pkgDir + pkgDir = path.join(npm.dir, pkg.name, pkg.version, "package") + } + if (!pkgDir) pkgDir = "." + + if (pkg._defaultsLoaded) return cb(null, pkg) + + pkg._defaultsLoaded = true + asyncMap + ( [pkg] + , function (pkg, cb) { log.verbose(pkg._id, "loadDefaults", cb) } + , readDefaultBins(pkgDir) + , readDefaultMans(pkgDir) + , function (er) { cb(er, pkg) } ) +} + +function objectForEach (obj, fn) { + Object.keys(obj).forEach(function (k) { + fn(obj[k]) + }) +} + +function readDefaultMans (pkgDir) { return function (pkg, cb) { + var man = pkg.directories && pkg.directories.man + , manDir = path.join(pkgDir, man) + if (pkg.man && !Array.isArray(pkg.man)) pkg.man = [pkg.man] + if (pkg.man || !man) return cb(null, pkg) + find(manDir, /\.[0-9]+(\.gz)?$/, function (er, filenames) { + if (er) return cb(er) + var cut = pkgDir === "." ? 0 : pkgDir.length + 1 + pkg.man = (filenames || []).map(function (filename) { + return filename.substr(cut) + }).filter(function (f) { + return !f.match(/(^|\/)\./) + }) + cb(null,pkg) + }) +}} + +function readDefaultBins (pkgDir) { return function (pkg, cb) { + var bin = pkg.directories && pkg.directories.bin + if (pkg.bins) pkg.bin = pkg.bins, delete pkg.bins + if (pkg.bin || !bin) return cb(null, pkg) + log.verbose("linking default bins", pkg._id) + var binDir = path.join(pkgDir, bin) + pkg.bin = {} + find(binDir, function (er, filenames) { + if (er || !filenames || !filenames.length) return cb(er, pkg) + var cut = pkgDir === "." ? 0 : pkgDir.length + 1 + , binCut = pkgDir === "." ? bin.length - 1 : binDir.length + 1 + filenames.filter(function (f) { + return !f.substr(binCut).match(/(^|\/)\./) + }).forEach(function (filename) { + var key = filename.substr(binCut) + .replace(/\.(js|node)$/, '') + , val = filename.substr(cut) + if (key.length && val.length) pkg.bin[key] = val + }) + log.silly(pkg.bin, pkg._id+".bin") + cb(null, pkg) + }) +}} diff --git a/deps/npm/lib/utils/log.js b/deps/npm/lib/utils/log.js new file mode 100644 index 0000000000..dece9e3d19 --- /dev/null +++ b/deps/npm/lib/utils/log.js @@ -0,0 +1,185 @@ + +/* +log levels: +0,1,2,3 +verbose,info,warn,error + +Default setting for logs is "info" +Default setting to show is "info" + +Possible values of level/loglevel: +silly,verbose,info,warn,error,win,silent + +silent quiets everything + + +*/ + + +module.exports = log + +var output = require("./output.js") + +function colorize (msg, color) { + return msg ? "\033["+color+"m"+msg+"\033[0m" : "" +} + +var l = -1 + , LEVEL = { silly : l++ + , verbose : l++ + , info : l++ + , WARN : l++ + , "ERR!" : l++ + , ERROR : "ERR!" + , ERR : "ERR!" + , win : 0x15AAC5 + , paused : 0x19790701 + , silent : 0xDECAFBAD + } + , COLOR = {} + , SHOWLEVEL = null + , normalNames = {} +log.LEVEL = LEVEL +normalNames[LEVEL["ERR!"]] = "error" +normalNames[LEVEL.WARN] = "warn" +normalNames[LEVEL.info] = "info" +normalNames[LEVEL.verbose] = "verbose" +normalNames[LEVEL.silly] = "silly" +normalNames[LEVEL.win] = "win" + +Object.keys(LEVEL).forEach(function (l) { + if (typeof LEVEL[l] === "string") LEVEL[l] = LEVEL[LEVEL[l]] + else LEVEL[LEVEL[l]] = l + LEVEL[l.toLowerCase()] = LEVEL[l] + if (l === "silent" || l === "paused") return + log[l] = log[l.toLowerCase()] = + function (msg, pref, cb) { return log(msg, pref, l, cb) } +}) + +COLOR[LEVEL.silly] = 30 +COLOR[LEVEL.verbose] = "34;40" +COLOR[LEVEL.info] = 32 +COLOR[LEVEL.warn] = "30;41" +COLOR[LEVEL.error] = "31;40" +for (var c in COLOR) COLOR[LEVEL[c]] = COLOR[c] +COLOR.npm = "37;40" +COLOR.pref = 35 + +var logBuffer = [] + , ini = require("./ini.js") + , waitForConfig +log.waitForConfig = function () { waitForConfig = true } + +// now the required stuff has been loaded, +// so the transitive module dep will work +var util = require("util") + , npm = require("../npm.js") + , net = require("net") + +Object.defineProperty(log, "level", + { get : function () { + if (SHOWLEVEL !== null) return SHOWLEVEL + var show = npm.config && npm.config.get("loglevel") || '' + show = show.split(",")[0] + if (!isNaN(show)) show = +show + else if (!LEVEL.hasOwnProperty(show)) { + util.error("Invalid loglevel config: "+JSON.stringify(show)) + show = "info" + } + if (isNaN(show)) show = LEVEL[show] + else show = +show + if (!waitForConfig || ini.resolved) SHOWLEVEL = show + return show + } + , set : function (l) { + SHOWLEVEL = null + npm.config.set("showlevel", l) + } + }) + +function log (msg, pref, level, cb) { + if (typeof level === "function") cb = level, level = null + var show = log.level + if (show === LEVEL.silent || show === LEVEL.paused) return cb && cb() + if (level == null) level = LEVEL.info + if (isNaN(level)) level = LEVEL[level] + else level = +level + + // logging just undefined is almost never the right thing. + // a lot of these are kicking around throughout the codebase + // with relatively unhelpful prefixes. + if (msg === undefined && level > LEVEL.silly) { + msg = new Error("undefined log message") + } + if (typeof msg === "object" && (msg instanceof Error)) level = LEVEL.error + if (!ini.resolved && waitForConfig || level === LEVEL.paused) { + return logBuffer.push([msg, pref, level, cb]) + } + if (logBuffer.length && !logBuffer.discharging) { + logBuffer.push([msg, pref, level, cb]) + logBuffer.discharging = true + logBuffer.forEach(function (l) { log.apply(null, l) }) + logBuffer.length = 0 + delete logBuffer.discharging + return + } + log.level = show + npm.emit("log", { level : level, msg : msg, pref : pref, cb : cb }) + npm.emit("log."+normalNames[level], { msg : msg, pref : pref, cb : cb }) +} + +var loglog = log.history = [] + , loglogLen = 0 +npm.on("log", function (logData) { + var level = logData.level + , msg = logData.msg + , pref = logData.pref + , cb = logData.cb || function () {} + , show = log.level + , spaces = " " + , logFD = npm.config.get("logfd") + if (msg instanceof Error) { + msg = logData.msg = msg.stack || msg.toString() + } + loglog.push(logData) + loglogLen ++ + if (loglogLen > 2000) { + loglog = loglog.slice(loglogLen - 1000) + loglogLen = 1000 + } + if (!isFinite(level) || level < show) return cb() + if (typeof msg !== "string" && !(msg instanceof Error)) { + msg = util.inspect(msg, 0, 4, true) + } + + // console.error("level, showlevel, show", level, show, (level >= show)) + if (pref && COLOR.pref) { + pref = colorize(pref, COLOR.pref) + } + if (!pref) pref = "" + + if (npm.config.get("logprefix")) { + pref = colorize("npm", COLOR.npm) + + (COLOR[level] ? " "+colorize( + (LEVEL[level]+spaces).substr(0,4), COLOR[level]) : "") + + (pref ? (" " + pref) : "") + } + if (pref) pref += " " + + + + if (msg.indexOf("\n") !== -1) { + msg = msg.split(/\n/).join("\n"+pref) + } + msg = pref+msg + return output.write(msg, logFD, cb) +}) + +log.er = function (cb, msg) { + if (!msg) throw new Error( + "Why bother logging it if you're not going to print a message?") + return function (er) { + if (er) log.error(msg) + cb.apply(this, arguments) + } +} diff --git a/deps/npm/lib/utils/mkdir-p.js b/deps/npm/lib/utils/mkdir-p.js new file mode 100644 index 0000000000..cc2b465fb6 --- /dev/null +++ b/deps/npm/lib/utils/mkdir-p.js @@ -0,0 +1,191 @@ + +var log = require("./log.js") + , fs = require("graceful-fs") + , path = require("path") + , npm = require("../npm.js") + , exec = require("./exec.js") + , uidNumber = require("./uid-number.js") + , umask = process.umask() + , umaskOrig = umask + , addedUmaskExit = false + , mkdirCache = {} + +module.exports = mkdir +function mkdir (ensure, mode, uid, gid, noChmod, cb_) { + if (typeof cb_ !== "function") cb_ = noChmod, noChmod = null + if (typeof cb_ !== "function") cb_ = gid, gid = null + if (typeof cb_ !== "function") cb_ = uid, uid = null + if (typeof cb_ !== "function") cb_ = mode, mode = npm.modes.exec + + if (mode & umask) { + log.verbose(mode.toString(8), "umasking from "+umask.toString(8)) + process.umask(umask = 0) + if (!addedUmaskExit) { + addedUmaskExit = true + process.on("exit", function () { process.umask(umask = umaskOrig) }) + } + } + + ensure = path.resolve(ensure).replace(/\/+$/, '') + + // mkdir("/") should not do anything, since that always exists. + if (!ensure + || ( process.platform === "win32" + && ensure.match(/^[a-zA-Z]:(\\|\/)?$/))) { + return cb_() + } + + if (mkdirCache.hasOwnProperty(ensure)) { + return mkdirCache[ensure].push(cb_) + } + mkdirCache[ensure] = [cb_] + + function cb (er) { + var cbs = mkdirCache[ensure] + delete mkdirCache[ensure] + cbs.forEach(function (c) { c(er) }) + } + + if (uid === null && gid === null) { + return mkdir_(ensure, mode, uid, gid, noChmod, cb) + } + + uidNumber(uid, gid, function (er, uid, gid) { + if (er) return cb(er) + mkdir_(ensure, mode, uid, gid, noChmod, cb) + }) +} + +function mkdir_ (ensure, mode, uid, gid, noChmod, cb) { + // if it's already a dir, then just check the bits and owner. + fs.stat(ensure, function (er, s) { + if (s && s.isDirectory()) { + // check mode, uid, and gid. + if ((noChmod || (s.mode & mode) === mode) + && (typeof uid !== "number" || s.uid === uid) + && (typeof gid !== "number" || s.gid === gid)) return cb() + return done(ensure, mode, uid, gid, noChmod, cb) + } + return walkDirs(ensure, mode, uid, gid, noChmod, cb) + }) +} + +function done (ensure, mode, uid, gid, noChmod, cb) { + // now the directory has been created. + // chown it to the desired uid/gid + // Don't chown the npm.root dir, though, in case we're + // in unsafe-perm mode. + log.verbose("done: "+ensure+" "+mode.toString(8), "mkdir") + + // only chmod if noChmod isn't set. + var d = done_(ensure, mode, uid, gid, cb) + if (noChmod) return d() + fs.chmod(ensure, mode, d) +} + +function done_ (ensure, mode, uid, gid, cb) { + return function (er) { + if (er + || ensure === npm.dir + || typeof uid !== "number" + || typeof gid !== "number" + || npm.config.get("unsafe-perm")) return cb(er) + uid = Math.floor(uid) + gid = Math.floor(gid) + fs.chown(ensure, uid, gid, cb) + } +} + +var pathSplit = process.platform === "win32" ? /\/|\\/ : "/" +function walkDirs (ensure, mode, uid, gid, noChmod, cb) { + var dirs = ensure.split(pathSplit) + , walker = [] + , foundUID = null + , foundGID = null + + // gobble the "/" or C: first + walker.push(dirs.shift()) + + // The loop that goes through and stats each dir. + ;(function S (d) { + // no more directory steps left. + if (d === undefined) { + // do the chown stuff + return done(ensure, mode, uid, gid, noChmod, cb) + } + + // get the absolute dir for the next piece being stat'd + walker.push(d) + var dir = walker.join(path.SPLIT_CHAR) + + // stat callback lambda + fs.stat(dir, function STATCB (er, s) { + if (er) { + // the stat failed - directory does not exist. + + log.verbose(er.message, "mkdir (expected) error") + + // use the same uid/gid as the nearest parent, if not set. + if (foundUID !== null) uid = foundUID + if (foundGID !== null) gid = foundGID + + // make the directory + fs.mkdir(dir, mode, function MKDIRCB (er) { + // since stat and mkdir are done as two separate syscalls, + // operating on a path rather than a file descriptor, it's + // possible that the directory didn't exist when we did + // the stat, but then *did* exist when we go to to the mkdir. + // If we didn't care about uid/gid, we could just mkdir + // repeatedly, failing on any error other than "EEXIST". + if (er && er.message.indexOf("EEXIST") === 0) { + return fs.stat(dir, STATCB) + } + + // any other kind of error is not saveable. + if (er) return cb(er) + + // at this point, we've just created a new directory successfully. + + // if we care about permissions + if (!npm.config.get("unsafe-perm") // care about permissions + // specified a uid and gid + && uid !== null + && gid !== null ) { + // set the proper ownership + return fs.chown(dir, uid, gid, function (er) { + if (er) return cb(er) + // attack the next portion of the path. + S(dirs.shift()) + }) + } else { + // either we don't care about ownership, or it's already right. + S(dirs.shift()) + } + }) // mkdir + + } else { + // the stat succeeded. + if (s.isDirectory()) { + // if it's a directory, that's good. + // if the uid and gid aren't already set, then try to preserve + // the ownership on up the tree. Things in ~ remain owned by + // the user, things in / remain owned by root, etc. + if (uid === null && typeof s.uid === "number") foundUID = s.uid + if (gid === null && typeof s.gid === "number") foundGID = s.gid + + // move onto next portion of path + S(dirs.shift()) + + } else { + // the stat succeeded, but it's not a directory + log.verbose(dir, "mkdir exists") + log.silly(s, "stat("+dir+")") + log.verbose(s.isDirectory(), "isDirectory()") + cb(new Error("Failed to mkdir "+dir+": File exists")) + }// if (isDirectory) else + } // if (stat failed) else + }) // stat + + // start the S function with the first item in the list of directories. + })(dirs.shift()) +} diff --git a/deps/npm/lib/utils/npm-registry-client/adduser.js b/deps/npm/lib/utils/npm-registry-client/adduser.js new file mode 100644 index 0000000000..5e6794a758 --- /dev/null +++ b/deps/npm/lib/utils/npm-registry-client/adduser.js @@ -0,0 +1,100 @@ + +module.exports = adduser + +var uuid = require("node-uuid") + , request = require("./request.js") + , log = require("../log.js") + , npm = require("../../npm.js") + , crypto + +try { + crypto = process.binding("crypto") && require("crypto") +} catch (ex) {} + +function sha (s) { + return crypto.createHash("sha1").update(s).digest("hex") +} + +function adduser (username, password, email, cb) { + if (!crypto) return cb(new Error( + "You must compile node with ssl support to use the adduser feature")) + + password = ("" + (password || "")).trim() + if (!password) return cb(new Error("No password supplied.")) + + email = ("" + (email || "")).trim() + if (!email) return cb(new Error("No email address supplied.")) + if (!email.match(/^[^@]+@[^\.]+\.[^\.]+/)) { + return cb(new Error("Please use a real email address.")) + } + + if (password.indexOf(":") !== -1) return cb(new Error( + "Sorry, ':' chars are not allowed in passwords.\n"+ + "See for why.")) + var salt = uuid() + , userobj = + { name : username + , salt : salt + , password_sha : sha(password + salt) + , email : email + , _id : 'org.couchdb.user:'+username + , type : "user" + , roles : [] + , date: new Date().toISOString() + } + cb = done(cb) + log.verbose(userobj, "before first PUT") + request.PUT + ( '/-/user/org.couchdb.user:'+encodeURIComponent(username) + , userobj + , function (error, data, json, response) { + // if it worked, then we just created a new user, and all is well. + // but if we're updating a current record, then it'll 409 first + if (error && !npm.config.get("_auth")) { + // must be trying to re-auth on a new machine. + // use this info as auth + npm.config.set("username", username) + npm.config.set("_password", password) + var b = new Buffer(username + ":" + password) + npm.config.set("_auth", b.toString("base64")) + } + if (!error || !response || response.statusCode !== 409) { + return cb(error, data, json, response) + } + log.verbose("update existing user", "adduser") + return request.GET + ( '/-/user/org.couchdb.user:'+encodeURIComponent(username) + , function (er, data, json, response) { + userobj._rev = data._rev + userobj.roles = data.roles + log.verbose(userobj, "userobj") + request.PUT + ( '/-/user/org.couchdb.user:'+encodeURIComponent(username) + + "/-rev/" + userobj._rev + , userobj + , cb ) + } + ) + } + ) +} + +function done (cb) { return function (error, data, json, response) { + if (!error && (!response || response.statusCode === 201)) { + return cb(error, data, json, response) + } + log.verbose([error, data, json], "back from adduser") + if (!error) { + error = new Error( (response && response.statusCode || "") + " "+ + "Could not create user\n"+JSON.stringify(data)) + } + if (response + && (response.statusCode === 401 || response.statusCode === 403)) { + log.warn("Incorrect username or password\n" + +"You can reset your account by visiting:\n" + +"\n" + +" http://admin.npmjs.org/reset\n") + } + + return cb(error) +}} diff --git a/deps/npm/lib/utils/npm-registry-client/get.js b/deps/npm/lib/utils/npm-registry-client/get.js new file mode 100644 index 0000000000..062244df3a --- /dev/null +++ b/deps/npm/lib/utils/npm-registry-client/get.js @@ -0,0 +1,171 @@ + +module.exports = get + +var GET = require("./request.js").GET + , fs = require("graceful-fs") + , npm = require("../../npm.js") + , path = require("path") + , log = require("../log.js") + , mkdir = require("../mkdir-p.js") + , cacheStat = null + +function get (project, version, timeout, nofollow, staleOk, cb) { + if (typeof cb !== "function") cb = staleOk, staleOk = false + if (typeof cb !== "function") cb = nofollow, nofollow = false + if (typeof cb !== "function") cb = timeout, timeout = -1 + if (typeof cb !== "function") cb = version, version = null + if (typeof cb !== "function") cb = project, project = null + if (typeof cb !== "function") { + throw new Error("No callback provided to registry.get") + } + + if ( process.env.COMP_CWORD !== undefined + && process.env.COMP_LINE !== undefined + && process.env.COMP_POINT !== undefined + ) timeout = Math.max(timeout, 60000) + + var uri = [] + uri.push(project || "") + if (version) uri.push(version) + uri = uri.join("/") + + // /-/all is special. + // It uses timestamp-based caching and partial updates, + // because it is a monster. + if (uri === "/-/all") { + return requestAll(cb) + } + + var cache = path.join(npm.cache, uri, ".cache.json") + fs.stat(cache, function (er, stat) { + if (!er) fs.readFile(cache, function (er, data) { + try { data = JSON.parse(data) } + catch (ex) { data = null } + get_(uri, timeout, cache, stat, data, nofollow, staleOk, cb) + }) + else get_(uri, timeout, cache, null, null, nofollow, staleOk, cb) + }) +} + +function requestAll (cb) { + var cache = path.join(npm.cache, "/-/all", ".cache.json") + + mkdir(path.join(npm.cache, "-", "all"), function (er) { + fs.readFile(cache, function (er, data) { + if (er) return requestAll_(0, {}, cb) + try { + data = JSON.parse(data) + } catch (ex) { + fs.writeFile(cache, "{}", function (er) { + if (er) return cb(new Error("Broken cache. " + +"Please run 'npm cache clean' " + +"and try again.")) + return requestAll_(0, {}, cb) + }) + } + var t = +data._updated || 0 + requestAll_(t, data, cb) + }) + }) +} + +function requestAll_ (c, data, cb) { + // use the cache and update in the background if it's not too old + if (Date.now() - c < 60000) { + cb(null, data) + cb = function () {} + } + + var uri = "/-/all/since?stale=update_after&startkey=" + c + + if (c === 0) { + log.warn("Building the index for the first time, please be patient") + uri = "/-/all" + } + + var cache = path.join(npm.cache, "-/all", ".cache.json") + GET(uri, function (er, updates, _, res) { + if (er) return cb(er, data) + var headers = res.headers + , updated = Date.parse(headers.date) + Object.keys(updates).forEach(function (p) { + data[p] = updates[p] + }) + data._updated = updated + fs.writeFile( cache, JSON.stringify(data) + , function (er) { + delete data._updated + return cb(er, data) + }) + }) +} + +function get_ (uri, timeout, cache, stat, data, nofollow, staleOk, cb) { + var etag + if (data && data._etag) etag = data._etag + if (timeout && timeout > 0 && stat && data) { + if ((Date.now() - stat.mtime.getTime())/1000 < timeout) { + log.verbose("not expired, no request", "registry.get " +uri) + delete data._etag + return cb(null, data, JSON.stringify(data), {statusCode:304}) + } + if (staleOk) { + log.verbose("staleOk, background update", "registry.get " +uri) + delete data._etag + process.nextTick(cb.bind( null, null, data, JSON.stringify(data) + , {statusCode: 304} )) + cb = function () {} + } + } + + GET(uri, etag, nofollow, function (er, remoteData, raw, response) { + if (response) { + log.silly([response.statusCode, response.headers], "get cb") + if (response.statusCode === 304 && etag) { + remoteData = data + log.verbose(uri+" from cache", "etag") + } + } + + data = remoteData + if (er) return cb(er, data, raw, response) + + // just give the write the old college try. if it fails, whatever. + function saved () { + delete data._etag + cb(er, data, raw, response) + } + + saveToCache(cache, data, saved) + }) +} + +function saveToCache (cache, data, saved) { + if (cacheStat) { + return saveToCache_(cache, data, cacheStat.uid, cacheStat.gid, saved) + } + fs.stat(npm.cache, function (er, st) { + if (er) { + return fs.stat(process.env.HOME || "", function (er, st) { + // if this fails, oh well. + if (er) return saved() + cacheStat = st + return saveToCache(cache, data, saved) + }) + } + cacheStat = st || { uid: null, gid: null } + return saveToCache(cache, data, saved) + }) +} + +function saveToCache_ (cache, data, uid, gid, saved) { + mkdir(path.dirname(cache), npm.modes.exec, uid, gid, function (er) { + if (er) return saved() + fs.writeFile(cache, JSON.stringify(data), function (er) { + if (er || uid === null || gid === null) { + return saved() + } + fs.chown(cache, uid, gid, saved) + }) + }) +} diff --git a/deps/npm/lib/utils/npm-registry-client/index.js b/deps/npm/lib/utils/npm-registry-client/index.js new file mode 100644 index 0000000000..2a42947450 --- /dev/null +++ b/deps/npm/lib/utils/npm-registry-client/index.js @@ -0,0 +1,16 @@ + +// utilities for working with the js-registry site. + +var cached = {} +function lazyGet (p) { return function () { + return cached[p] || (cached[p] = require("./"+p+".js")) +}} + +function setLazyGet (p) { + Object.defineProperty(exports, p, + { get : lazyGet(p) + , enumerable : true }) +} + +; ["publish", "unpublish", "tag", "adduser", "get", "request", "star"] + .forEach(setLazyGet) diff --git a/deps/npm/lib/utils/npm-registry-client/publish.js b/deps/npm/lib/utils/npm-registry-client/publish.js new file mode 100644 index 0000000000..86cfdc6a89 --- /dev/null +++ b/deps/npm/lib/utils/npm-registry-client/publish.js @@ -0,0 +1,163 @@ + +module.exports = publish + +var request = require("./request.js") + , GET = request.GET + , PUT = request.PUT + , DELETE = request.DELETE + , reg = request.reg + , upload = request.upload + , log = require("../log.js") + , path = require("path") + , npm = require("../../npm.js") + , url = require("url") + +function publish (data, prebuilt, readme, cb) { + if (typeof readme === "function") cb = readme, readme = null + if (typeof prebuilt === "function") cb = prebuilt, prebuilt = null + // add the dist-url to the data, pointing at the tarball. + // if the {name} isn't there, then create it. + // if the {version} is already there, then fail. + // then: + // PUT the data to {config.registry}/{data.name}/{data.version} + var registry = reg() + if (registry instanceof Error) return cb(registry) + + var fullData = + { _id : data.name + , name : data.name + , description : data.description + , "dist-tags" : {} + , versions : {} + , readme: readme ? "" + readme : null + , maintainers : + [ { name : npm.config.get("username") + , email : npm.config.get("email") + } + ] + } + + var tbName = data.name + "-" + data.version + ".tgz" + , bd = npm.config.get("bindist") + , pbName = data.name + "-" + data.version + "-" + bd + ".tgz" + , tbURI = data.name + "/-/" + tbName + , pbURI = data.name + "/-/" + pbName + + data._id = data.name+"@"+data.version + data.dist = data.dist || {} + data.dist.tarball = url.resolve(registry, tbURI) + .replace(/^https:\/\//, "http://") + + if (prebuilt && bd) { + data.dist.bin[bd] = data.dist.bin[bd] || {} + data.dist.bin[bd].tarball = url.resolve(registry, pbURI) + .replace(/^https:\/\//, "http://") + } + + + + + // first try to just PUT the whole fullData, and this will fail if it's + // already there, because it'll be lacking a _rev, so couch'll bounce it. + PUT(encodeURIComponent(data.name), fullData, + function (er, parsed, json, response) { + // get the rev and then upload the attachment + // a 409 is expected here, if this is a new version of an existing package. + if (er + && !(response && response.statusCode === 409) + && !( parsed + && parsed.reason === + "must supply latest _rev to update existing package" )) { + return log.er(cb, "Failed PUT response " + +(response && response.statusCode))(er) + } + var dataURI = encodeURIComponent(data.name) + + "/" + encodeURIComponent(data.version) + + var tag = data.tag || npm.config.get("tag") + if (npm.config.get("pre")) dataURI += "/-pre/true" + else if (tag) dataURI += "/-tag/" + tag + else dataURI += "/-tag/latest" + + // let's see what verions are already published. + // could be that we just need to update the bin dist values. + GET(data.name, function (er, fullData) { + if (er) return cb(er) + + var exists = fullData.versions && fullData.versions[data.version] + if (exists) { + log(exists._id, "Already published") + var ebin = exists.dist.bin || {} + , nbin = data.dist.bin || {} + , needs = Object.keys(nbin).filter(function (bd) { + return !ebin.hasOwnProperty(bd) + }) + log.verbose(needs, "uploading bin dists") + if (!needs.length) return cb(conflictError(data._id)) + // attach the needed bindists, upload the new metadata + exists.dist.bin = ebin + needs.forEach(function (bd) { exists.dist.bin[bd] = nbin[bd] }) + return PUT(dataURI + "/-rev/" + fullData._rev, exists, function (er) { + if (er) return cb(er) + attach(data.name, prebuilt, pbName, cb) + }) + } + + PUT(dataURI, data, function (er) { + if (er) { + if (er.message.indexOf("conflict Document update conflict.") === 0) { + return cb(conflictError(data._id)) + } + return log.er(cb, "Error sending version data")(er) + } + + var c = path.resolve(npm.cache, data.name, data.version) + , tb = path.resolve(c, "package.tgz") + + cb = rollbackFailure(data, cb) + + log.verbose([data.name, tb, tbName], "attach 2") + attach(data.name, tb, tbName, function (er) { + log.verbose([er, data.name, prebuilt, pbName], "attach 3") + if (er || !prebuilt) return cb(er) + attach(data.name, prebuilt, pbName, cb) + }) + }) + }) + }) +} + +function conflictError (pkgid) { + var e = new Error("publish fail") + e.errno = npm.EPUBLISHCONFLICT + e.pkgid = pkgid + return e +} + +function attach (doc, file, filename, cb) { + doc = encodeURIComponent(doc) + GET(doc, function (er, d) { + if (er) return cb(er) + if (!d) return cb(new Error( + "Attempting to upload to invalid doc "+doc)) + var rev = "-rev/"+d._rev + , attURI = doc + "/-/" + encodeURIComponent(filename) + "/" + rev + log.verbose([attURI, file], "uploading") + upload(attURI, file, cb) + }) +} + +function rollbackFailure (data, cb) { return function (er) { + if (!er) return cb() + npm.ROLLBACK = true + log.error(er, "publish failed") + log("rollback", "publish failed") + npm.commands.unpublish([data.name+"@"+data.version], function (er_) { + if (er_) { + log.error(er_, "rollback failed") + log.error( "Invalid data in registry! Please report this." + , "rollback failed" ) + } else log("rolled back", "publish failed") + cb(er) + }) +}} diff --git a/deps/npm/lib/utils/npm-registry-client/request.js b/deps/npm/lib/utils/npm-registry-client/request.js new file mode 100644 index 0000000000..d19e3ac31f --- /dev/null +++ b/deps/npm/lib/utils/npm-registry-client/request.js @@ -0,0 +1,242 @@ +module.exports = regRequest + +regRequest.GET = GET +regRequest.PUT = PUT +regRequest.reg = reg +regRequest.upload = upload + +var npm = require("../../npm.js") + , url = require("url") + , log = require("../log.js") + , fs = require("graceful-fs") + , rm = require("rimraf") + , asyncMap = require("slide").asyncMap + , warnedAuth = false + , newloctimeout = 0 + , stream = require("stream") + , Stream = stream.Stream + , request = require("request") + , getAgent = require("../get-agent.js") + +function regRequest (method, where, what, etag, nofollow, cb_) { + if (typeof cb_ !== "function") cb_ = nofollow, nofollow = false + if (typeof cb_ !== "function") cb_ = etag, etag = null + if (typeof cb_ !== "function") cb_ = what, what = null + + log.verbose(where||"/", method) + + // Since there are multiple places where an error could occur, + // don't let the cb be called more than once. + var errState = null + function cb (er) { + if (errState) return + if (er) errState = er + cb_.apply(null, arguments) + } + + if (where.match(/^\/?favicon.ico/)) { + return cb(new Error("favicon.ico isn't a package, it's a picture.")) + } + + var registry = reg() + if (registry instanceof Error) return cb(registry) + + var adduserChange = /^\/?-\/user\/org\.couchdb\.user:([^\/]+)\/-rev/ + , adduserNew = /^\/?-\/user\/org\.couchdb\.user:([^\/]+)/ + , authRequired = (what || npm.config.get("always-auth")) + && !where.match(adduserNew) + || where.match(adduserChange) + || method === "DELETE" + + // resolve to a full url on the registry + if (!where.match(/^https?:\/\//)) { + log.verbose(where, "raw, before any munging") + + var q = where.split("?") + where = q.shift() + q = q.join("?") + + if (where.charAt(0) !== "/") where = "/" + where + where = "." + where.split("/").map(function (p) { + p = p.trim() + if (p.match(/^org.couchdb.user/)) { + return p.replace(/\//g, encodeURIComponent("/")) + } + return encodeURIComponent(p) + }).join("/") + if (q) where += "?" + q + log.verbose([registry, where], "url resolving") + where = url.resolve(registry, where) + log.verbose(where, "url resolved") + } + + var remote = url.parse(where) + , auth = authRequired && npm.config.get("_auth") + + if (authRequired && !auth) { + return cb(new Error( + "Cannot insert data into the registry without authorization\n" + + "See: npm-adduser(1)")) + } + + if (auth) remote.auth = new Buffer(auth, "base64").toString("utf8") + + makeRequest(method, remote, where, what, etag, nofollow, cb) +} + +function makeRequest (method, remote, where, what, etag, nofollow, cb) { + var opts = { url: remote + , method: method + , agent: getAgent(remote) + , strictSSL: npm.config.get("strict-ssl") } + , headers = opts.headers = {} + if (etag) { + log.verbose(etag, "etag") + headers[method === "GET" ? "if-none-match" : "if-match"] = etag + } + + headers.accept = "application/json" + + opts.proxy = npm.config.get( remote.protocol === "https:" + ? "https-proxy" : "proxy" ) + + // figure out wth 'what' is + if (what) { + if (Buffer.isBuffer(what) || typeof what === "string") { + opts.body = what + headers["content-type"] = "application/json" + headers["content-length"] = Buffer.byteLength(what) + } else if (what instanceof Stream) { + headers["content-type"] = "application/octet-stream" + if (what.size) headers["content-length"] = what.size + } else { + delete what._etag + opts.json = what + } + } + + if (nofollow) { + opts.followRedirect = false + } + + var req = request(opts, requestDone(method, where, cb)) + var r = npm.config.get("registry") + if (!r) { + return new Error("Must define registry URL before accessing registry.") + } + + req.on("error", cb) + + if (what && (what instanceof Stream)) { + what.pipe(req) + } +} + +// cb(er, parsed, raw, response) +function requestDone (method, where, cb) { return function (er, response, data) { + if (er) return cb(er) + + var parsed + + if (Buffer.isBuffer(data)) { + data = data.toString() + } + + if (data && typeof data === "string" && response.statusCode !== 304) { + try { + parsed = JSON.parse(data) + } catch (ex) { + ex.message += "\n" + data + log.verbose(data, "bad json") + log.error("error parsing json", "registry") + return cb(ex, null, data, response) + } + } else if (data) { + parsed = data + data = JSON.stringify(parsed) + } + + // expect data with any error codes + if (!data && response.statusCode >= 400) { + return cb( response.statusCode + " " + + require("http").STATUS_CODES[response.statusCode] + , null, data, response ) + } + + var er = null + if (parsed && response.headers.etag) { + parsed._etag = response.headers.etag + } + + if (parsed && parsed.error && response.statusCode >= 400) { + var w = url.parse(where).pathname.substr(1) + if (!w.match(/^-/) && parsed.error === "not_found") { + w = w.split("/") + name = w[w.indexOf("_rewrite") + 1] + er = new Error("404 Not Found: "+name) + er.errno = npm.E404 + er.pkgid = name + } else { + er = new Error( + parsed.error + " " + (parsed.reason || "") + ": " + w) + } + } else if (method !== "HEAD" && method !== "GET") { + // invalidate cache + // This is irrelevant for commands that do etag caching, but + // ls and view also have a timed cache, so this keeps the user + // from thinking that it didn't work when it did. + // Note that failure is an acceptable option here, since the + // only result will be a stale cache for some helper commands. + var path = require("path") + , p = url.parse(where).pathname.split("/") + , _ = "/" + , caches = p.map(function (part) { + return _ = path.join(_, part) + }).map(function (cache) { + return path.join(npm.cache, cache, ".cache.json") + }) + + // if the method is DELETE, then also remove the thing itself. + // Note that the search index is probably invalid. Whatever. + // That's what you get for deleting stuff. Don't do that. + if (method === "DELETE") { + p = p.slice(0, p.indexOf("-rev")) + caches.push(path.join(npm.cache, p.join("/"))) + } + + asyncMap(caches, rm, function () {}) + } + return cb(er, parsed, data, response) +}} + +function GET (where, etag, nofollow, cb) { + regRequest("GET", where, null, etag, nofollow, cb) +} + +function PUT (where, what, etag, nofollow, cb) { + regRequest("PUT", where, what, etag, nofollow, cb) +} + +function upload (where, filename, etag, nofollow, cb) { + if (typeof nofollow === "function") cb = nofollow, nofollow = false + if (typeof etag === "function") cb = etag, etag = null + + fs.stat(filename, function (er, stat) { + if (er) return cb(er) + var s = fs.createReadStream(filename) + s.size = stat.size + s.on("error", cb) + + PUT(where, s, etag, nofollow, cb) + }) +} + +function reg () { + var r = npm.config.get("registry") + if (!r) { + return new Error("Must define registry URL before accessing registry.") + } + if (r.substr(-1) !== "/") r += "/" + npm.config.set("registry", r) + return r +} diff --git a/deps/npm/lib/utils/npm-registry-client/star.js b/deps/npm/lib/utils/npm-registry-client/star.js new file mode 100644 index 0000000000..474a1786d4 --- /dev/null +++ b/deps/npm/lib/utils/npm-registry-client/star.js @@ -0,0 +1,32 @@ + +module.exports = star + +var request = require("./request.js") + , GET = request.GET + , PUT = request.PUT + , log = require("../log.js") + , npm = require("../../npm.js") + +function star (package, starred, cb) { + var users = {} + + GET(package, function (er, fullData) { + if (er) return cb(er) + + fullData = { _id: fullData._id + , _rev: fullData._rev + , users: fullData.users || {} } + + if (starred) { + log.info("starring", fullData._id) + fullData.users[npm.config.get("username")] = true + log.verbose(fullData) + } else { + delete fullData.users[npm.config.get("username")] + log.info("unstarring", fullData._id) + log.verbose(fullData) + } + + return PUT(package, fullData, cb) + }) +} diff --git a/deps/npm/lib/utils/npm-registry-client/tag.js b/deps/npm/lib/utils/npm-registry-client/tag.js new file mode 100644 index 0000000000..4d88a98dba --- /dev/null +++ b/deps/npm/lib/utils/npm-registry-client/tag.js @@ -0,0 +1,8 @@ + +module.exports = tag + +var PUT = require("./request.js").PUT + +function tag (project, version, tag, cb) { + PUT(project+"/"+tag, JSON.stringify(version), cb) +} diff --git a/deps/npm/lib/utils/npm-registry-client/unpublish.js b/deps/npm/lib/utils/npm-registry-client/unpublish.js new file mode 100644 index 0000000000..0c7c449912 --- /dev/null +++ b/deps/npm/lib/utils/npm-registry-client/unpublish.js @@ -0,0 +1,98 @@ + +// fetch the data +// modify to remove the version in question +// If no versions remaining, then DELETE +// else, PUT the modified data +// delete the tarball + +module.exports = unpublish + +var request = require("./request.js") + , log = require("../log.js") + , get = require("./get.js") + , semver = require("semver") + , url = require("url") + , chain = require("slide").chain + +function unpublish (name, ver, cb) { + if (!cb) cb = ver, ver = null + if (!cb) throw new Error( + "Not enough arguments for registry unpublish") + + get(name, null, -1, true, function (er, data) { + if (er) return log(name+" not published", "unpublish", cb) + // remove all if no version specified + if (!ver) { + log("No version specified, removing all", "unpublish") + return request("DELETE", name+'/-rev/'+data._rev, cb) + } + + var versions = data.versions || {} + , versionPublic = versions.hasOwnProperty(ver) + + if (!versionPublic) log(name+"@"+ver+" not published", "unpublish") + else { + var dist = versions[ver].dist + log.verbose(dist, "removing attachments") + } + + delete versions[ver] + // if it was the only version, then delete the whole package. + if (!Object.keys(versions).length) { + log("No versions remain, removing entire package", "unpublish") + return request("DELETE", name+"/-rev/"+data._rev, cb) + } + + if (!versionPublic) return cb() + + var latestVer = data["dist-tags"].latest + for (var tag in data["dist-tags"]) { + if (data["dist-tags"][tag] === ver) delete data["dist-tags"][tag] + } + + if (latestVer === ver) { + data["dist-tags"].latest = + Object.getOwnPropertyNames(versions).sort(semver.compare).pop() + } + + var rev = data._rev + delete data._revisions + delete data._attachments + // log(data._rev, "rev") + request.PUT(name+"/-rev/"+rev, data, + log.er(detacher(data, dist, cb), "Failed to update the data")) + }) +} + +function detacher (data, dist, cb) { return function (er) { + if (er) return cb(er) + get(data.name, function (er, data) { + if (er) return cb(er) + + var tb = url.parse(dist.tarball) + + detach(data, tb.pathname, data._rev, function (er) { + if (er || !dist.bin) return cb(er) + chain(Object.keys(dist.bin).map(function (bt) { + return function (cb) { + var d = dist.bin[bt] + detach(data, url.parse(d.tarball).pathname, null, cb) + } + }), cb) + }) + }) +}} + +function detach (data, path, rev, cb) { + if (rev) { + path += "/-rev/" + rev + log(path, "detach") + return request("DELETE", path, cb) + } + get(data.name, function (er, data) { + rev = data._rev + if (!rev) return cb(new Error( + "No _rev found in "+data._id)) + detach(data, path, rev, cb) + }) +} diff --git a/deps/npm/lib/utils/output.js b/deps/npm/lib/utils/output.js new file mode 100644 index 0000000000..00da9f69f7 --- /dev/null +++ b/deps/npm/lib/utils/output.js @@ -0,0 +1,156 @@ + +// centralized stdout writer. + +exports.doColor = doColor +exports.write = write + +var npm = require("../npm.js") + , tty = require("tty") + , streams = {} + , ttys = {} + , net = require("net") + , util = require("util") + , deadStreams = {} + +function doColor (stream) { + var conf = npm.config.get("color") + return (!conf) ? false + : (conf === "always") ? true + : isatty(stream) +} +function isatty (stream) { + // console.error("isatty?", stream) + if (!tty.isatty) return true + if (!stream) return false + if (stream.isTTY) return true + if (stream && (typeof stream.fd === "number")) { + stream.isTTY = tty.isatty(stream.fd) + } + return stream.isTTY +} + +function write (args, stream, lf, cb) { + // console.error("write", [args, stream, lf, cb]) + if (typeof cb !== "function" && typeof lf === "function") { + cb = lf + lf = null + } + if (typeof cb !== "function" && typeof stream === "function") { + cb = stream + stream = npm.config.get("outfd") + } + + stream = getStream(stream) + // console.error("gotStream", stream) + if (lf == null) lf = isatty(stream) + if (!stream) return cb && cb(), false + if (!Array.isArray(args)) args = [args] + + // console.error("write", args) + + var msg = "" + , colored = doColor(stream) + msg = args.map(function (arg) { + if (typeof arg !== "string") { + return util.inspect(arg, false, 5, colored) + "\n" + } + if (!colored) arg = arg.replace(/\033\[[0-9;]*m/g, '') + if (!npm.config.get("unicode")) { + arg = arg.replace(/└/g, "`") + .replace(/─/g, "-") + .replace(/├/g, "+") + .replace(/┬/g, "-") + } + return arg + }).join(" ") + + // listen to the "output" event to cancel/modify/redirect + npm.output = {stream:stream, message:msg} + npm.emit("output", npm.output) + if (!npm.output) return cb && cb(), false // cancelled + stream = npm.output.stream + msg = npm.output.message + + // EPIPE errors just mean that the stream is not listening + // any more. Mark the stream as dead, and return. + if (deadStreams[stream.fd]) { + return cb && cb(), false + } + if (!deadStreams.hasOwnProperty(stream.fd)) { + deadStreams[stream.fd] = false + stream.on("error", function (er) { + if (er.code === "EPIPE") { + deadStreams[stream.fd] = true + return cb && cb() + } + if (stream.listeners("error").length === 1) { + throw er + } + }) + } + + // use the \r\n in case we're in raw mode. + msg = msg.split(/\r?\n/).concat("").join(lf ? "\r\n" : "\n") + // output to stderr should be synchronous + if (stream === process.stderr || stream.fd === 2) { + process.stderr.write(msg) + if (cb) cb() + return true + } + // console.error("writing ", msg) + var flushed = stream.write(msg) + if (flushed && cb) { + process.nextTick(cb) + } else if (cb) { + stream.once("drain", cb) + } + return flushed +} + +var hadError = false +function getStream (fd) { + if (hadError) return + + var stream + if (!fd && fd !== 0) return + if (typeof fd === "string") fd = +fd + + // console.error("getStream", fd, hadError) + + if (fd && typeof fd === "object") { + stream = fd + fd = fd.fd + } else if (streams[fd]) { + stream = streams[fd] + } else { + switch (fd) { + case 1: + stream = process.stdout + stream.fd = fd + stream.writable = true + break + case 2: + stream = process.stderr + stream.fd = fd + stream.writable = true + break + default: + try { + stream = new net.Stream(fd) + if (!stream || !stream.writable) { + throw new Error("Stream not writable") + } + } catch (ex) { + // if this fails, then regular logging is most likely broken. + var er = new Error("cannot output to fd "+fd + ": "+ + (ex.stack || ex.message).substr(7) + "\n") + console.error(er.stack) + hadError = true + process.exit(1) + } + } + } + + if (!stream || !stream.writable) return + return streams[fd] = stream +} diff --git a/deps/npm/lib/utils/promise-chain.js b/deps/npm/lib/utils/promise-chain.js new file mode 100644 index 0000000000..c206c91ecb --- /dev/null +++ b/deps/npm/lib/utils/promise-chain.js @@ -0,0 +1,39 @@ + +module.exports = promiseChain + +// usage: +// +// promiseChain(cb) <-- this is the callback for eventual success or error +// ( fn, [arg, arg, arg], function (a,b,c) { success(a,b,c) }) +// ( fn2, [args] ) +// () <-- this kicks it off. +// +// promiseChain.call(someObj, cb) <-- bind this-context for all functions + +function promiseChain (cb) { + var steps = [] + , vals = [] + , context = this + function go () { + var step = steps.shift() + if (!step) return cb() + try { step[0].apply(context, step[1]) } + catch (ex) { cb(ex) } + } + return function pc (fn, args, success) { + if (arguments.length === 0) return go() + // add the step + steps.push + ( [ fn + , (args || []).concat([ function (er) { + if (er) return cb(er) + var a = Array.prototype.slice.call(arguments, 1) + try { success && success.apply(context, a) } + catch (ex) { return cb(ex) } + go() + }]) + ] + ) + return pc + } +} diff --git a/deps/npm/lib/utils/prompt.js b/deps/npm/lib/utils/prompt.js new file mode 100644 index 0000000000..d23e4ac587 --- /dev/null +++ b/deps/npm/lib/utils/prompt.js @@ -0,0 +1,69 @@ + +module.exports = prompt + +var log = require("./log.js") + , buffer = "" + , tty = require("tty") + +function prompt (p, def, silent, cb) { + if (!cb) cb = silent, silent = false + if (!cb) cb = def, def = null + if (def) p += "("+(silent ? "" : def)+") " + var r = (silent ? silentRead : read).bind(null, def, cb) + if (!process.stdout.write(p)) process.stdout.on("drain", function D () { + process.stdout.removeListener("drain", D) + r() + }) + else r() +} + +function read (def, cb) { + var stdin = process.openStdin() + , val = "" + stdin.resume() + stdin.setEncoding("utf8") + stdin.on("error", cb) + stdin.on("data", function D (chunk) { + val += buffer + chunk + buffer = "" + val = val.replace(/\r/g, '') + if (val.indexOf("\n") !== -1) { + if (val !== "\n") val = val.replace(/^\n+/, "") + buffer = val.substr(val.indexOf("\n")) + val = val.substr(0, val.indexOf("\n")) + stdin.pause() + stdin.removeListener("data", D) + stdin.removeListener("error", cb) + val = val.trim() || def + cb(null, val) + } + }) +} + +function silentRead (def, cb) { + var stdin = process.openStdin() + , val = "" + tty.setRawMode(true) + stdin.resume() + stdin.on("error", cb) + stdin.on("data", function D (c) { + c = "" + c + switch (c) { + case "\n": case "\r": case "\r\n": case "\u0004": + tty.setRawMode(false) + stdin.removeListener("data", D) + stdin.removeListener("error", cb) + val = val.trim() || def + process.stdout.write("\n") + stdin.pause() + return cb(null, val) + case "\u0003": case "\0": + return cb("cancelled") + break + default: + val += buffer + c + buffer = "" + break + } + }) +} diff --git a/deps/npm/lib/utils/read-installed.js b/deps/npm/lib/utils/read-installed.js new file mode 100644 index 0000000000..59fc6c490b --- /dev/null +++ b/deps/npm/lib/utils/read-installed.js @@ -0,0 +1,295 @@ + +// Walk through the file-system "database" of installed +// packages, and create a data object related to the +// installed versions of each package. + +/* +This will traverse through all node_modules folders, +resolving the dependencies object to the object corresponding to +the package that meets that dep, or just the version/range if +unmet. + +Assuming that you had this folder structure: + +/path/to ++-- package.json { name = "root" } +`-- node_modules + +-- foo {bar, baz, asdf} + | +-- node_modules + | +-- bar { baz } + | `-- baz + `-- asdf + +where "foo" depends on bar, baz, and asdf, bar depends on baz, +and bar and baz are bundled with foo, whereas "asdf" is at +the higher level (sibling to foo), you'd get this object structure: + +{ +, path: "/path/to" +, parent: null +, dependencies: + { foo : + { version: "1.2.3" + , path: "/path/to/node_modules/foo" + , parent: + , dependencies: + { bar: + { parent: + , path: "/path/to/node_modules/foo/node_modules/bar" + , version: "2.3.4" + , dependencies: { baz: } + } + , baz: { ... } + , asdf: + } + } + , asdf: { ... } + } +} + +Unmet deps are left as strings. +Extraneous deps are marked with extraneous:true +deps that don't meet a requirement are marked with invalid:true + +to READ(packagefolder, parentobj, name, reqver) +obj = read package.json +installed = ./node_modules/* +if parentobj is null, and no package.json + obj = {dependencies:{:"*"}} +deps = Object.keys(obj.dependencies) +obj.path = packagefolder +obj.parent = parentobj +if name, && obj.name !== name, obj.invalid = true +if reqver, && obj.version !satisfies reqver, obj.invalid = true +if !reqver && parentobj, obj.extraneous = true +for each folder in installed + obj.dependencies[folder] = READ(packagefolder+node_modules+folder, + obj, folder, obj.dependencies[folder]) +# walk tree to find unmet deps +for each dep in obj.dependencies not in installed + r = obj.parent + while r + if r.dependencies[dep] + if r.dependencies[dep].verion !satisfies obj.dependencies[dep] + WARN + r.dependencies[dep].invalid = true + obj.dependencies[dep] = r.dependencies[dep] + r = null + else r = r.parent +return obj + + +TODO: +1. Find unmet deps in parent directories, searching as node does up +as far as the left-most node_modules folder. +2. Ignore anything in node_modules that isn't a package folder. + +*/ + + +var npm = require("../npm.js") + , fs = require("graceful-fs") + , path = require("path") + , asyncMap = require("slide").asyncMap + , semver = require("semver") + , readJson = require("./read-json.js") + , log = require("./log.js") + +module.exports = readInstalled + +function readInstalled (folder, cb) { + var d = npm.config.get("depth") + readInstalled_(folder, null, null, null, 0, d, function (er, obj) { + if (er) return cb(er) + // now obj has all the installed things, where they're installed + // figure out the inheritance links, now that the object is built. + resolveInheritance(obj) + cb(null, obj) + }) +} + +var rpSeen = {} +function readInstalled_ (folder, parent, name, reqver, depth, maxDepth, cb) { + //console.error(folder, name) + + var installed + , obj + , real + , link + + fs.readdir(path.resolve(folder, "node_modules"), function (er, i) { + // error indicates that nothing is installed here + if (er) i = [] + installed = i.filter(function (f) { return f.charAt(0) !== "." }) + next() + }) + + readJson(path.resolve(folder, "package.json"), function (er, data) { + obj = data + if (!parent) { + obj = obj || true + er = null + } + return next(er) + }) + + fs.lstat(folder, function (er, st) { + if (er) { + if (!parent) real = true + return next(er) + } + fs.realpath(folder, function (er, rp) { + //console.error("realpath(%j) = %j", folder, rp) + real = rp + if (st.isSymbolicLink()) link = rp + next(er) + }) + }) + + var errState = null + , called = false + function next (er) { + if (errState) return + if (er) { + errState = er + return cb(null, []) + } + //console.error('next', installed, obj && typeof obj, name, real) + if (!installed || !obj || !real || called) return + called = true + if (rpSeen[real]) return cb(null, rpSeen[real]) + if (obj === true) { + obj = {dependencies:{}, path:folder} + installed.forEach(function (i) { obj.dependencies[i] = "*" }) + } + if (name && obj.name !== name) obj.invalid = true + obj.realName = name || obj.name + obj.dependencies = obj.dependencies || {} + + // "foo":"http://blah" is always presumed valid + if (reqver + && semver.validRange(reqver) + && !semver.satisfies(obj.version, reqver)) { + obj.invalid = true + } + + if (parent + && !(name in parent.dependencies) + && !(name in (parent.devDependencies || {}))) { + obj.extraneous = true + } + obj.path = obj.path || folder + obj.realPath = real + obj.link = link + if (parent && !obj.link) obj.parent = parent + rpSeen[real] = obj + obj.depth = depth + if (depth >= maxDepth) return cb(null, obj) + asyncMap(installed, function (pkg, cb) { + var rv = obj.dependencies[pkg] + if (!rv && obj.devDependencies) rv = obj.devDependencies[pkg] + readInstalled_( path.resolve(folder, "node_modules/"+pkg) + , obj, pkg, obj.dependencies[pkg], depth + 1, maxDepth + , cb ) + }, function (er, installedData) { + if (er) return cb(er) + installedData.forEach(function (dep) { + obj.dependencies[dep.realName] = dep + }) + return cb(null, obj) + }) + } +} + +// starting from a root object, call findUnmet on each layer of children +var riSeen = [] +function resolveInheritance (obj) { + if (typeof obj !== "object") return + if (riSeen.indexOf(obj) !== -1) return + riSeen.push(obj) + if (typeof obj.dependencies !== "object") { + obj.dependencies = {} + } + Object.keys(obj.dependencies).forEach(function (dep) { + findUnmet(obj.dependencies[dep]) + }) + Object.keys(obj.dependencies).forEach(function (dep) { + resolveInheritance(obj.dependencies[dep]) + }) +} + +// find unmet deps by walking up the tree object. +// No I/O +var fuSeen = [] +function findUnmet (obj) { + if (fuSeen.indexOf(obj) !== -1) return + fuSeen.push(obj) + //console.error("find unmet", obj.name, obj.parent && obj.parent.name) + var deps = obj.dependencies = obj.dependencies || {} + //console.error(deps) + Object.keys(deps) + .filter(function (d) { return typeof deps[d] === "string" }) + .forEach(function (d) { + //console.error("find unmet", obj.name, d, deps[d]) + var r = obj.parent + , found = null + while (r && !found && typeof deps[d] === "string") { + // if r is a valid choice, then use that. + found = r.dependencies[d] + if (!found && r.realName === d) found = r + + if (!found) { + r = r.link ? null : r.parent + continue + } + if ( typeof deps[d] === "string" + && !semver.satisfies(found.version, deps[d])) { + // the bad thing will happen + log.warn(obj.path + " requires "+d+"@'"+deps[d] + +"' but will load\n" + +found.path+",\nwhich is version "+found.version + ,"unmet dependency") + found.invalid = true + } + deps[d] = found + } + }) + log.verbose([obj._id], "returning") + return obj +} + +if (module === require.main) { + var util = require("util") + console.error("testing") + + var called = 0 + readInstalled(process.cwd(), function (er, map) { + console.error(called ++) + if (er) return console.error(er.stack || er.message) + cleanup(map) + console.error(util.inspect(map, true, 10, true)) + }) + + var seen = [] + function cleanup (map) { + if (seen.indexOf(map) !== -1) return + seen.push(map) + for (var i in map) switch (i) { + case "_id": + case "path": + case "extraneous": case "invalid": + case "dependencies": case "name": + continue + default: delete map[i] + } + var dep = map.dependencies +// delete map.dependencies + if (dep) { +// map.dependencies = dep + for (var i in dep) if (typeof dep[i] === "object") { + cleanup(dep[i]) + } + } + return map + } +} diff --git a/deps/npm/lib/utils/read-json.js b/deps/npm/lib/utils/read-json.js new file mode 100644 index 0000000000..e71128c1b1 --- /dev/null +++ b/deps/npm/lib/utils/read-json.js @@ -0,0 +1,496 @@ + +module.exports = readJson +readJson.processJson = processJson +readJson.unParsePeople = unParsePeople +readJson.parsePeople = parsePeople +readJson.clearCache = clearCache + +var fs = require("graceful-fs") + , semver = require("semver") + , path = require("path") + , log = require("./log.js") + , npm = require("../npm.js") + , cache = {} + , timers = {} + , loadPackageDefaults = require("./load-package-defaults.js") + +function readJson (jsonFile, opts, cb) { + if (typeof cb !== "function") cb = opts, opts = {} + if (cache.hasOwnProperty(jsonFile)) { + log.verbose(jsonFile, "from cache") + return cb(null, cache[jsonFile]) + } + opts.file = jsonFile + if (!opts.tag) { + var parsedPath = jsonFile.indexOf(npm.dir) === 0 && jsonFile.match( + /\/([^\/]+)\/([^\/]+)\/package\/package\.json$/) + if (parsedPath && semver.valid(parsedPath[2])) { + // this is a package.json in some installed package. + // infer the opts.tag so that linked packages behave right. + opts.tag = parsedPath[2] + } + } + + var wscript = null + , contributors = null + , serverjs = null + + if (opts.wscript != null) { + wscript = opts.wscript + next() + } else fs.readFile( path.join(path.dirname(jsonFile), "wscript") + , function (er, data) { + if (er) opts.wscript = false + else opts.wscript = !!(data.toString().match(/(^|\n)def build\b/) + && data.toString().match(/(^|\n)def configure\b/)) + wscript = opts.wscript + next() + }) + + if (opts.contributors != null) { + contributors = opts.contributors + next() + } else fs.readFile( path.join(path.dirname(jsonFile), "AUTHORS") + , function (er, data) { + if (er) opts.contributors = false + else { + data = data.toString().split(/\r?\n/).map(function (l) { + l = l.trim().split("#").shift() + return l + }).filter(function (l) { return l }) + opts.contributors = data + } + contributors = opts.contributors + next() + }) + + if (opts.serverjs != null) { + serverjs = opts.serverjs + next() + } else fs.stat( path.join(path.dirname(jsonFile), "server.js") + , function (er, st) { + if (er) opts.serverjs = false + else opts.serverjs = st.isFile() + serverjs = opts.serverjs + next() + }) + + function next () { + if (wscript === null + || contributors === null + || serverjs === null) { + return + } + + fs.readFile(jsonFile, processJson(opts, function (er, data) { + if (er) return cb(er) + var doLoad = !(jsonFile.indexOf(npm.cache) === 0 && + path.basename(path.dirname(jsonFile)) !== "package") + if (!doLoad) return cb(er, data) + loadPackageDefaults(data, path.dirname(jsonFile), cb) + })) + } +} + +function processJson (opts, cb) { + if (typeof cb !== "function") cb = opts, opts = {} + if (typeof cb !== "function") { + var thing = cb, cb = null + return P(null, thing) + } else return P + + function P (er, thing) { + if (er) { + if (cb) return cb(er, thing) + throw er + } + if (typeof thing === "object" && !Buffer.isBuffer(thing)) { + return processObject(opts, cb)(er, thing) + } else { + return processJsonString(opts, cb)(er, thing) + } + } +} + +function processJsonString (opts, cb) { return function (er, jsonString) { + jsonString += "" + if (er) return cb(er, jsonString) + var json + try { + json = JSON.parse(jsonString) + } catch (ex) { + if (opts.file && opts.file.indexOf(npm.dir) === 0) { + try { + json = require("vm").runInNewContext("(\n"+jsonString+"\n)") + log.error(opts.file, "Error parsing json") + log.error(ex, "parse error ") + } catch (ex2) { + return jsonParseFail(ex, opts.file, cb) + } + } else { + return jsonParseFail(ex, opts.file, cb) + } + } + return processObject(opts, cb)(er, json) +}} + + +function jsonParseFail (ex, file, cb) { + var e = new Error( + "Failed to parse json\n"+ex.message) + e.errno = npm.EJSONPARSE + e.file = file + if (cb) return cb(e) + throw e +} + +// a warning for deprecated or likely-incorrect fields +var typoWarned = {} +function typoWarn (json) { + if (typoWarned[json._id]) return + typoWarned[json._id] = true + + if (json.modules) { + log.warn("package.json: 'modules' object is deprecated", json._id) + delete json.modules + } + + // http://registry.npmjs.org/-/fields + var typos = { "dependancies": "dependencies" + , "dependecies": "dependencies" + , "depdenencies": "dependencies" + , "devEependencies": "devDependencies" + , "depends": "dependencies" + , "devDependences": "devDependencies" + , "devDepenencies": "devDependencies" + , "devdependencies": "devDependencies" + , "repostitory": "repository" + , "prefereGlobal": "preferGlobal" + , "hompage": "homepage" + , "hampage": "homepage" // XXX maybe not a typo, just delicious? + , "autohr": "author" + , "autor": "author" + , "contributers": "contributors" + , "publicationConfig": "publishConfig" + } + + Object.keys(typos).forEach(function (d) { + if (json.hasOwnProperty(d)) { + log.warn( "package.json: '" + d + "' should probably be '" + + typos[d] + "'", json._id) + } + }) + + // bugs typos + var bugsTypos = { "web": "url" + , "name": "url" + } + + if (typeof json.bugs === "object") { + Object.keys(bugsTypos).forEach(function (d) { + if (json.bugs.hasOwnProperty(d)) { + log.warn( "package.json: bugs['" + d + "'] should probably be " + + "bugs['" + bugsTypos[d] + "']", json._id) + } + }) + } + + // script typos + var scriptTypos = { "server": "start" } + if (json.scripts) Object.keys(scriptTypos).forEach(function (d) { + if (json.scripts.hasOwnProperty(d)) { + log.warn( "package.json: scripts['" + d + "'] should probably be " + + "scripts['" + scriptTypos[d] + "']", json._id) + } + }) +} + + +function processObject (opts, cb) { return function (er, json) { + // json._npmJsonOpts = opts + // log.warn(json, "processing json") + if (npm.config.get("username")) { + json._npmUser = { name: npm.config.get("username") + , email: npm.config.get("email") } + } + + // slashes would be a security risk. + // anything else will just fail harmlessly. + if (!json.name) { + var e = new Error("No 'name' field found in package.json") + if (cb) return cb(e) + throw e + } + json.name = json.name.trim() + if (json.name.charAt(0) === "." || json.name.match(/[\/@\s\+%:]/)) { + var msg = "Invalid name: " + + JSON.stringify(json.name) + + " may not start with '.' or contain %/@+: or whitespace" + , e = new Error(msg) + if (cb) return cb(e) + throw e + } + if (json.name.toLowerCase() === "node_modules") { + var msg = "Invalid package name: node_modules" + , e = new Error(msg) + if (cb) return cb(e) + throw e + } + if (json.name.toLowerCase() === "favicon.ico") { + var msg = "Sorry, favicon.ico is a picture, not a package." + , e = new Error(msg) + if (cb) return cb(e) + throw e + } + + if (json.repostories) { + var msg = "'repositories' (plural) No longer supported.\n" + + "Please pick one, and put it in the 'repository' field." + , e = new Error(msg) + // uncomment once this is no longer an issue. + // if (cb) return cb(e) + // throw e + log.error(msg, "incorrect json: "+json.name) + json.repostory = json.repositories[0] + delete json.repositories + } + + if (json.repository) { + if (typeof json.repository === "string") { + json.repository = { type : "git" + , url : json.repository } + } + var repo = json.repository.url || "" + repo = repo.replace(/^(https?|git):\/\/[^\@]+\@github.com/ + ,'$1://github.com') + if (json.repository.type === "git" + && ( repo.match(/^https?:\/\/github.com/) + || repo.match(/github.com\/[^\/]+\/[^\/]+\/?$/) + && !repo.match(/\.git$/) + )) { + repo = repo.replace(/^https?:\/\/github.com/, 'git://github.com') + if (!repo.match(/\.git$/)) { + repo = repo.replace(/\/?$/, '.git') + } + } + if (repo.match(/github\.com\/[^\/]+\/[^\/]+\/?$/) + && repo.match(/\.git\.git$/)) { + log.warn(repo, "Probably broken git url") + } + json.repository.url = repo + } + + var files = json.files + if (files && !Array.isArray(files)) { + log.warn(files, "Invalid 'files' member. See 'npm help json'") + delete json.files + } + + var kw = json.keywords + if (typeof kw === "string") { + kw = kw.split(/,\s+/) + json.keywords = kw + } + + json._id = json.name+"@"+json.version + + var tag = opts.tag + if (tag) json.version = tag + + var scripts = json.scripts || {} + + // if it has a wscript, then build it. + if (opts.wscript && !json.prebuilt) { + log.verbose([json.prebuilt, opts], "has wscript") + if (!scripts.install && !scripts.preinstall) { + // don't fail if it was unexpected, just try. + scripts.preinstall = "node-waf clean || true; node-waf configure build" + json.scripts = scripts + } + } + + // if it has an AUTHORS, then credit them + if (opts.contributors && Array.isArray(opts.contributors) + && opts.contributors.length) { + json.contributors = opts.contributors + } + + // if it has a server.js, then start it. + if (opts.serverjs && !scripts.start) { + scripts.start = "node server.js" + json.scripts = scripts + } + + if (!(semver.valid(json.version))) { + var m + if (!json.version) { + m = "'version' field missing\n" + } else { + m = "Invalid 'version' field: "+json.version+"\n" + } + + m += "'version' Must be X.Y.Z, with an optional trailing tag.\n" + + "See the section on 'version' in `npm help json`" + + var e = new Error(m) + if (cb) return cb(e) + throw e + } + json.version = semver.clean(json.version) + + if (json.bin && typeof json.bin === "string") { + var b = {} + b[ json.name ] = json.bin + json.bin = b + } + + if (json.bundledDependencies && !json.bundleDependencies) { + json.bundleDependencies = json.bundledDependencies + delete json.bundledDependencies + } + + if (json.bundleDependencies && !Array.isArray(json.bundleDependencies)) { + var e = new Error("bundleDependencies must be an array.\n" + +"See `npm help json`") + if (cb) return cb(e) + throw e + } + + if (json["dev-dependencies"] && !json.devDependencies) { + json.devDependencies = json["dev-dependencies"] + delete json["dev-dependencies"] + } + + ;["dependencies", "devDependencies"].forEach(function (d) { + json[d] = json[d] ? depObjectify(json[d]) : {} + }) + + if (opts.dev || npm.config.get("dev") || npm.config.get("npat")) { + // log.warn(json._id, "Adding devdeps") + Object.keys(json.devDependencies || {}).forEach(function (d) { + json.dependencies[d] = json.devDependencies[d] + }) + // log.warn(json.dependencies, "Added devdeps") + } + + typoWarn(json) + + json = testEngine(json) + json = parsePeople(unParsePeople(json)) + if ( json.bugs ) json.bugs = parsePerson(unParsePerson(json.bugs)) + json._npmVersion = npm.version + json._nodeVersion = process.version + if (opts.file) { + log.verbose(opts.file, "caching") + cache[opts.file] = json + // arbitrary + var keys = Object.keys(cache) + , l = keys.length + if (l > 10000) for (var i = 0; i < l - 5000; i ++) { + delete cache[keys[i]] + } + } + if (cb) cb(null,json) + return json +}} + +function depObjectify (deps) { + if (!Array.isArray(deps)) return deps + var o = {} + deps.forEach(function (d) { + d = d.trim().split(/(:?[@\s><=])/) + o[d.shift()] = d.join("").trim().replace(/^@/, "") + }) + return o +} + +function testEngine (json) { + // if engines is empty, then assume that node is allowed. + if ( !json.engines + || Array.isArray(json.engines) + && !json.engines.length + || typeof json.engines === "object" + && !Object.keys(json.engines).length + ) { + json.engines = { "node" : "*" } + } + if (typeof json.engines === "string") { + if (semver.validRange(json.engines) !== null) { + json.engines = { "node" : json.engines } + } else json.engines = [ json.engines ] + } + + var nodeVer = npm.config.get("node-version") + , ok = false + if (nodeVer) nodeVer = nodeVer.replace(/\+$/, '') + if (Array.isArray(json.engines)) { + // Packages/1.0 commonjs style, with an array. + // hack it to just hang a "node" member with the version range, + // then do the npm-style check below. + for (var i = 0, l = json.engines.length; i < l; i ++) { + var e = json.engines[i].trim() + if (e.substr(0, 4) === "node") { + json.engines.node = e.substr(4) + } else if (e.substr(0, 3) === "npm") { + json.engines.npm = e.substr(3) + } + } + } + if (json.engines.node === "") json.engines.node = "*" + if (json.engines.node && null === semver.validRange(json.engines.node)) { + log.warn( json.engines.node + , "Invalid range in engines.node. Please see `npm help json`" ) + } + + if (nodeVer) { + json._engineSupported = semver.satisfies( nodeVer + , json.engines.node || "null" ) + } + if (json.engines.hasOwnProperty("npm") && json._engineSupported) { + json._engineSupported = semver.satisfies(npm.version, json.engines.npm) + } + return json +} + +function unParsePeople (json) { return parsePeople(json, true) } + +function parsePeople (json, un) { + var fn = un ? unParsePerson : parsePerson + if (json.author) json.author = fn(json.author) + ;["maintainers", "contributors"].forEach(function (set) { + if (Array.isArray(json[set])) json[set] = json[set].map(fn) + }) + return json +} + +function unParsePerson (person) { + if (typeof person === "string") return person + var name = person.name || "" + , u = person.url || person.web + , url = u ? (" ("+u+")") : "" + , e = person.email || person.mail + , email = e ? (" <"+e+">") : "" + return name+email+url +} + +function parsePerson (person) { + if (typeof person !== "string") return person + var name = person.match(/^([^\(<]+)/) + , url = person.match(/\(([^\)]+)\)/) + , email = person.match(/<([^>]+)>/) + , obj = {} + if (name && name[0].trim()) obj.name = name[0].trim() + if (email) obj.email = email[1] + if (url) obj.url = url[1] + return obj +} + +function clearCache (prefix) { + if (!prefix) { + cache = {} + return + } + Object.keys(cache).forEach(function (c) { + if (c.indexOf(prefix) === 0) delete cache[c] + }) +} diff --git a/deps/npm/lib/utils/relativize.js b/deps/npm/lib/utils/relativize.js new file mode 100644 index 0000000000..74c45c5d3e --- /dev/null +++ b/deps/npm/lib/utils/relativize.js @@ -0,0 +1,70 @@ + +module.exports = relativize + +// return the shortest path between two folders. +// if the original path is shorter, then use that, +// unless forceRelative is set to true. +var path = require("path") +function relativize (dest, src, forceRelative) { + var orig = dest + if (!isAbsolute(src)) forceRelative = true + else if (!isAbsolute(dest)) return false + src = path.resolve(src) + dest = path.resolve(dest) + if (src === dest) return "." + src = src.split(split) + dest = dest.split(split) + var i = 0 + while (src[i] === dest[i]) i++ + if (!forceRelative && i === 1) return orig // nothing in common + src.splice(0, i + 1) + var dots = [0, i, "."] + for (var i = 0, l = src.length; i < l; i ++) dots.push("..") + dest.splice.apply(dest, dots) + if (dest[0] === "." && dest[1] === "..") dest.shift() + dest = dest.join("/") + return !forceRelative && orig.length < dest.length ? orig : dest +} + +var split = process.platform === "win32" ? /[\/\\]/ : "/" + +function isAbsolute (p) { + if (process.platform !== "win32") return p.charAt(0) === "/" + return path.resolve(p) === p +} + +if (module === require.main) { + // from, to, result, relativeForced + var assert = require("assert") + + ; [ ["/bar" ,"/foo" ,"/bar" ,"./bar" ] + , ["/foo/baz" ,"/foo/bar/baz" ,"../baz" ,"../baz" ] + , ["/a/d" ,"/a/b/c/d/e/f" ,"/a/d" ,"../../../../d" ] + // trailing slashes are ignored. + , ["/a/d" ,"/a/b/c/d/e/" ,"/a/d" ,"../../../d" ] + , ["./foo/bar" ,"./foo/baz" ,"./bar" ,"./bar" ] + // force relative when the src is relative. + , ["./d" ,"./a/b/c/d/e" ,"../../../../d" ,"../../../../d" ] + // if src is abs and dest is relative, then fail + , ["./d" ,"/a/b" ,false ,false ] + ].forEach(function (test) { + var d = test[0] + , s = test[1] + , r = test[2] + , rr = test[3] + , ra = relativize(d, s) + , rra = relativize(d, s, true) + console.log([d, s, r, rr], [ra, rra], [r === ra, rr === rra]) + assert.equal(r, ra) + assert.equal(rr, rra) + if (!r) return + // contract: this is the relative path from absolute A to absolute B + var ad = path.resolve(d) + , as = path.resolve(s) + , dir = path.dirname(as) + assert.equal(path.resolve(dir, rr), ad) + assert.equal(path.resolve(dir, r), ad) + }) + + console.log("ok") +} diff --git a/deps/npm/lib/utils/set.js b/deps/npm/lib/utils/set.js new file mode 100644 index 0000000000..4d9241aab7 --- /dev/null +++ b/deps/npm/lib/utils/set.js @@ -0,0 +1,25 @@ + +module.exports = set +var get = require("./get.js") + , processJson = require("./read-json.js").processJson +function set (obj, key, val) { + for (var i in obj) { + if (i.toLowerCase() === key.toLowerCase()) return obj[i] = val + } + obj[key] = val + if (!val) return + // if it's a package set, then assign all the versions. + if (val.versions) return Object.keys(val.versions).forEach(function (v) { + if (typeof val.versions[v] !== "object") return + set(obj, key+"@"+v, val.versions[v]) + }) + // Note that this doesn't put the dist-tags there, only updates the versions + if (key === val.name+"@"+val.version) { + processJson(val) + var reg = get(obj, val.name) || {} + reg.name = reg._id = val.name + set(obj, val.name, reg) + reg.versions = get(reg, "versions") || {} + if (!get(reg.versions, val.version)) set(reg.versions, val.version, val) + } +} diff --git a/deps/npm/lib/utils/sha.js b/deps/npm/lib/utils/sha.js new file mode 100644 index 0000000000..17b8c38f9c --- /dev/null +++ b/deps/npm/lib/utils/sha.js @@ -0,0 +1,51 @@ + +var fs = require("graceful-fs") + , crypto = require("crypto") + , log = require("./log.js") + , binding + +try { binding = process.binding("crypto") } +catch (e) { binding = null } + +exports.check = check +exports.get = get + +function check (file, sum, cb) { + if (!binding) { + log.warn("crypto binding not found. Cannot verify shasum.", "shasum") + return cb() + } + get(file, function (er, actual) { + if (er) return log.er(cb, "Error getting shasum")(er) + var expected = sum.toLowerCase().trim() + , ok = actual === expected + cb(ok ? null : new Error( + "shasum check failed for "+file+"\n" + +"Expected: "+expected+"\n" + +"Actual: "+actual)) + }) +} + +function get (file, cb) { + if (!binding) { + log.warn("crypto binding not found. Cannot verify shasum.", "shasum") + return cb() + } + var h = crypto.createHash("sha1") + , s = fs.createReadStream(file) + , errState = null + s.on("error", function (er) { + if (errState) return + log.silly(er.stack || er.message, "sha error") + return cb(errState = er) + }).on("data", function (chunk) { + if (errState) return + log.silly(chunk.length, "updated sha bytes") + h.update(chunk) + }).on("end", function () { + if (errState) return + var actual = h.digest("hex").toLowerCase().trim() + log(actual+"\n"+file, "shasum") + cb(null, actual) + }) +} diff --git a/deps/npm/lib/utils/tar.js b/deps/npm/lib/utils/tar.js new file mode 100644 index 0000000000..d6db6ea528 --- /dev/null +++ b/deps/npm/lib/utils/tar.js @@ -0,0 +1,551 @@ +// XXX lib/cache.js and this file need to be rewritten. + +// commands for packing and unpacking tarballs +// this file is used by lib/cache.js + +var npm = require("../npm.js") + , fs = require("graceful-fs") + , exec = require("./exec.js") + , find = require("./find.js") + , mkdir = require("./mkdir-p.js") + , asyncMap = require("slide").asyncMap + , path = require("path") + , log = require("./log.js") + , uidNumber = require("./uid-number.js") + , rm = require("rimraf") + , readJson = require("./read-json.js") + , relativize = require("./relativize.js") + , cache = require("../cache.js") + , excludes = require("./excludes.js") + , myUid = process.getuid && process.getuid() + , myGid = process.getgid && process.getgid() + , tar = require("tar") + , zlib = require("zlib") + , fstream = require("fstream") + +exports.pack = pack +exports.unpack = unpack +exports.makeList = makeList + +function pack (targetTarball, folder, pkg, dfc, cb) { + if (typeof cb !== "function") cb = dfc, dfc = true + folder = path.resolve(process.cwd(), folder) + if (typeof pkg === "function") { + cb = pkg, pkg = null + return readJson(path.resolve(folder, "package.json"), function (er, pkg) { + if (er) return log.er(cb, "Couldn't find package.json in "+folder)(er) + pack(targetTarball, folder, pkg, dfc, cb) + }) + } + log.verbose(folder+" "+targetTarball, "pack") + var parent = path.dirname(folder) + , addFolder = path.basename(folder) + + var confEx = npm.config.get("ignore") + log.silly(folder, "makeList") + makeList(folder, pkg, dfc, function (er, files, cleanup) { + if (er) return cb(er) + return packFiles(targetTarball, parent, files, pkg, function (er) { + if (!cleanup || !cleanup.length) return cb(er) + // try to be a good citizen, even/especially in the event of failure. + cleanupResolveLinkDep(cleanup, function (er2) { + if (er || er2) { + if (er) log(er, "packing tarball") + if (er2) log(er2, "while cleaning up resolved deps") + } + return cb(er || er2) + }) + }) + }) +} + +function packFiles (targetTarball, parent, files, pkg, cb) { + + var p + + files = files.map(function (f) { + p = f.split(/\/|\\/)[0] + return path.resolve(parent, f) + }) + + parent = path.resolve(parent, p) + + log.verbose(targetTarball, "tarball") + log.verbose(parent, "parent") + fstream.Reader({ type: "Directory" + , path: parent + , filter: function () { + return -1 !== files.indexOf(this.path) + // || (this.type === "Directory" && + // this.basename !== ".git") + + } + }) + .on("error", log.er(cb, "error reading "+parent)) + .on("entry", function E (entry) { + entry.on("entry", E) + }) + .pipe(tar.Pack({})) + .on("error", log.er(cb, "tar creation error "+targetTarball)) + .pipe(zlib.Gzip()) + .on("error", log.er(cb, "gzip error "+targetTarball)) + .pipe(fstream.Writer({ type: "File", path: targetTarball })) + .on("error", log.er(cb, "Could not write "+targetTarball)) + .on("close", cb) +} + + +function unpack (tarball, unpackTarget, dMode, fMode, uid, gid, cb) { + if (typeof cb !== "function") cb = gid, gid = null + if (typeof cb !== "function") cb = uid, uid = null + if (typeof cb !== "function") cb = fMode, fMode = npm.modes.file + if (typeof cb !== "function") cb = dMode, dMode = npm.modes.exec + + uidNumber(uid, gid, function (er, uid, gid) { + if (er) return cb(er) + unpack_(tarball, unpackTarget, dMode, fMode, uid, gid, cb) + }) +} + +function unpack_ ( tarball, unpackTarget, dMode, fMode, uid, gid, cb ) { + // If the desired target is /path/to/foo, + // then unpack into /path/to/.foo.npm/{something} + // rename that to /path/to/foo, and delete /path/to/.foo.npm + var parent = path.dirname(unpackTarget) + , base = path.basename(unpackTarget) + , tmp = path.resolve(parent, "___" + base + ".npm") + + mkdir(tmp, dMode || npm.modes.exec, uid, gid, function (er) { + log.verbose([uid, gid], "unpack_ uid, gid") + log.verbose(unpackTarget, "unpackTarget") + if (er) return log.er(cb, "Could not create "+tmp)(er) + // cp the gzip of the tarball, pipe the stdout into tar's stdin + // gzip {tarball} --decompress --stdout \ + // | tar -mvxpf - --strip-components=1 -C {unpackTarget} + gunzTarPerm( tarball, tmp + , dMode, fMode + , uid, gid + , function (er, folder) { + if (er) return cb(er) + log.verbose(folder, "gunzed") + + rm(unpackTarget, function (er) { + if (er) return cb(er) + log.verbose(unpackTarget, "rm'ed") + fs.rename(folder, unpackTarget, function (er) { + if (er) return cb(er) + log.verbose([folder, unpackTarget], "renamed") + // curse you, nfs! It will lie and tell you that the + // mv is done, when in fact, it isn't. In theory, + // reading the file should cause it to wait until it's done. + readJson( path.resolve(unpackTarget, "package.json") + , function (er, data) { + // now we read the json, so we know it's there. + rm(tmp, function (er2) { cb(er || er2, data) }) + }) + }) + }) + }) + }) +} + +function gunzTarPerm (tarball, tmp, dMode, fMode, uid, gid, cb) { + if (!dMode) dMode = npm.modes.exec + if (!fMode) fMode = npm.modes.file + log.silly([dMode.toString(8), fMode.toString(8)], "gunzTarPerm modes") + + fs.createReadStream(tarball) + .pipe(zlib.Unzip()) + .on("error", log.er(cb, "unzip error")) + .pipe(tar.Extract({ type: "Directory", path: tmp })) + .on("error", log.er(cb, "Failed unpacking "+tarball)) + .on("close", afterUntar) + + // + // XXX Do all this in an Extract filter. + // + function afterUntar (er) { + // if we're not doing ownership management, + // then we're done now. + if (er) return log.er(cb, "Failed unpacking "+tarball)(er) + + // HACK skip on windows + if (npm.config.get("unsafe-perm") && process.platform !== "win32") { + uid = process.getuid() + gid = process.getgid() + if (uid === 0) { + if (process.env.SUDO_UID) uid = +process.env.SUDO_UID + if (process.env.SUDO_GID) gid = +process.env.SUDO_GID + } + } + + if (process.platform === "win32") { + return fs.readdir(tmp, function (er, files) { + files = files.filter(function (f) { + return f && f.indexOf("\0") === -1 + }) + cb(er, files && path.resolve(tmp, files[0])) + }) + } + + find(tmp, function (f) { + return f !== tmp + }, function (er, files) { + if (er) return cb(er) + asyncMap(files, function (f, cb) { + f = path.resolve(f) + log.silly(f, "asyncMap in gTP") + fs.lstat(f, function (er, stat) { + + if (er || stat.isSymbolicLink()) return cb(er) + if (typeof uid === "number" && typeof gid === "number") { + fs.chown(f, uid, gid, chown) + } else chown() + + function chown (er) { + if (er) return cb(er) + var mode = stat.isDirectory() ? dMode : fMode + , oldMode = stat.mode & 0777 + , newMode = (oldMode | mode) & (~npm.modes.umask) + if (mode && newMode !== oldMode) { + log.silly(newMode.toString(8), "chmod "+path.basename(f)) + fs.chmod(f, newMode, cb) + } else cb() + } + }) + }, function (er) { + + if (er) return cb(er) + if (typeof myUid === "number" && typeof myGid === "number") { + fs.chown(tmp, myUid, myGid, chown) + } else chown() + + function chown (er) { + if (er) return cb(er) + fs.readdir(tmp, function (er, folder) { + folder = folder && folder.filter(function (f) { + return f && !f.match(/^\._/) + }) + cb(er, folder && path.resolve(tmp, folder[0])) + }) + } + }) + }) + } +} + +function makeList (dir, pkg, dfc, cb) { + if (typeof cb !== "function") cb = dfc, dfc = true + if (typeof cb !== "function") cb = pkg, pkg = null + dir = path.resolve(dir) + + if (!pkg.path) pkg.path = dir + + var name = path.basename(dir) + + // since this is a top-level traversal, get the user and global + // exclude files, as well as the "ignore" config setting. + var confIgnore = npm.config.get("ignore").trim() + .split(/[\n\r\s\t]+/) + .filter(function (i) { return i.trim() }) + , userIgnore = npm.config.get("userignorefile") + , globalIgnore = npm.config.get("globalignorefile") + , userExclude + , globalExclude + + confIgnore.dir = dir + confIgnore.name = "confIgnore" + + var defIgnore = ["build/"] + defIgnore.dir = dir + + // TODO: only look these up once, and cache outside this function + excludes.parseIgnoreFile( userIgnore, null, dir + , function (er, uex) { + if (er) return cb(er) + userExclude = uex + next() + }) + + excludes.parseIgnoreFile( globalIgnore, null, dir + , function (er, gex) { + if (er) return cb(er) + globalExclude = gex + next() + }) + + function next () { + if (!globalExclude || !userExclude) return + var exList = [ defIgnore, confIgnore, globalExclude, userExclude ] + + makeList_(dir, pkg, exList, dfc, function (er, files, cleanup) { + if (er) return cb(er) + var dirLen = dir.length + 1 + files = files.map(function (file) { + return path.join(name, file.substr(dirLen)) + }) + return cb(null, files, cleanup) + }) + } +} + +// Patterns ending in slashes will only match targets +// ending in slashes. To implement this, add a / to +// the filename iff it lstats isDirectory() +function readDir (dir, pkg, dfc, cb) { + fs.readdir(dir, function (er, files) { + if (er) return cb(er) + files = files.filter(function (f) { + return f && f.charAt(0) !== "/" && f.indexOf("\0") === -1 + }) + asyncMap(files, function (file, cb) { + fs.lstat(path.resolve(dir, file), function (er, st) { + if (er) return cb(null, []) + // if it's a directory, then tack "/" onto the name + // so that it can match dir-only patterns in the + // include/exclude logic later. + if (st.isDirectory()) return cb(null, file + "/") + + // if it's a symlink, then we need to do some more + // complex stuff for GH-691 + if (st.isSymbolicLink()) return readSymlink(dir, file, pkg, dfc, cb) + + // otherwise, just let it on through. + return cb(null, file) + }) + }, cb) + }) +} + +// just see where this link is pointing, and resolve relative paths. +function shallowReal (link, cb) { + link = path.resolve(link) + fs.readlink(link, function (er, t) { + if (er) return cb(er) + return cb(null, path.resolve(path.dirname(link), t), t) + }) +} + +function readSymlink (dir, file, pkg, dfc, cb) { + var isNM = dfc + && path.basename(dir) === "node_modules" + && path.dirname(dir) === pkg.path + // see if this thing is pointing outside of the package. + // external symlinks are resolved for deps, ignored for other things. + // internal symlinks are allowed through. + var df = path.resolve(dir, file) + shallowReal(df, function (er, r, target) { + if (er) return cb(null, []) // wtf? exclude file. + if (r.indexOf(dir) === 0) return cb(null, file) // internal + if (!isNM) return cb(null, []) // external non-dep + // now the fun stuff! + fs.realpath(df, function (er, resolved) { + if (er) return cb(null, []) // can't add it. + readJson(path.resolve(resolved, "package.json"), function (er) { + if (er) return cb(null, []) // not a package + resolveLinkDep(dir, file, resolved, target, pkg, function (er, f, c) { + cb(er, f, c) + }) + }) + }) + }) +} + +// put the link back the way it was. +function cleanupResolveLinkDep (cleanup, cb) { + // cut it out of the list, so that cycles will be broken. + if (!cleanup) return cb() + + asyncMap(cleanup, function (d, cb) { + rm(d[1], function (er) { + if (er) return cb(er) + fs.symlink(d[0], d[1], cb) + }) + }, cb) +} + +function resolveLinkDep (dir, file, resolved, target, pkg, cb) { + // we've already decided that this is a dep that will be bundled. + // make sure the data reflects this. + var bd = pkg.bundleDependencies || pkg.bundledDependencies || [] + delete pkg.bundledDependencies + pkg.bundleDependencies = bd + var f = path.resolve(dir, file) + , cleanup = [[target, f, resolved]] + + if (bd.indexOf(file) === -1) { + // then we don't do this one. + // just move the symlink out of the way. + return rm(f, function (er) { + cb(er, file, cleanup) + }) + } + + rm(f, function (er) { + if (er) return cb(er) + cache.add(resolved, function (er, data) { + if (er) return cb(er) + cache.unpack(data.name, data.version, f, function (er, data) { + if (er) return cb(er) + // now clear out the cache entry, since it's weird, probably. + // pass the cleanup object along so that the thing getting the + // list of files knows what to clean up afterwards. + cache.clean([data._id], function (er) { cb(er, file, cleanup) }) + }) + }) + }) +} + +// exList is a list of ignore lists. +// Each exList item is an array of patterns of files to ignore +// +function makeList_ (dir, pkg, exList, dfc, cb) { + var files = null + , cleanup = null + + readDir(dir, pkg, dfc, function (er, f, c) { + if (er) return cb(er) + cleanup = c + files = f.map(function (f) { + // no nulls in paths! + return f.split(/\0/)[0] + }).filter(function (f) { + // always remove all source control folders and + // waf/vim/OSX garbage. this is a firm requirement. + return !( f === ".git/" + || f === ".lock-wscript" + || f === "CVS/" + || f === ".svn/" + || f === ".hg/" + || f.match(/^\..*\.swp/) + || f === ".DS_Store" + || f.match(/^\._/) + || f === "npm-debug.log" + || f === "" + || f.charAt(0) === "/" + ) + }) + + // if (files.length > 0) files.push(".") + + if (files.indexOf("package.json") !== -1 && dir !== pkg.path) { + // a package.json file starts the whole exclude/include + // logic all over. Otherwise, a parent could break its + // deps with its files list or .npmignore file. + readJson(path.resolve(dir, "package.json"), function (er, data) { + if (!er && typeof data === "object") { + data.path = dir + return makeList(dir, data, dfc, function (er, files) { + // these need to be mounted onto the directory now. + cb(er, files && files.map(function (f) { + return path.resolve(path.dirname(dir), f) + })) + }) + } + next() + }) + //next() + } else next() + + // add a local ignore file, if found. + if (files.indexOf(".npmignore") === -1 + && files.indexOf(".gitignore") === -1) next() + else { + excludes.addIgnoreFile( path.resolve(dir, ".npmignore") + , ".gitignore" + , exList + , dir + , function (er, list) { + if (!er) exList = list + next(er) + }) + } + }) + + var n = 2 + , errState = null + function next (er) { + if (errState) return + if (er) return cb(errState = er, [], cleanup) + if (-- n > 0) return + + if (!pkg) return cb(new Error("No package.json file in "+dir)) + if (pkg.path === dir && pkg.files) { + pkg.files = pkg.files.filter(function (f) { + f = f.trim() + return f && f.charAt(0) !== "#" + }) + if (!pkg.files.length) pkg.files = null + } + if (pkg.path === dir && pkg.files) { + // stuff on the files list MUST be there. + // ignore everything, then include the stuff on the files list. + var pkgFiles = ["*"].concat(pkg.files.map(function (f) { + return "!" + f + })) + pkgFiles.dir = dir + pkgFiles.packageFiles = true + exList.push(pkgFiles) + } + + if (path.basename(dir) === "node_modules" + && pkg.path === path.dirname(dir) + && dfc) { // do fancy crap + files = filterNodeModules(files, pkg) + } else { + // If a directory is excluded, we still need to be + // able to *include* a file within it, and have that override + // the prior exclusion. + // + // This whole makeList thing probably needs to be rewritten + files = files.filter(function (f) { + return excludes.filter(dir, exList)(f) || f.slice(-1) === "/" + }) + } + + + asyncMap(files, function (file, cb) { + // if this is a dir, then dive into it. + // otherwise, don't. + file = path.resolve(dir, file) + + // in 0.6.0, fs.readdir can produce some really odd results. + // XXX: remove this and make the engines hash exclude 0.6.0 + if (file.indexOf(dir) !== 0) { + return cb(null, []) + } + + fs.lstat(file, function (er, st) { + if (er) return cb(er) + if (st.isDirectory()) { + return makeList_(file, pkg, exList, dfc, cb) + } + return cb(null, file) + }) + }, function (er, files, c) { + if (c) cleanup = (cleanup || []).concat(c) + if (files.length > 0) files.push(dir) + return cb(er, files, cleanup) + }) + } +} + +// only include node_modules folder that are: +// 1. not on the dependencies list or +// 2. on the "bundleDependencies" list. +function filterNodeModules (files, pkg) { + var bd = pkg.bundleDependencies || pkg.bundledDependencies || [] + , deps = Object.keys(pkg.dependencies || {}) + .filter(function (key) { return !pkg.dependencies[key].extraneous }) + .concat(Object.keys(pkg.devDependencies || {})) + + delete pkg.bundledDependencies + pkg.bundleDependencies = bd + + return files.filter(function (f) { + f = f.replace(/\/$/, "") + return f.charAt(0) !== "." + && f.charAt(0) !== "_" + && bd.indexOf(f) !== -1 + }) +} diff --git a/deps/npm/lib/utils/uid-number.js b/deps/npm/lib/utils/uid-number.js new file mode 100644 index 0000000000..3756275534 --- /dev/null +++ b/deps/npm/lib/utils/uid-number.js @@ -0,0 +1,55 @@ +module.exports = uidNumber + +// This module calls into bin/npm-get-uid-gid.js, which sets the +// uid and gid to the supplied argument, in order to find out their +// numeric value. This can't be done in the main node process, +// because otherwise npm would be running as that user. + +var exec = require("./exec.js") + , path = require("path") + , log = require("./log.js") + , constants = require("constants") + , npm = require("../npm.js") + , uidSupport = process.getuid && process.setuid + , uidCache = {} + , gidCache = {} + +function uidNumber (uid, gid, cb) { + if (!uidSupport || npm.config.get("unsafe-perm")) return cb() + if (typeof cb !== "function") cb = gid, gid = null + if (typeof cb !== "function") cb = uid, uid = null + if (gid == null) gid = process.getgid() + if (uid == null) uid = process.getuid() + if (!isNaN(gid)) gid = +gid + if (!isNaN(uid)) uid = +uid + + if (uidCache[uid]) uid = uidCache[uid] + if (gidCache[gid]) gid = gidCache[gid] + + if (typeof gid === "number" && typeof uid === "number") { + return cb(null, uid, gid) + } + + var getter = path.join(__dirname, "..", "..", "bin", "npm-get-uid-gid.js") + return exec( process.execPath, [getter, uid, gid], process.env, false + , null, process.getuid(), process.getgid() + , function (er, code, out, err) { + if (er) return log.er(cb, "Could not get uid/gid "+err)(er) + log.silly(out, "output from getuid/gid") + out = JSON.parse(out+"") + if (out.error) { + if (!npm.config.get("unsafe-perm")) { + var er = new Error(out.error) + er.errno = out.errno + return cb(er) + } else { + return cb(null, +process.getuid(), +process.getgid()) + } + } + if (isNaN(out.uid) || isNaN(out.gid)) return cb(new Error( + "Could not get uid/gid: "+JSON.stringify(out))) + uidCache[uid] = out.uid + uidCache[gid] = out.gid + cb(null, out.uid, out.gid) + }) +} diff --git a/deps/npm/lib/version.js b/deps/npm/lib/version.js new file mode 100644 index 0000000000..e626e474a3 --- /dev/null +++ b/deps/npm/lib/version.js @@ -0,0 +1,66 @@ +// npm version + +module.exports = version + +var exec = require("./utils/exec.js") + , readJson = require("./utils/read-json.js") + , semver = require("semver") + , path = require("path") + , fs = require("graceful-fs") + , chain = require("slide").chain + , log = require("./utils/log.js") + , npm = require("./npm.js") + +version.usage = "npm version [--message commit-message]" + + "\n(run in package dir)\n" + + "'npm -v' or 'npm --version' to print npm version " + + "("+npm.version+")\n" + + "'npm view version' to view a package's " + + "published version\n" + + "'npm ls' to inspect current package/dependency versions" + +function version (args, cb) { + if (args.length !== 1) return cb(version.usage) + readJson(path.join(process.cwd(), "package.json"), function (er, data) { + if (er) return log.er(cb, "No package.json found")(er) + var newVer = semver.valid(args[0]) + if (!newVer) newVer = semver.inc(data.version, args[0]) + if (!newVer) return cb(version.usage) + if (data.version === newVer) return cb(new Error("Version not changed")) + data.version = newVer + Object.keys(data).forEach(function (k) { + if (k.charAt(0) === "_") delete data[k] + }) + readJson.unParsePeople(data) + fs.stat(path.join(process.cwd(), ".git"), function (er, s) { + var doGit = !er && s.isDirectory() + if (!doGit) return write(data, cb) + else checkGit(data, cb) + }) + }) +} +function checkGit (data, cb) { + exec( "git", ["status", "--porcelain"], process.env, false + , function (er, code, stdout, stderr) { + var lines = stdout.trim().split("\n").filter(function (line) { + return line.trim() && !line.match(/^\?\? /) + }) + if (lines.length) return cb(new Error( + "Git working directory not clean.\n"+lines.join("\n"))) + write(data, function (er) { + if (er) return cb(er) + var message = npm.config.get("message").replace(/%s/g, data.version) + chain + ( [ [ exec, "git", ["add","package.json"], process.env, false ] + , [ exec, "git", ["commit", "-m", message ] + , process.env, false ] + , [ exec, "git", ["tag", "v"+data.version], process.env, false ] ] + , cb ) + }) + }) +} +function write (data, cb) { + fs.writeFile( path.join(process.cwd(), "package.json") + , new Buffer(JSON.stringify(data, null, 2)) + , cb ) +} diff --git a/deps/npm/lib/view.js b/deps/npm/lib/view.js new file mode 100644 index 0000000000..3e39f76de9 --- /dev/null +++ b/deps/npm/lib/view.js @@ -0,0 +1,244 @@ +// npm view [pkg [pkg ...]] + +module.exports = view +view.usage = "npm view pkg[@version] [[.subfield]...]" + +view.completion = function (opts, cb) { + if (opts.conf.argv.remain.length <= 2) { + return registry.get("/-/short", cb) + } + // have the package, get the fields. + var tag = npm.config.get("tag") + registry.get(opts.conf.argv.remain[2], function (er, d) { + if (er) return cb(er) + var dv = d.versions[d["dist-tags"][tag]] + , fields = [] + d.versions = Object.keys(d.versions).sort(semver.compare) + fields = getFields(d).concat(getFields(dv)) + cb(null, fields) + }) + + function getFields (d, f, pref) { + f = f || [] + if (!d) return f + pref = pref || [] + Object.keys(d).forEach(function (k) { + if (k.charAt(0) === "_" || k.indexOf(".") !== -1) return + var p = pref.concat(k).join(".") + f.push(p) + if (Array.isArray(d[k])) { + return d[k].forEach(function (val, i) { + var pi = p + "[" + i + "]" + if (val && typeof val === "object") getFields(val, f, [p]) + else f.push(pi) + }) + } + if (typeof d[k] === "object") getFields(d[k], f, [p]) + }) + return f + } +} + +var registry = require("./utils/npm-registry-client/index.js") + , ini = require("ini") + , log = require("./utils/log.js") + , util = require("util") + , output + , npm = require("./npm.js") + , semver = require("semver") + , readJson = require("./utils/read-json.js") + +function view (args, silent, cb) { + if (typeof cb !== "function") cb = silent, silent = false + if (!args.length) return cb("Usage: "+view.usage) + var pkg = args.shift() + , nv = pkg.split("@") + , name = nv.shift() + , version = nv.join("@") || npm.config.get("tag") + + if (name === ".") return cb(view.usage) + + // get the data about this package + registry.get(name, null, 600, function (er, data) { + if (er) return cb(er) + if (data["dist-tags"].hasOwnProperty(version)) { + version = data["dist-tags"][version] + } + var results = [] + , error = null + , versions = data.versions + data.versions = Object.keys(data.versions).sort(semver.compare) + if (!args.length) args = [""] + + if (-1 === args.indexOf("readme")) { + delete data.readme + } + + Object.keys(versions).forEach(function (v) { + try { + versions[v] = readJson.processJson(versions[v]) + } catch (ex) { + delete versions[v] + } + if (semver.satisfies(v, version)) args.forEach(function (args) { + results.push(showFields(data, versions[v], args)) + }) + }) + results = results.reduce(reducer, {}) + var retval = results + + if (args.length === 1 && args[0] === "") { + retval = cleanBlanks(retval) + log.silly(retval, "cleanup") + } + + if (error || silent) cb(error, retval) + else printData(results, data._id, cb.bind(null, error, retval)) + }) +} + +function cleanBlanks (obj) { + var clean = {} + Object.keys(obj).forEach(function (version) { + clean[version] = obj[version][""] + }) + return clean +} + +function reducer (l, r) { + if (r) Object.keys(r).forEach(function (v) { + l[v] = l[v] || {} + Object.keys(r[v]).forEach(function (t) { + l[v][t] = r[v][t] + }) + }) + return l +} + +// return whatever was printed +function showFields (data, version, fields) { + var o = {} + ;[data, version].forEach(function (s) { + Object.keys(s).forEach(function (k) { + o[k] = s[k] + }) + }) + return search(o, fields.split("."), version.version, fields) +} + +function search (data, fields, version, title) { + var field + , tail = fields + while (!field && fields.length) field = tail.shift() + fields = [field].concat(tail) + if (!field && !tail.length) { + var o = {} + o[version] = {} + o[version][title] = data + return o + } + var index = field.match(/(.+)\[([^\]]+)\]$/) + if (index) { + field = index[1] + index = index[2] + if (data.field && data.field.hasOwnProperty(index)) { + return search(data[field][index], tail, version, title) + } else { + field = field + "[" + index + "]" + } + } + if (Array.isArray(data)) { + if (data.length === 1) { + return search(data[0], fields, version, title) + } + var results = [] + , res = null + data.forEach(function (data, i) { + var tl = title.length + , newt = title.substr(0, tl-(fields.join(".").length) - 1) + + "["+i+"]" + [""].concat(fields).join(".") + results.push(search(data, fields.slice(), version, newt)) + }) + results = results.reduce(reducer, {}) + return results + } + if (!data.hasOwnProperty(field)) { + return + } + data = data[field] + if (tail.length) { + if (typeof data === "object") { + // there are more fields to deal with. + return search(data, tail, version, title) + } else { + return new Error("Not an object: "+data) + } + } + var o = {} + o[version] = {} + o[version][title] = data + return o +} + +function printData (data, name, cb) { + var versions = Object.keys(data) + , msg = "" + , showVersions = versions.length > 1 + , showFields + function cb_ (er) { return cb(er, data) } + + versions.forEach(function (v, i) { + var fields = Object.keys(data[v]) + showFields = showFields || (fields.length > 1) + fields.forEach(function (f) { + var d = cleanup(data[v][f]) + if (showVersions || showFields || typeof d !== "string") { + d = util.inspect(cleanup(data[v][f]), false, 5, true) + } + if (f && showFields) f += " = " + if (d.indexOf("\n") !== -1) d = "\n" + d + msg += (showVersions ? name + "@" + v + " " : "") + + (showFields ? f : "") + d + "\n" + }) + }) + output = output || require("./utils/output.js") + output.write(msg, cb_) +} +function cleanup (data) { + if (Array.isArray(data)) { + if (data.length === 1) { + data = data[0] + } else { + return data.map(cleanup) + } + } + if (!data || typeof data !== "object") return data + + if (typeof data.versions === "object" + && data.versions + && !Array.isArray(data.versions)) { + data.versions = Object.keys(data.versions || {}) + } + + var keys = Object.keys(data) + keys.forEach(function (d) { + if (d.charAt(0) === "_") delete data[d] + else if (typeof data[d] === "object") data[d] = cleanup(data[d]) + }) + keys = Object.keys(data) + if (keys.length <= 3 + && data.name + && (keys.length === 1 + || keys.length === 3 && data.email && data.url + || keys.length === 2 && (data.email || data.url))) { + data = unparsePerson(data) + } + return data +} +function unparsePerson (d) { + if (typeof d === "string") return d + return d.name + + (d.email ? " <"+d.email+">" : "") + + (d.url ? " ("+d.url+")" : "") +} + diff --git a/deps/npm/lib/whoami.js b/deps/npm/lib/whoami.js new file mode 100644 index 0000000000..c48f04b1d6 --- /dev/null +++ b/deps/npm/lib/whoami.js @@ -0,0 +1,13 @@ +module.exports = whoami + +var npm = require("./npm.js") + , output = require("./utils/output.js") + , log = require("./utils/log.js") + +whoami.usage = "npm whoami\n(just prints the 'username' config)" + +function whoami (args, cb) { + var me = npm.config.get("username") + if (!me) me = "Not authed. Run 'npm adduser'" + output.write(me, cb) +} diff --git a/deps/npm/lib/xmas.js b/deps/npm/lib/xmas.js new file mode 100644 index 0000000000..90282d0f7b --- /dev/null +++ b/deps/npm/lib/xmas.js @@ -0,0 +1,54 @@ +// happy xmas +var npm = require("./npm.js") + , log = require("./utils/log.js") + +module.exports = function (args, cb) { +npm.config.set("loglevel", "win") +var s = process.platform === "win32" ? " *" : " \u2605" + , f = "\uFF0F" + , b = "\uFF3C" + , x = process.platform === "win32" ? " " : "" + , o = [ "\u0069" , "\u0020", "\u0020", "\u0020", "\u0020", "\u0020" + , "\u0020", "\u0020", "\u0020", "\u0020", "\u0020", "\u0020" + , "\u0020", "\u2E1B","\u2042","\u2E2E","&","@","\uFF61" ] + , oc = [21,33,34,35,36,37] + , l = "\u005e" + +function w (s) { process.stderr.write(s) } + +w("\n") +;(function T (H) { + for (var i = 0; i < H; i ++) w(" ") + w(x+"\033[33m"+s+"\n") + var M = H * 2 - 1 + for (L = 1; L <= H; L ++) { + var O = L * 2 - 2 + var S = (M - O) / 2 + for (var i = 0; i < S; i ++) w(" ") + w(x+"\033[32m"+f) + for (var i = 0; i < O; i ++) w( + "\033["+oc[Math.floor(Math.random()*oc.length)]+"m"+ + o[Math.floor(Math.random() * o.length)] + ) + w(x+"\033[32m"+b+"\n") + } + w(" ") + for (var i = 1; i < H; i ++) w("\033[32m"+l) + w("| "+x+" |") + for (var i = 1; i < H; i ++) w("\033[32m"+l) + if (H > 10) { + w("\n ") + for (var i = 1; i < H; i ++) w(" ") + w("| "+x+" |") + for (var i = 1; i < H; i ++) w(" ") + } +})(20) +w("\n\n") +log.win("Happy Xmas, Noders!", "loves you", cb) +} +var dg=false +Object.defineProperty(module.exports, "usage", {get:function () { + if (dg) module.exports([], function () {}) + dg = true + return " " +}}) diff --git a/deps/npm/node_modules/abbrev/README.md b/deps/npm/node_modules/abbrev/README.md new file mode 100644 index 0000000000..99746fe67c --- /dev/null +++ b/deps/npm/node_modules/abbrev/README.md @@ -0,0 +1,23 @@ +# abbrev-js + +Just like [ruby's Abbrev](http://apidock.com/ruby/Abbrev). + +Usage: + + var abbrev = require("abbrev"); + abbrev("foo", "fool", "folding", "flop"); + + // returns: + { fl: 'flop' + , flo: 'flop' + , flop: 'flop' + , fol: 'folding' + , fold: 'folding' + , foldi: 'folding' + , foldin: 'folding' + , folding: 'folding' + , foo: 'foo' + , fool: 'fool' + } + +This is handy for command-line scripts, or other cases where you want to be able to accept shorthands. diff --git a/deps/npm/node_modules/abbrev/lib/abbrev.js b/deps/npm/node_modules/abbrev/lib/abbrev.js new file mode 100644 index 0000000000..037de2d8da --- /dev/null +++ b/deps/npm/node_modules/abbrev/lib/abbrev.js @@ -0,0 +1,106 @@ + +module.exports = exports = abbrev.abbrev = abbrev + +abbrev.monkeyPatch = monkeyPatch + +function monkeyPatch () { + Array.prototype.abbrev = function () { return abbrev(this) } + Object.prototype.abbrev = function () { return abbrev(Object.keys(this)) } +} + +function abbrev (list) { + if (arguments.length !== 1 || !Array.isArray(list)) { + list = Array.prototype.slice.call(arguments, 0) + } + for (var i = 0, l = list.length, args = [] ; i < l ; i ++) { + args[i] = typeof list[i] === "string" ? list[i] : String(list[i]) + } + + // sort them lexicographically, so that they're next to their nearest kin + args = args.sort(lexSort) + + // walk through each, seeing how much it has in common with the next and previous + var abbrevs = {} + , prev = "" + for (var i = 0, l = args.length ; i < l ; i ++) { + var current = args[i] + , next = args[i + 1] || "" + , nextMatches = true + , prevMatches = true + if (current === next) continue + for (var j = 0, cl = current.length ; j < cl ; j ++) { + var curChar = current.charAt(j) + nextMatches = nextMatches && curChar === next.charAt(j) + prevMatches = prevMatches && curChar === prev.charAt(j) + if (nextMatches || prevMatches) continue + else { + j ++ + break + } + } + prev = current + if (j === cl) { + abbrevs[current] = current + continue + } + for (var a = current.substr(0, j) ; j <= cl ; j ++) { + abbrevs[a] = current + a += current.charAt(j) + } + } + return abbrevs +} + +function lexSort (a, b) { + return a === b ? 0 : a > b ? 1 : -1 +} + + +// tests +if (module === require.main) { + +var assert = require("assert") + , sys +sys = require("util") + +console.log("running tests") +function test (list, expect) { + var actual = abbrev(list) + assert.deepEqual(actual, expect, + "abbrev("+sys.inspect(list)+") === " + sys.inspect(expect) + "\n"+ + "actual: "+sys.inspect(actual)) + actual = abbrev.apply(exports, list) + assert.deepEqual(abbrev.apply(exports, list), expect, + "abbrev("+list.map(JSON.stringify).join(",")+") === " + sys.inspect(expect) + "\n"+ + "actual: "+sys.inspect(actual)) +} + +test([ "ruby", "ruby", "rules", "rules", "rules" ], +{ rub: 'ruby' +, ruby: 'ruby' +, rul: 'rules' +, rule: 'rules' +, rules: 'rules' +}) +test(["fool", "foom", "pool", "pope"], +{ fool: 'fool' +, foom: 'foom' +, poo: 'pool' +, pool: 'pool' +, pop: 'pope' +, pope: 'pope' +}) +test(["a", "ab", "abc", "abcd", "abcde", "acde"], +{ a: 'a' +, ab: 'ab' +, abc: 'abc' +, abcd: 'abcd' +, abcde: 'abcde' +, ac: 'acde' +, acd: 'acde' +, acde: 'acde' +}) + +console.log("pass") + +} diff --git a/deps/npm/node_modules/abbrev/package.json b/deps/npm/node_modules/abbrev/package.json new file mode 100644 index 0000000000..ebd082f5dd --- /dev/null +++ b/deps/npm/node_modules/abbrev/package.json @@ -0,0 +1,8 @@ +{ "name" : "abbrev" +, "version" : "1.0.3" +, "description" : "Like ruby's abbrev module, but in js" +, "author" : "Isaac Z. Schlueter " +, "main" : "./lib/abbrev.js" +, "scripts" : { "test" : "node lib/abbrev.js" } +, "repository" : "http://github.com/isaacs/abbrev-js" +} diff --git a/deps/npm/node_modules/block-stream/README.md b/deps/npm/node_modules/block-stream/README.md new file mode 100644 index 0000000000..c16e9c4688 --- /dev/null +++ b/deps/npm/node_modules/block-stream/README.md @@ -0,0 +1,14 @@ +# block-stream + +A stream of blocks. + +Write data into it, and it'll output data in buffer blocks the size you +specify, padding with zeroes if necessary. + +```javascript +var block = new BlockStream(512) +fs.createReadStream("some-file").pipe(block) +block.pipe(fs.createWriteStream("block-file")) +``` + +When `.end()` or `.flush()` is called, it'll pad the block with zeroes. diff --git a/deps/npm/node_modules/block-stream/bench/block-stream.js b/deps/npm/node_modules/block-stream/bench/block-stream.js new file mode 100644 index 0000000000..1141f3a84c --- /dev/null +++ b/deps/npm/node_modules/block-stream/bench/block-stream.js @@ -0,0 +1,68 @@ +var BlockStream = require("../block-stream.js") + +var blockSizes = [16, 25, 1024] + , writeSizes = [4, 8, 15, 16, 17, 64, 100] + , writeCounts = [1, 10, 100] + , tap = require("tap") + +writeCounts.forEach(function (writeCount) { +blockSizes.forEach(function (blockSize) { +writeSizes.forEach(function (writeSize) { + tap.test("writeSize=" + writeSize + + " blockSize="+blockSize + + " writeCount="+writeCount, function (t) { + var f = new BlockStream(blockSize, {nopad: true }) + + var actualChunks = 0 + var actualBytes = 0 + var timeouts = 0 + + f.on("data", function (c) { + timeouts ++ + + actualChunks ++ + actualBytes += c.length + + // make sure that no data gets corrupted, and basic sanity + var before = c.toString() + // simulate a slow write operation + setTimeout(function () { + timeouts -- + + var after = c.toString() + t.equal(after, before, "should not change data") + + // now corrupt it, to find leaks. + for (var i = 0; i < c.length; i ++) { + c[i] = "x".charCodeAt(0) + } + }, 100) + }) + + f.on("end", function () { + // round up to the nearest block size + var expectChunks = Math.ceil(writeSize * writeCount * 2 / blockSize) + var expectBytes = writeSize * writeCount * 2 + t.equal(actualBytes, expectBytes, + "bytes=" + expectBytes + " writeSize=" + writeSize) + t.equal(actualChunks, expectChunks, + "chunks=" + expectChunks + " writeSize=" + writeSize) + + // wait for all the timeout checks to finish, then end the test + setTimeout(function WAIT () { + if (timeouts > 0) return setTimeout(WAIT) + t.end() + }, 100) + }) + + for (var i = 0; i < writeCount; i ++) { + var a = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) a[j] = "a".charCodeAt(0) + var b = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) b[j] = "b".charCodeAt(0) + f.write(a) + f.write(b) + } + f.end() + }) +}) }) }) diff --git a/deps/npm/node_modules/block-stream/bench/dropper-pause.js b/deps/npm/node_modules/block-stream/bench/dropper-pause.js new file mode 100644 index 0000000000..93e4068eea --- /dev/null +++ b/deps/npm/node_modules/block-stream/bench/dropper-pause.js @@ -0,0 +1,70 @@ +var BlockStream = require("dropper") + +var blockSizes = [16, 25, 1024] + , writeSizes = [4, 8, 15, 16, 17, 64, 100] + , writeCounts = [1, 10, 100] + , tap = require("tap") + +writeCounts.forEach(function (writeCount) { +blockSizes.forEach(function (blockSize) { +writeSizes.forEach(function (writeSize) { + tap.test("writeSize=" + writeSize + + " blockSize="+blockSize + + " writeCount="+writeCount, function (t) { + var f = new BlockStream(blockSize, {nopad: true }) + + var actualChunks = 0 + var actualBytes = 0 + var timeouts = 0 + + f.on("data", function (c) { + timeouts ++ + + actualChunks ++ + actualBytes += c.length + + // make sure that no data gets corrupted, and basic sanity + var before = c.toString() + // simulate a slow write operation + f.pause() + setTimeout(function () { + timeouts -- + + var after = c.toString() + t.equal(after, before, "should not change data") + + // now corrupt it, to find leaks. + for (var i = 0; i < c.length; i ++) { + c[i] = "x".charCodeAt(0) + } + f.resume() + }, 100) + }) + + f.on("end", function () { + // round up to the nearest block size + var expectChunks = Math.ceil(writeSize * writeCount * 2 / blockSize) + var expectBytes = writeSize * writeCount * 2 + t.equal(actualBytes, expectBytes, + "bytes=" + expectBytes + " writeSize=" + writeSize) + t.equal(actualChunks, expectChunks, + "chunks=" + expectChunks + " writeSize=" + writeSize) + + // wait for all the timeout checks to finish, then end the test + setTimeout(function WAIT () { + if (timeouts > 0) return setTimeout(WAIT) + t.end() + }, 100) + }) + + for (var i = 0; i < writeCount; i ++) { + var a = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) a[j] = "a".charCodeAt(0) + var b = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) b[j] = "b".charCodeAt(0) + f.write(a) + f.write(b) + } + f.end() + }) +}) }) }) diff --git a/deps/npm/node_modules/block-stream/bench/dropper.js b/deps/npm/node_modules/block-stream/bench/dropper.js new file mode 100644 index 0000000000..55fa133054 --- /dev/null +++ b/deps/npm/node_modules/block-stream/bench/dropper.js @@ -0,0 +1,68 @@ +var BlockStream = require("dropper") + +var blockSizes = [16, 25, 1024] + , writeSizes = [4, 8, 15, 16, 17, 64, 100] + , writeCounts = [1, 10, 100] + , tap = require("tap") + +writeCounts.forEach(function (writeCount) { +blockSizes.forEach(function (blockSize) { +writeSizes.forEach(function (writeSize) { + tap.test("writeSize=" + writeSize + + " blockSize="+blockSize + + " writeCount="+writeCount, function (t) { + var f = new BlockStream(blockSize, {nopad: true }) + + var actualChunks = 0 + var actualBytes = 0 + var timeouts = 0 + + f.on("data", function (c) { + timeouts ++ + + actualChunks ++ + actualBytes += c.length + + // make sure that no data gets corrupted, and basic sanity + var before = c.toString() + // simulate a slow write operation + setTimeout(function () { + timeouts -- + + var after = c.toString() + t.equal(after, before, "should not change data") + + // now corrupt it, to find leaks. + for (var i = 0; i < c.length; i ++) { + c[i] = "x".charCodeAt(0) + } + }, 100) + }) + + f.on("end", function () { + // round up to the nearest block size + var expectChunks = Math.ceil(writeSize * writeCount * 2 / blockSize) + var expectBytes = writeSize * writeCount * 2 + t.equal(actualBytes, expectBytes, + "bytes=" + expectBytes + " writeSize=" + writeSize) + t.equal(actualChunks, expectChunks, + "chunks=" + expectChunks + " writeSize=" + writeSize) + + // wait for all the timeout checks to finish, then end the test + setTimeout(function WAIT () { + if (timeouts > 0) return setTimeout(WAIT) + t.end() + }, 100) + }) + + for (var i = 0; i < writeCount; i ++) { + var a = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) a[j] = "a".charCodeAt(0) + var b = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) b[j] = "b".charCodeAt(0) + f.write(a) + f.write(b) + } + f.end() + }) +}) }) }) diff --git a/deps/npm/node_modules/block-stream/block-stream.js b/deps/npm/node_modules/block-stream/block-stream.js new file mode 100644 index 0000000000..008de035c2 --- /dev/null +++ b/deps/npm/node_modules/block-stream/block-stream.js @@ -0,0 +1,209 @@ +// write data to it, and it'll emit data in 512 byte blocks. +// if you .end() or .flush(), it'll emit whatever it's got, +// padded with nulls to 512 bytes. + +module.exports = BlockStream + +var Stream = require("stream").Stream + , inherits = require("inherits") + , assert = require("assert").ok + , debug = process.env.DEBUG ? console.error : function () {} + +function BlockStream (size, opt) { + this.writable = this.readable = true + this._opt = opt || {} + this._chunkSize = size || 512 + this._offset = 0 + this._buffer = [] + this._bufferLength = 0 + if (this._opt.nopad) this._zeroes = false + else { + this._zeroes = new Buffer(this._chunkSize) + for (var i = 0; i < this._chunkSize; i ++) { + this._zeroes[i] = 0 + } + } +} + +inherits(BlockStream, Stream) + +BlockStream.prototype.write = function (c) { + // debug(" BS write", c) + if (this._ended) throw new Error("BlockStream: write after end") + if (c && !Buffer.isBuffer(c)) c = new Buffer(c + "") + if (c.length) { + this._buffer.push(c) + this._bufferLength += c.length + } + // debug("pushed onto buffer", this._bufferLength) + if (this._bufferLength >= this._chunkSize) { + if (this._paused) { + // debug(" BS paused, return false, need drain") + this._needDrain = true + return false + } + this._emitChunk() + } + return true +} + +BlockStream.prototype.pause = function () { + // debug(" BS pausing") + this._paused = true +} + +BlockStream.prototype.resume = function () { + // debug(" BS resume") + this._paused = false + return this._emitChunk() +} + +BlockStream.prototype.end = function (chunk) { + // debug("end", chunk) + if (typeof chunk === "function") cb = chunk, chunk = null + if (chunk) this.write(chunk) + this._ended = true + this.flush() +} + +BlockStream.prototype.flush = function () { + this._emitChunk(true) +} + +BlockStream.prototype._emitChunk = function (flush) { + // debug("emitChunk flush=%j emitting=%j paused=%j", flush, this._emitting, this._paused) + + // emit a chunk + if (flush && this._zeroes) { + // debug(" BS push zeroes", this._bufferLength) + // push a chunk of zeroes + var padBytes = (this._bufferLength % this._chunkSize) + if (padBytes !== 0) padBytes = this._chunkSize - padBytes + if (padBytes > 0) { + // debug("padBytes", padBytes, this._zeroes.slice(0, padBytes)) + this._buffer.push(this._zeroes.slice(0, padBytes)) + this._bufferLength += padBytes + // debug(this._buffer[this._buffer.length - 1].length, this._bufferLength) + } + } + + if (this._emitting || this._paused) return + this._emitting = true + + // debug(" BS entering loops") + var bufferIndex = 0 + while (this._bufferLength >= this._chunkSize && + (flush || !this._paused)) { + // debug(" BS data emission loop", this._bufferLength) + + var out + , outOffset = 0 + , outHas = this._chunkSize + + while (outHas > 0 && (flush || !this._paused) ) { + // debug(" BS data inner emit loop", this._bufferLength) + var cur = this._buffer[bufferIndex] + , curHas = cur.length - this._offset + // debug("cur=", cur) + // debug("curHas=%j", curHas) + // If it's not big enough to fill the whole thing, then we'll need + // to copy multiple buffers into one. However, if it is big enough, + // then just slice out the part we want, to save unnecessary copying. + // Also, need to copy if we've already done some copying, since buffers + // can't be joined like cons strings. + if (out || curHas < outHas) { + out = out || new Buffer(this._chunkSize) + cur.copy(out, outOffset, + this._offset, this._offset + Math.min(curHas, outHas)) + } else if (cur.length === outHas && this._offset === 0) { + // shortcut -- cur is exactly long enough, and no offset. + out = cur + } else { + // slice out the piece of cur that we need. + out = cur.slice(this._offset, this._offset + outHas) + } + + if (curHas > outHas) { + // means that the current buffer couldn't be completely output + // update this._offset to reflect how much WAS written + this._offset += outHas + outHas = 0 + } else { + // output the entire current chunk. + // toss it away + outHas -= curHas + outOffset += curHas + bufferIndex ++ + this._offset = 0 + } + } + + this._bufferLength -= this._chunkSize + assert(out.length === this._chunkSize) + // debug("emitting data", out) + // debug(" BS emitting, paused=%j", this._paused, this._bufferLength) + this.emit("data", out) + out = null + } + // debug(" BS out of loops", this._bufferLength) + + // whatever is left, it's not enough to fill up a block, or we're paused + this._buffer = this._buffer.slice(bufferIndex) + if (this._paused) { + // debug(" BS paused, leaving", this._bufferLength) + this._needsDrain = true + this._emitting = false + return + } + + // if flushing, and not using null-padding, then need to emit the last + // chunk(s) sitting in the queue. We know that it's not enough to + // fill up a whole block, because otherwise it would have been emitted + // above, but there may be some offset. + var l = this._buffer.length + if (flush && !this._zeroes && l) { + if (l === 1) { + if (this._offset) { + this.emit("data", this._buffer[0].slice(this._offset)) + } else { + this.emit("data", this._buffer[0]) + } + } else { + var outHas = this._bufferLength + , out = new Buffer(outHas) + , outOffset = 0 + for (var i = 0; i < l; i ++) { + var cur = this._buffer[i] + , curHas = cur.length - this._offset + cur.copy(out, outOffset, this._offset) + this._offset = 0 + outOffset += curHas + this._bufferLength -= curHas + } + this.emit("data", out) + } + // truncate + this._buffer.length = 0 + this._bufferLength = 0 + this._offset = 0 + } + + // now either drained or ended + // debug("either draining, or ended", this._bufferLength, this._ended) + // means that we've flushed out all that we can so far. + if (this._needDrain) { + // debug("emitting drain", this._bufferLength) + this._needDrain = false + this.emit("drain") + } + + if ((this._bufferLength === 0) && this._ended && !this._endEmitted) { + // debug("emitting end", this._bufferLength) + this._endEmitted = true + this.emit("end") + } + + this._emitting = false + + // debug(" BS no longer emitting", flush, this._paused, this._emitting, this._bufferLength, this._chunkSize) +} diff --git a/deps/npm/node_modules/block-stream/package.json b/deps/npm/node_modules/block-stream/package.json new file mode 100644 index 0000000000..203961a144 --- /dev/null +++ b/deps/npm/node_modules/block-stream/package.json @@ -0,0 +1,23 @@ +{ + "author": "Isaac Z. Schlueter (http://blog.izs.me/)", + "name": "block-stream", + "description": "a stream of blocks", + "version": "0.0.4", + "repository": { + "type": "git", + "url": "git://github.com/isaacs/block-stream.git" + }, + "engines": { + "node": "0.4 || ~0.5.8 || 0.6" + }, + "main": "block-stream.js", + "dependencies": { + "inherits": "~1.0.0" + }, + "devDependencies": { + "tap": "0.x" + }, + "scripts": { + "test": "tap test/" + } +} diff --git a/deps/npm/node_modules/block-stream/test/basic.js b/deps/npm/node_modules/block-stream/test/basic.js new file mode 100644 index 0000000000..b4b930511e --- /dev/null +++ b/deps/npm/node_modules/block-stream/test/basic.js @@ -0,0 +1,27 @@ +var tap = require("tap") + , BlockStream = require("../block-stream.js") + +tap.test("basic test", function (t) { + var b = new BlockStream(16) + var fs = require("fs") + var fstr = fs.createReadStream(__filename, {encoding: "utf8"}) + fstr.pipe(b) + + var stat + t.doesNotThrow(function () { + stat = fs.statSync(__filename) + }, "stat should not throw") + + var totalBytes = 0 + b.on("data", function (c) { + t.equal(c.length, 16, "chunks should be 16 bytes long") + t.type(c, Buffer, "chunks should be buffer objects") + totalBytes += c.length + }) + b.on("end", function () { + var expectedBytes = stat.size + (16 - stat.size % 16) + t.equal(totalBytes, expectedBytes, "Should be multiple of 16") + t.end() + }) + +}) diff --git a/deps/npm/node_modules/block-stream/test/nopad-thorough.js b/deps/npm/node_modules/block-stream/test/nopad-thorough.js new file mode 100644 index 0000000000..1141f3a84c --- /dev/null +++ b/deps/npm/node_modules/block-stream/test/nopad-thorough.js @@ -0,0 +1,68 @@ +var BlockStream = require("../block-stream.js") + +var blockSizes = [16, 25, 1024] + , writeSizes = [4, 8, 15, 16, 17, 64, 100] + , writeCounts = [1, 10, 100] + , tap = require("tap") + +writeCounts.forEach(function (writeCount) { +blockSizes.forEach(function (blockSize) { +writeSizes.forEach(function (writeSize) { + tap.test("writeSize=" + writeSize + + " blockSize="+blockSize + + " writeCount="+writeCount, function (t) { + var f = new BlockStream(blockSize, {nopad: true }) + + var actualChunks = 0 + var actualBytes = 0 + var timeouts = 0 + + f.on("data", function (c) { + timeouts ++ + + actualChunks ++ + actualBytes += c.length + + // make sure that no data gets corrupted, and basic sanity + var before = c.toString() + // simulate a slow write operation + setTimeout(function () { + timeouts -- + + var after = c.toString() + t.equal(after, before, "should not change data") + + // now corrupt it, to find leaks. + for (var i = 0; i < c.length; i ++) { + c[i] = "x".charCodeAt(0) + } + }, 100) + }) + + f.on("end", function () { + // round up to the nearest block size + var expectChunks = Math.ceil(writeSize * writeCount * 2 / blockSize) + var expectBytes = writeSize * writeCount * 2 + t.equal(actualBytes, expectBytes, + "bytes=" + expectBytes + " writeSize=" + writeSize) + t.equal(actualChunks, expectChunks, + "chunks=" + expectChunks + " writeSize=" + writeSize) + + // wait for all the timeout checks to finish, then end the test + setTimeout(function WAIT () { + if (timeouts > 0) return setTimeout(WAIT) + t.end() + }, 100) + }) + + for (var i = 0; i < writeCount; i ++) { + var a = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) a[j] = "a".charCodeAt(0) + var b = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) b[j] = "b".charCodeAt(0) + f.write(a) + f.write(b) + } + f.end() + }) +}) }) }) diff --git a/deps/npm/node_modules/block-stream/test/nopad.js b/deps/npm/node_modules/block-stream/test/nopad.js new file mode 100644 index 0000000000..6d38429fbc --- /dev/null +++ b/deps/npm/node_modules/block-stream/test/nopad.js @@ -0,0 +1,57 @@ +var BlockStream = require("../") +var tap = require("tap") + + +tap.test("don't pad, small writes", function (t) { + var f = new BlockStream(16, { nopad: true }) + t.plan(1) + + f.on("data", function (c) { + t.equal(c.toString(), "abc", "should get 'abc'") + }) + + f.on("end", function () { t.end() }) + + f.write(new Buffer("a")) + f.write(new Buffer("b")) + f.write(new Buffer("c")) + f.end() +}) + +tap.test("don't pad, exact write", function (t) { + var f = new BlockStream(16, { nopad: true }) + t.plan(1) + + var first = true + f.on("data", function (c) { + if (first) { + first = false + t.equal(c.toString(), "abcdefghijklmnop", "first chunk") + } else { + t.fail("should only get one") + } + }) + + f.on("end", function () { t.end() }) + + f.end(new Buffer("abcdefghijklmnop")) +}) + +tap.test("don't pad, big write", function (t) { + var f = new BlockStream(16, { nopad: true }) + t.plan(2) + + var first = true + f.on("data", function (c) { + if (first) { + first = false + t.equal(c.toString(), "abcdefghijklmnop", "first chunk") + } else { + t.equal(c.toString(), "q") + } + }) + + f.on("end", function () { t.end() }) + + f.end(new Buffer("abcdefghijklmnopq")) +}) diff --git a/deps/npm/node_modules/block-stream/test/pause-resume.js b/deps/npm/node_modules/block-stream/test/pause-resume.js new file mode 100644 index 0000000000..248cf9cbd5 --- /dev/null +++ b/deps/npm/node_modules/block-stream/test/pause-resume.js @@ -0,0 +1,73 @@ +var BlockStream = require("../block-stream.js") + +var blockSizes = [16] + , writeSizes = [15, 16, 17] + , writeCounts = [1, 10, 100] + , tap = require("tap") + +writeCounts.forEach(function (writeCount) { +blockSizes.forEach(function (blockSize) { +writeSizes.forEach(function (writeSize) { + tap.test("writeSize=" + writeSize + + " blockSize="+blockSize + + " writeCount="+writeCount, function (t) { + var f = new BlockStream(blockSize) + + var actualChunks = 0 + var actualBytes = 0 + var timeouts = 0 + var paused = false + + f.on("data", function (c) { + timeouts ++ + t.notOk(paused, "should not be paused when emitting data") + + actualChunks ++ + actualBytes += c.length + + // make sure that no data gets corrupted, and basic sanity + var before = c.toString() + // simulate a slow write operation + paused = true + f.pause() + process.nextTick(function () { + var after = c.toString() + t.equal(after, before, "should not change data") + + // now corrupt it, to find leaks. + for (var i = 0; i < c.length; i ++) { + c[i] = "x".charCodeAt(0) + } + paused = false + f.resume() + timeouts -- + }) + }) + + f.on("end", function () { + // round up to the nearest block size + var expectChunks = Math.ceil(writeSize * writeCount * 2 / blockSize) + var expectBytes = expectChunks * blockSize + t.equal(actualBytes, expectBytes, + "bytes=" + expectBytes + " writeSize=" + writeSize) + t.equal(actualChunks, expectChunks, + "chunks=" + expectChunks + " writeSize=" + writeSize) + + // wait for all the timeout checks to finish, then end the test + setTimeout(function WAIT () { + if (timeouts > 0) return setTimeout(WAIT) + t.end() + }, 200) + }) + + for (var i = 0; i < writeCount; i ++) { + var a = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) a[j] = "a".charCodeAt(0) + var b = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) b[j] = "b".charCodeAt(0) + f.write(a) + f.write(b) + } + f.end() + }) +}) }) }) diff --git a/deps/npm/node_modules/block-stream/test/thorough.js b/deps/npm/node_modules/block-stream/test/thorough.js new file mode 100644 index 0000000000..4ab14ff208 --- /dev/null +++ b/deps/npm/node_modules/block-stream/test/thorough.js @@ -0,0 +1,68 @@ +var BlockStream = require("../block-stream.js") + +var blockSizes = [16, 25, 1024] + , writeSizes = [4, 8, 15, 16, 17, 64, 100] + , writeCounts = [1, 10, 100] + , tap = require("tap") + +writeCounts.forEach(function (writeCount) { +blockSizes.forEach(function (blockSize) { +writeSizes.forEach(function (writeSize) { + tap.test("writeSize=" + writeSize + + " blockSize="+blockSize + + " writeCount="+writeCount, function (t) { + var f = new BlockStream(blockSize) + + var actualChunks = 0 + var actualBytes = 0 + var timeouts = 0 + + f.on("data", function (c) { + timeouts ++ + + actualChunks ++ + actualBytes += c.length + + // make sure that no data gets corrupted, and basic sanity + var before = c.toString() + // simulate a slow write operation + setTimeout(function () { + timeouts -- + + var after = c.toString() + t.equal(after, before, "should not change data") + + // now corrupt it, to find leaks. + for (var i = 0; i < c.length; i ++) { + c[i] = "x".charCodeAt(0) + } + }, 100) + }) + + f.on("end", function () { + // round up to the nearest block size + var expectChunks = Math.ceil(writeSize * writeCount * 2 / blockSize) + var expectBytes = expectChunks * blockSize + t.equal(actualBytes, expectBytes, + "bytes=" + expectBytes + " writeSize=" + writeSize) + t.equal(actualChunks, expectChunks, + "chunks=" + expectChunks + " writeSize=" + writeSize) + + // wait for all the timeout checks to finish, then end the test + setTimeout(function WAIT () { + if (timeouts > 0) return setTimeout(WAIT) + t.end() + }, 100) + }) + + for (var i = 0; i < writeCount; i ++) { + var a = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) a[j] = "a".charCodeAt(0) + var b = new Buffer(writeSize); + for (var j = 0; j < writeSize; j ++) b[j] = "b".charCodeAt(0) + f.write(a) + f.write(b) + } + f.end() + }) +}) }) }) diff --git a/deps/npm/node_modules/fstream/.gitignore b/deps/npm/node_modules/fstream/.gitignore new file mode 100644 index 0000000000..66880db1ab --- /dev/null +++ b/deps/npm/node_modules/fstream/.gitignore @@ -0,0 +1,3 @@ +.*.swp +examples/deep-copy +node_modules/ diff --git a/deps/npm/node_modules/fstream/README.md b/deps/npm/node_modules/fstream/README.md new file mode 100644 index 0000000000..9d8cb77e5c --- /dev/null +++ b/deps/npm/node_modules/fstream/README.md @@ -0,0 +1,76 @@ +Like FS streams, but with stat on them, and supporting directories and +symbolic links, as well as normal files. Also, you can use this to set +the stats on a file, even if you don't change its contents, or to create +a symlink, etc. + +So, for example, you can "write" a directory, and it'll call `mkdir`. You +can specify a uid and gid, and it'll call `chown`. You can specify a +`mtime` and `atime`, and it'll call `utimes`. You can call it a symlink +and provide a `linkpath` and it'll call `symlink`. + +Note that it won't automatically resolve symbolic links. So, if you +call `fstream.Reader('/some/symlink')` then you'll get an object +that stats and then ends immediately (since it has no data). To follow +symbolic links, do this: `fstream.Reader({path:'/some/symlink', follow: +true })`. + +There are various checks to make sure that the bytes emitted are the +same as the intended size, if the size is set. + +## Examples + +```javascript +fstream + .Writer({ path: "path/to/file" + , mode: 0755 + , size: 6 + }) + .write("hello\n") + .end() +``` + +This will create the directories if they're missing, and then write +`hello\n` into the file, chmod it to 0755, and assert that 6 bytes have +been written when it's done. + +```javascript +fstream + .Writer({ path: "path/to/file" + , mode: 0755 + , size: 6 + , flags: "a" + }) + .write("hello\n") + .end() +``` + +You can pass flags in, if you want to append to a file. + +```javascript +fstream + .Writer({ path: "path/to/symlink" + , linkpath: "./file" + , SymbolicLink: true + , mode: "0755" // octal strings supported + }) + .end() +``` + +If isSymbolicLink is a function, it'll be called, and if it returns +true, then it'll treat it as a symlink. If it's not a function, then +any truish value will make a symlink, or you can set `type: +'SymbolicLink'`, which does the same thing. + +Note that the linkpath is relative to the symbolic link location, not +the parent dir or cwd. + +```javascript +fstream + .Reader("path/to/dir") + .pipe(fstream.Writer("path/to/other/dir")) +``` + +This will do like `cp -Rp path/to/dir path/to/other/dir`. If the other +dir exists and isn't a directory, then it'll emit an error. It'll also +set the uid, gid, mode, etc. to be identical. In this way, it's more +like `rsync -a` than simply a copy. diff --git a/deps/npm/node_modules/fstream/examples/pipe.js b/deps/npm/node_modules/fstream/examples/pipe.js new file mode 100644 index 0000000000..b08d79481b --- /dev/null +++ b/deps/npm/node_modules/fstream/examples/pipe.js @@ -0,0 +1,113 @@ +var fstream = require("../fstream.js") +var path = require("path") + +var r = fstream.Reader({ path: path.dirname(__dirname) + , filter: function () { + return !this.basename.match(/^\./) && + !this.basename.match(/^node_modules$/) + !this.basename.match(/^deep-copy$/) + } + }) + +var w = fstream.Writer({ path: path.resolve(__dirname, "deep-copy") + , type: "Directory" + }) + +var indent = "" +var escape = {} + +r.on("entry", appears) +//r.on("ready", function () { +// appears(r) +//}) + +function appears (entry) { + console.error(indent + "a %s appears!", entry.type, entry.basename) + if (foggy) { + console.error("FOGGY!") + var p = entry + do { + console.error(p.depth, p.path, p._paused) + } while (p = p.parent) + + throw new Error("\033[mshould not have entries while foggy") + } + indent += "\t" + entry.on("data", missile(entry)) + entry.on("end", runaway(entry)) + entry.on("entry", appears) +} + +var foggy +function missile (entry) { + if (entry.type === "Directory") { + var ended = false + entry.once("end", function () { ended = true }) + return function (c) { + // throw in some pathological pause()/resume() behavior + // just for extra fun. + process.nextTick(function () { + if (!foggy && !ended) { // && Math.random() < 0.3) { + console.error(indent +"%s casts a spell", entry.basename) + console.error("\na slowing fog comes over the battlefield...\n\033[32m") + entry.pause() + entry.once("resume", liftFog) + foggy = setTimeout(liftFog, 10) + + function liftFog (who) { + if (!foggy) return + if (who) { + console.error("%s breaks the spell!", who && who.path) + } else { + console.error("the spell expires!") + } + console.error("\033[mthe fog lifts!\n") + clearTimeout(foggy) + foggy = null + if (entry._paused) entry.resume() + } + + } + }) + } + } + + return function (c) { + var e = Math.random() < 0.5 + console.error(indent + "%s %s for %d damage!", + entry.basename, + e ? "is struck" : "fires a chunk", + c.length) + } +} + +function runaway (entry) { return function () { + var e = Math.random() < 0.5 + console.error(indent + "%s %s", + entry.basename, + e ? "turns to flee" : "is vanquished!") + indent = indent.slice(0, -1) +}} + + +w.on("entry", attacks) +//w.on("ready", function () { attacks(w) }) +function attacks (entry) { + console.error(indent + "%s %s!", entry.basename, + entry.type === "Directory" ? "calls for backup" : "attacks") + entry.on("entry", attacks) +} + +ended = false +r.on("end", function () { + if (foggy) clearTimeout(foggy) + console.error("\033[mIT'S OVER!!") + console.error("A WINNAR IS YOU!") + ended = true +}) + +process.on("exit", function () { + console.error("ended? "+ended) +}) +r.pipe(w) + diff --git a/deps/npm/node_modules/fstream/examples/reader.js b/deps/npm/node_modules/fstream/examples/reader.js new file mode 100644 index 0000000000..73075102f4 --- /dev/null +++ b/deps/npm/node_modules/fstream/examples/reader.js @@ -0,0 +1,29 @@ +var fstream = require("../fstream.js") +var path = require("path") + +var r = fstream.Reader({ path: path.dirname(__dirname) + , filter: function () { + return !this.basename.match(/^\./) + } + }) + +console.error(r instanceof fstream.Reader) +console.error(r instanceof require("stream").Stream) +console.error(r instanceof require("events").EventEmitter) +console.error(r.on) + +r.on("stat", function () { + console.error("a %s !!!\t", r.type, r.path) +}) + +r.on("entries", function (entries) { + console.error("\t" + entries.join("\n\t")) +}) + +r.on("entry", function (entry) { + console.error("a %s !!!\t", entry.type, entry.path) +}) + +r.on("end", function () { + console.error("IT'S OVER!!") +}) diff --git a/deps/npm/node_modules/fstream/examples/symlink-write.js b/deps/npm/node_modules/fstream/examples/symlink-write.js new file mode 100644 index 0000000000..657375b988 --- /dev/null +++ b/deps/npm/node_modules/fstream/examples/symlink-write.js @@ -0,0 +1,9 @@ +var fstream = require("../fstream.js") + +fstream + .Writer({ path: "path/to/symlink" + , linkpath: "./file" + , isSymbolicLink: true + , mode: "0755" // octal strings supported + }) + .end() diff --git a/deps/npm/node_modules/fstream/fstream.js b/deps/npm/node_modules/fstream/fstream.js new file mode 100644 index 0000000000..c66d26f519 --- /dev/null +++ b/deps/npm/node_modules/fstream/fstream.js @@ -0,0 +1,31 @@ +exports.Abstract = require("./lib/abstract.js") +exports.Reader = require("./lib/reader.js") +exports.Writer = require("./lib/writer.js") + +exports.File = + { Reader: require("./lib/file-reader.js") + , Writer: require("./lib/file-writer.js") } + +exports.Dir = + { Reader : require("./lib/dir-reader.js") + , Writer : require("./lib/dir-writer.js") } + +exports.Link = + { Reader : require("./lib/link-reader.js") + , Writer : require("./lib/link-writer.js") } + +exports.Proxy = + { Reader : require("./lib/proxy-reader.js") + , Writer : require("./lib/proxy-writer.js") } + +exports.Reader.Dir = exports.DirReader = exports.Dir.Reader +exports.Reader.File = exports.FileReader = exports.File.Reader +exports.Reader.Link = exports.LinkReader = exports.Link.Reader +exports.Reader.Proxy = exports.ProxyReader = exports.Proxy.Reader + +exports.Writer.Dir = exports.DirWriter = exports.Dir.Writer +exports.Writer.File = exports.FileWriter = exports.File.Writer +exports.Writer.Link = exports.LinkWriter = exports.Link.Writer +exports.Writer.Proxy = exports.ProxyWriter = exports.Proxy.Writer + +exports.collect = require("./lib/collect.js") diff --git a/deps/npm/node_modules/fstream/lib/abstract.js b/deps/npm/node_modules/fstream/lib/abstract.js new file mode 100644 index 0000000000..add48b945d --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/abstract.js @@ -0,0 +1,82 @@ +// the parent class for all fstreams. + +module.exports = Abstract + +var Stream = require("stream").Stream + , inherits = require("inherits") + +function Abstract () { + Stream.call(this) +} + +inherits(Abstract, Stream) + +Abstract.prototype.on = function (ev, fn) { + if (ev === "ready" && this.ready) { + process.nextTick(fn.bind(this)) + } else { + Stream.prototype.on.call(this, ev, fn) + } + return this +} + +Abstract.prototype.destroy = function () {} + +Abstract.prototype.warn = function (msg, code) { + var me = this + , er = decorate(msg, code, me) + if (!me.listeners("warn")) { + console.error("%s %s\n" + + "path = %s\n" + + "syscall = %s\n" + + "fstream_type = %s\n" + + "fstream_path = %s\n" + + "fstream_unc_path = %s\n" + + "fstream_class = %s\n" + + "fstream_stack =\n%s\n", + code || "UNKNOWN", + er.stack, + er.path, + er.syscall, + er.fstream_type, + er.fstream_path, + er.fstream_unc_path, + er.fstream_class, + er.fstream_stack.join("\n")) + } else { + me.emit("warn", er) + } +} + +Abstract.prototype.info = function (msg, code) { + var me = this + if (!me.listeners("info")) return + me.emit("info", msg, code) +} + +Abstract.prototype.error = function (msg, code, th) { + var er = decorate(msg, code, this) + if (th) throw er + else this.emit("error", er) +} + +function decorate (er, code, me) { + if (!(er instanceof Error)) er = new Error(er) + er.code = er.code || code + er.path = er.path || me.path + er.fstream_type = er.fstream_type || me.type + er.fstream_path = er.fstream_path || me.path + if (me._path !== me.path) { + er.fstream_unc_path = er.fstream_unc_path || me._path + } + if (me.linkpath) { + er.fstream_linkpath = er.fstream_linkpath || me.linkpath + } + er.fstream_class = er.fstream_class || me.constructor.name + er.fstream_stack = er.fstream_stack || + new Error().stack.split(/\n/).slice(3).map(function (s) { + return s.replace(/^ at /, "") + }) + + return er +} diff --git a/deps/npm/node_modules/fstream/lib/collect.js b/deps/npm/node_modules/fstream/lib/collect.js new file mode 100644 index 0000000000..a36f780eb2 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/collect.js @@ -0,0 +1,67 @@ +module.exports = collect + +function collect (stream) { + if (stream._collected) return + + stream._collected = true + stream.pause() + + stream.on("data", save) + stream.on("end", save) + var buf = [] + function save (b) { + if (typeof b === "string") b = new Buffer(b) + if (Buffer.isBuffer(b) && !b.length) return + buf.push(b) + } + + stream.on("entry", saveEntry) + var entryBuffer = [] + function saveEntry (e) { + collect(e) + entryBuffer.push(e) + } + + stream.on("proxy", proxyPause) + function proxyPause (p) { + p.pause() + } + + + // replace the pipe method with a new version that will + // unlock the buffered stuff. if you just call .pipe() + // without a destination, then it'll re-play the events. + stream.pipe = (function (orig) { return function (dest) { + // console.error(" === open the pipes", dest && dest.path) + + // let the entries flow through one at a time. + // Once they're all done, then we can resume completely. + var e = 0 + ;(function unblockEntry () { + var entry = entryBuffer[e++] + // console.error(" ==== unblock entry", entry && entry.path) + if (!entry) return resume() + entry.on("end", unblockEntry) + if (dest) dest.add(entry) + else stream.emit("entry", entry) + })() + + function resume () { + stream.removeListener("entry", saveEntry) + stream.removeListener("data", save) + stream.removeListener("end", save) + + stream.pipe = orig + if (dest) stream.pipe(dest) + + buf.forEach(function (b) { + if (b) stream.emit("data", b) + else stream.emit("end") + }) + + stream.resume() + } + + return dest + }})(stream.pipe) +} diff --git a/deps/npm/node_modules/fstream/lib/dir-reader.js b/deps/npm/node_modules/fstream/lib/dir-reader.js new file mode 100644 index 0000000000..a7806dcbb0 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/dir-reader.js @@ -0,0 +1,180 @@ +// A thing that emits "entry" events with Reader objects +// Pausing it causes it to stop emitting entry events, and also +// pauses the current entry if there is one. + +module.exports = DirReader + +var fs = require("graceful-fs") + , fstream = require("../fstream.js") + , Reader = fstream.Reader + , inherits = require("inherits") + , mkdir = require("mkdirp") + , path = require("path") + , Reader = require("./reader.js") + +inherits(DirReader, Reader) + +function DirReader (props) { + var me = this + if (!(me instanceof DirReader)) throw new Error( + "DirReader must be called as constructor.") + + // should already be established as a Directory type + if (props.type !== "Directory" || !props.Directory) { + throw new Error("Non-directory type "+ props.type) + } + + me._entries = null + me._index = -1 + me._paused = false + me._length = -1 + + Reader.call(this, props) +} + +DirReader.prototype._getEntries = function () { + var me = this + fs.readdir(me._path, function (er, entries) { + if (er) return me.error(er) + me._entries = entries + me._length = entries.length + if (typeof me.props.sort === "function") { + me._entries.sort(me.props.sort) + } + me._read() + }) +} + +// start walking the dir, and emit an "entry" event for each one. +DirReader.prototype._read = function () { + var me = this + + if (!me._entries) return me._getEntries() + + if (me._paused || me._currentEntry || me._aborted) { + // console.error("DR paused=%j, current=%j, aborted=%j", me._paused, !!me._currentEntry, me._aborted) + return + } + + me._index ++ + if (me._index >= me._length) { + // console.error(" DR End/close", me._path) + me.emit("end") + me.emit("close") + return + } + + // ok, handle this one, then. + + // save creating a proxy, by stat'ing the thing now. + var p = path.resolve(me._path, me._entries[me._index]) + // set this to prevent trying to _read() again in the stat time. + me._currentEntry = p + fs[ me.props.follow ? "stat" : "lstat" ](p, function (er, stat) { + if (er) return me.error(er) + + var entry = Reader({ path: p + , depth: me.depth + 1 + , root: me.root || me + , parent: me + , follow: me.follow + , filter: me.filter + }, stat) + + // console.error("DR Entry", p, stat.size) + + me._currentEntry = entry + + // "entry" events are for direct entries in a specific dir. + // "child" events are for any and all children at all levels. + // This nomenclature is not completely final. + + entry.on("pause", function (who) { + if (!me._paused) { + me.pause(who) + } + }) + + entry.on("resume", function (who) { + if (me._paused) { + me.resume(who) + } + }) + + entry.on("ready", function EMITCHILD () { + // console.error("DR emit child", entry._path) + if (me._paused) { + // console.error(" DR emit child - try again later") + // pause the child, and emit the "entry" event once we drain. + // console.error("DR pausing child entry") + entry.pause(me) + return me.once("resume", EMITCHILD) + } + + me.emit("entry", entry) + me.emit("child", entry) + }) + + var ended = false + entry.on("close", onend) + function onend () { + if (ended) return + ended = true + me.emit("childEnd", entry) + me.emit("entryEnd", entry) + me._currentEntry = null + me._read() + } + + // XXX Make this work in node. + // Long filenames should not break stuff. + entry.on("error", function (er) { + if (entry._swallowErrors) { + me.warn(er) + entry.emit("end") + entry.emit("close") + } else { + me.emit("error", er) + } + }) + + // proxy up some events. + ; [ "child" + , "childEnd" + , "warn" + ].forEach(function (ev) { + entry.on(ev, me.emit.bind(me, ev)) + }) + }) +} + +DirReader.prototype.pause = function (who) { + var me = this + if (me._paused) return + who = who || me + me._paused = true + if (me._currentEntry && me._currentEntry.pause) { + me._currentEntry.pause(who) + } + me.emit("pause", who) +} + +DirReader.prototype.resume = function (who) { + var me = this + if (!me._paused) return + who = who || me + + me._paused = false + // console.error("DR Emit Resume", me._path) + me.emit("resume", who) + if (me._paused) { + // console.error("DR Re-paused", me._path) + return + } + + if (me._currentEntry) { + if (me._currentEntry.resume) { + me._currentEntry.resume(who) + } + } else me._read() +} diff --git a/deps/npm/node_modules/fstream/lib/dir-writer.js b/deps/npm/node_modules/fstream/lib/dir-writer.js new file mode 100644 index 0000000000..01920244c1 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/dir-writer.js @@ -0,0 +1,165 @@ +// It is expected that, when .add() returns false, the consumer +// of the DirWriter will pause until a "drain" event occurs. Note +// that this is *almost always going to be the case*, unless the +// thing being written is some sort of unsupported type, and thus +// skipped over. + +module.exports = DirWriter + +var fs = require("graceful-fs") + , fstream = require("../fstream.js") + , Writer = require("./writer.js") + , inherits = require("inherits") + , mkdir = require("mkdirp") + , path = require("path") + , collect = require("./collect.js") + +inherits(DirWriter, Writer) + +function DirWriter (props) { + var me = this + if (!(me instanceof DirWriter)) me.error( + "DirWriter must be called as constructor.", null, true) + + // should already be established as a Directory type + if (props.type !== "Directory" || !props.Directory) { + me.error("Non-directory type "+ props.type + " " + + JSON.stringify(props), null, true) + } + + Writer.call(this, props) +} + +DirWriter.prototype._create = function () { + var me = this + mkdir(me._path, Writer.dirmode, function (er) { + if (er) return me.error(er) + // ready to start getting entries! + me.ready = true + me.emit("ready") + }) +} + +// a DirWriter has an add(entry) method, but its .write() doesn't +// do anything. Why a no-op rather than a throw? Because this +// leaves open the door for writing directory metadata for +// gnu/solaris style dumpdirs. +DirWriter.prototype.write = function () { + return true +} + +DirWriter.prototype.end = function () { + this._ended = true + this._process() +} + +DirWriter.prototype.add = function (entry) { + var me = this + + // console.error("\tadd", entry._path, "->", me._path) + collect(entry) + if (!me.ready || me._currentEntry) { + me._buffer.push(entry) + return false + } + + // create a new writer, and pipe the incoming entry into it. + if (me._ended) { + return me.error("add after end") + } + + me._buffer.push(entry) + me._process() + + return 0 === this._buffer.length +} + +DirWriter.prototype._process = function () { + var me = this + + // console.error("DW Process p=%j", me._processing, me.basename) + + if (me._processing) return + + var entry = me._buffer.shift() + if (!entry) { + // console.error("DW Drain") + me.emit("drain") + if (me._ended) me._finish() + return + } + + me._processing = true + // console.error("DW Entry", entry._path) + + me.emit("entry", entry) + + // ok, add this entry + // + // don't allow recursive copying + var p = entry + do { + if (p._path === me.root._path || p._path === me._path) { + // console.error("DW Exit (recursive)", entry.basename, me._path) + me._processing = false + if (entry._collected) entry.pipe() + return me._process() + } + } while (p = p.parent) + + // console.error("DW not recursive") + + // chop off the entry's root dir, replace with ours + var props = { parent: me + , root: me.root || me + , type: entry.type + , depth: me.depth + 1 } + + var p = entry._path || entry.path || entry.props.path + if (entry.parent) { + p = p.substr(entry.parent._path.length + 1) + } + // get rid of any ../../ shenanigans + props.path = path.join(me.path, path.join("/", p)) + + // all the rest of the stuff, copy over from the source. + Object.keys(entry.props).forEach(function (k) { + if (!props.hasOwnProperty(k)) { + props[k] = entry.props[k] + } + }) + + // not sure at this point what kind of writer this is. + var child = me._currentChild = new Writer(props) + child.on("ready", function () { + // console.error("DW Child Ready", child.type, child._path) + // console.error(" resuming", entry._path) + entry.pipe(child) + entry.resume() + }) + + // XXX Make this work in node. + // Long filenames should not break stuff. + child.on("error", function (er) { + if (child._swallowErrors) { + me.warn(er) + child.emit("end") + child.emit("close") + } else { + me.emit("error", er) + } + }) + + // we fire _end internally *after* end, so that we don't move on + // until any "end" listeners have had their chance to do stuff. + child.on("close", onend) + var ended = false + function onend () { + if (ended) return + ended = true + // console.error("* DW Child end", child.basename) + me._currentChild = null + me._processing = false + me._process() + } +} diff --git a/deps/npm/node_modules/fstream/lib/file-reader.js b/deps/npm/node_modules/fstream/lib/file-reader.js new file mode 100644 index 0000000000..b1f9861838 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/file-reader.js @@ -0,0 +1,147 @@ +// Basically just a wrapper around an fs.ReadStream + +module.exports = FileReader + +var fs = require("graceful-fs") + , fstream = require("../fstream.js") + , Reader = fstream.Reader + , inherits = require("inherits") + , mkdir = require("mkdirp") + , Reader = require("./reader.js") + , EOF = {EOF: true} + , CLOSE = {CLOSE: true} + +inherits(FileReader, Reader) + +function FileReader (props) { + // console.error(" FR create", props.path, props.size, new Error().stack) + var me = this + if (!(me instanceof FileReader)) throw new Error( + "FileReader must be called as constructor.") + + // should already be established as a File type + // XXX Todo: preserve hardlinks by tracking dev+inode+nlink, + // with a HardLinkReader class. + if (!((props.type === "Link" && props.Link) || + (props.type === "File" && props.File))) { + throw new Error("Non-file type "+ props.type) + } + + me._buffer = [] + me._bytesEmitted = 0 + Reader.call(me, props) +} + +FileReader.prototype._getStream = function () { + var me = this + , stream = me._stream = fs.createReadStream(me._path, me.props) + + if (me.props.blksize) { + stream.bufferSize = me.props.blksize + } + + stream.on("open", me.emit.bind(me, "open")) + + stream.on("data", function (c) { + // console.error("\t\t%d %s", c.length, me.basename) + me._bytesEmitted += c.length + // no point saving empty chunks + if (!c.length) return + else if (me._paused || me._buffer.length) { + me._buffer.push(c) + me._read() + } else me.emit("data", c) + }) + + stream.on("end", function () { + if (me._paused || me._buffer.length) { + // console.error("FR Buffering End", me._path) + me._buffer.push(EOF) + me._read() + } else { + me.emit("end") + } + + if (me._bytesEmitted !== me.props.size) { + me.error("Didn't get expected byte count\n"+ + "expect: "+me.props.size + "\n" + + "actual: "+me._bytesEmitted) + } + }) + + stream.on("close", function () { + if (me._paused || me._buffer.length) { + // console.error("FR Buffering Close", me._path) + me._buffer.push(CLOSE) + me._read() + } else { + // console.error("FR close 1", me._path) + me.emit("close") + } + }) + + me._read() +} + +FileReader.prototype._read = function () { + var me = this + // console.error("FR _read", me._path) + if (me._paused) { + // console.error("FR _read paused", me._path) + return + } + + if (!me._stream) { + // console.error("FR _getStream calling", me._path) + return me._getStream() + } + + // clear out the buffer, if there is one. + if (me._buffer.length) { + // console.error("FR _read has buffer", me._buffer.length, me._path) + var buf = me._buffer + for (var i = 0, l = buf.length; i < l; i ++) { + var c = buf[i] + if (c === EOF) { + // console.error("FR Read emitting buffered end", me._path) + me.emit("end") + } else if (c === CLOSE) { + // console.error("FR Read emitting buffered close", me._path) + me.emit("close") + } else { + // console.error("FR Read emitting buffered data", me._path) + me.emit("data", c) + } + + if (me._paused) { + // console.error("FR Read Re-pausing at "+i, me._path) + me._buffer = buf.slice(i) + return + } + } + me._buffer.length = 0 + } + // console.error("FR _read done") + // that's about all there is to it. +} + +FileReader.prototype.pause = function (who) { + var me = this + // console.error("FR Pause", me._path) + if (me._paused) return + who = who || me + me._paused = true + if (me._stream) me._stream.pause() + me.emit("pause", who) +} + +FileReader.prototype.resume = function (who) { + var me = this + // console.error("FR Resume", me._path) + if (!me._paused) return + who = who || me + me.emit("resume", who) + me._paused = false + if (me._stream) me._stream.resume() + me._read() +} diff --git a/deps/npm/node_modules/fstream/lib/file-writer.js b/deps/npm/node_modules/fstream/lib/file-writer.js new file mode 100644 index 0000000000..4ed0c9c6ca --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/file-writer.js @@ -0,0 +1,95 @@ +module.exports = FileWriter + +var fs = require("graceful-fs") + , mkdir = require("mkdirp") + , Writer = require("./writer.js") + , inherits = require("inherits") + , EOF = {} + +inherits(FileWriter, Writer) + +function FileWriter (props) { + var me = this + if (!(me instanceof FileWriter)) throw new Error( + "FileWriter must be called as constructor.") + + // should already be established as a File type + if (props.type !== "File" || !props.File) { + throw new Error("Non-file type "+ props.type) + } + + me._buffer = [] + me._bytesWritten = 0 + + Writer.call(this, props) +} + +FileWriter.prototype._create = function () { + var me = this + if (me._stream) return + + var so = {} + if (me.props.flags) so.flags = me.props.flags + so.mode = Writer.filemode + if (me._old && me._old.blksize) so.bufferSize = me._old.blksize + + me._stream = fs.createWriteStream(me._path, so) + + me._stream.on("open", function (fd) { + me.ready = true + me._buffer.forEach(function (c) { + if (c === EOF) me._stream.end() + else me._stream.write(c) + }) + me.emit("ready") + }) + + me._stream.on("drain", function () { me.emit("drain") }) + + me._stream.on("close", function () { + // console.error("\n\nFW Stream Close", me._path, me.size) + me._finish() + }) +} + +FileWriter.prototype.write = function (c) { + var me = this + + me._bytesWritten += c.length + + if (!me.ready) { + me._buffer.push(c) + return false + } + + var ret = me._stream.write(c) + // console.error("\t-- fw wrote, _stream says", ret, me._stream._queue.length) + + // allow 2 buffered writes, because otherwise there's just too + // much stop and go bs. + return ret || (me._stream._queue && me._stream._queue.length <= 2) +} + +FileWriter.prototype.end = function (c) { + var me = this + + if (c) me.write(c) + + if (!me.ready) { + me._buffer.push(EOF) + return false + } + + return me._stream.end() +} + +FileWriter.prototype._finish = function () { + var me = this + if (typeof me.size === "number" && me._bytesWritten != me.size) { + me.error( + "Did not get expected byte count.\n" + + "expect: " + me.size + "\n" + + "actual: " + me._bytesWritten) + } + Writer.prototype._finish.call(me) +} diff --git a/deps/npm/node_modules/fstream/lib/get-type.js b/deps/npm/node_modules/fstream/lib/get-type.js new file mode 100644 index 0000000000..cd65c41d8b --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/get-type.js @@ -0,0 +1,32 @@ +module.exports = getType + +function getType (st) { + var types = + [ "Directory" + , "File" + , "SymbolicLink" + , "Link" // special for hardlinks from tarballs + , "BlockDevice" + , "CharacterDevice" + , "FIFO" + , "Socket" ] + , type + + if (st.type && -1 !== types.indexOf(st.type)) { + st[st.type] = true + return st.type + } + + for (var i = 0, l = types.length; i < l; i ++) { + type = types[i] + var is = st[type] || st["is" + type] + if (typeof is === "function") is = is.call(st) + if (is) { + st[type] = true + st.type = type + return type + } + } + + return null +} diff --git a/deps/npm/node_modules/fstream/lib/link-reader.js b/deps/npm/node_modules/fstream/lib/link-reader.js new file mode 100644 index 0000000000..7e7ab6ce5c --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/link-reader.js @@ -0,0 +1,54 @@ +// Basically just a wrapper around an fs.readlink +// +// XXX: Enhance this to support the Link type, by keeping +// a lookup table of {:}, so that hardlinks +// can be preserved in tarballs. + +module.exports = LinkReader + +var fs = require("graceful-fs") + , fstream = require("../fstream.js") + , inherits = require("inherits") + , mkdir = require("mkdirp") + , Reader = require("./reader.js") + +inherits(LinkReader, Reader) + +function LinkReader (props) { + var me = this + if (!(me instanceof LinkReader)) throw new Error( + "LinkReader must be called as constructor.") + + if (!((props.type === "Link" && props.Link) || + (props.type === "SymbolicLink" && props.SymbolicLink))) { + throw new Error("Non-link type "+ props.type) + } + + Reader.call(me, props) +} + +// When piping a LinkReader into a LinkWriter, we have to +// already have the linkpath property set, so that has to +// happen *before* the "ready" event, which means we need to +// override the _stat method. +LinkReader.prototype._stat = function (currentStat) { + var me = this + fs.readlink(me._path, function (er, linkpath) { + if (er) return me.error(er) + me.linkpath = me.props.linkpath = linkpath + me.emit("linkpath", linkpath) + Reader.prototype._stat.call(me, currentStat) + }) +} + +LinkReader.prototype._read = function () { + var me = this + if (me._paused) return + // basically just a no-op, since we got all the info we need + // from the _stat method + if (!me._ended) { + me.emit("end") + me.emit("close") + me._ended = true + } +} diff --git a/deps/npm/node_modules/fstream/lib/link-writer.js b/deps/npm/node_modules/fstream/lib/link-writer.js new file mode 100644 index 0000000000..0ccce15094 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/link-writer.js @@ -0,0 +1,82 @@ + +module.exports = LinkWriter + +var fs = require("graceful-fs") + , Writer = require("./writer.js") + , inherits = require("inherits") + , collect = require("./collect.js") + , path = require("path") + , rimraf = require("rimraf") + +inherits(LinkWriter, Writer) + +function LinkWriter (props) { + var me = this + if (!(me instanceof LinkWriter)) throw new Error( + "LinkWriter must be called as constructor.") + + // should already be established as a Link type + if (!((props.type === "Link" && props.Link) || + (props.type === "SymbolicLink" && props.SymbolicLink))) { + throw new Error("Non-link type "+ props.type) + } + + if (!props.linkpath) { + me.error("Need linkpath property to create " + props.type) + } + + Writer.call(this, props) +} + +LinkWriter.prototype._create = function () { + var me = this + , hard = me.type === "Link" || process.platform === "win32" + , link = hard ? "link" : "symlink" + , lp = hard ? path.resolve(me.dirname, me.linkpath) : me.linkpath + + // can only change the link path by clobbering + // For hard links, let's just assume that's always the case, since + // there's no good way to read them if we don't already know. + if (hard) return clobber(me, lp, link) + + fs.readlink(me._path, function (er, p) { + // only skip creation if it's exactly the same link + if (p && p === lp) return finish(me) + clobber(me, lp, link) + }) +} + +function clobber (me, lp, link) { + rimraf(me._path, function (er) { + if (er) return me.error(er) + create(me, lp, link) + }) +} + +function create (me, lp, link) { + fs[link](lp, me._path, function (er) { + // if this is a hard link, and we're in the process of writing out a + // directory, it's very possible that the thing we're linking to + // doesn't exist yet (especially if it was intended as a symlink), + // so swallow ENOENT errors here and just soldier in. + if (er) { + if (er.code === "ENOENT" && process.platform === "win32") { + me.ready = true + me.emit("ready") + me.emit("end") + me.emit("close") + me.end = me._finish = function () {} + } else return me.error(er) + } + finish(me) + }) +} + +function finish (me) { + me.ready = true + me.emit("ready") +} + +LinkWriter.prototype.end = function () { + this._finish() +} diff --git a/deps/npm/node_modules/fstream/lib/proxy-reader.js b/deps/npm/node_modules/fstream/lib/proxy-reader.js new file mode 100644 index 0000000000..e9237d60b1 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/proxy-reader.js @@ -0,0 +1,87 @@ +// A reader for when we don't yet know what kind of thing +// the thing is. + +module.exports = ProxyReader + +var Reader = require("./reader.js") + , getType = require("./get-type.js") + , inherits = require("inherits") + , fs = require("graceful-fs") + +inherits(ProxyReader, Reader) + +function ProxyReader (props) { + var me = this + if (!(me instanceof ProxyReader)) throw new Error( + "ProxyReader must be called as constructor.") + + me.props = props + me._buffer = [] + me.ready = false + + Reader.call(me, props) +} + +ProxyReader.prototype._stat = function () { + var me = this + , props = me.props + // stat the thing to see what the proxy should be. + , stat = props.follow ? "stat" : "lstat" + + fs[stat](props.path, function (er, current) { + var type + if (er || !current) { + type = "File" + } else { + type = getType(current) + } + + props[type] = true + props.type = me.type = type + + me._old = current + me._addProxy(Reader(props, current)) + }) +} + +ProxyReader.prototype._addProxy = function (proxy) { + var me = this + if (me._proxy) { + return me.error("proxy already set") + } + + me._proxy = proxy + ; [ "error" + , "close" + , "data" + , "end" + , "close" + , "linkpath" + , "entry" + , "warn" + ].forEach(function (ev) { + proxy.on(ev, me.emit.bind(me, ev)) + }) + + me.emit("proxy", proxy) + + proxy.on("ready", function () { + // console.error("~~ proxy is ready!", me.path) + me.ready = true + me.emit("ready") + }) + + var calls = me._buffer + me._buffer.length = 0 + calls.forEach(function (c) { + proxy[c[0]].apply(proxy, c[1]) + }) +} + +ProxyReader.prototype.pause = function () { + return this._proxy ? this._proxy.pause() : false +} + +ProxyReader.prototype.resume = function (c) { + return this._proxy ? this._proxy.resume() : false +} diff --git a/deps/npm/node_modules/fstream/lib/proxy-writer.js b/deps/npm/node_modules/fstream/lib/proxy-writer.js new file mode 100644 index 0000000000..2c78fc6736 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/proxy-writer.js @@ -0,0 +1,109 @@ +// A writer for when we don't know what kind of thing +// the thing is. That is, it's not explicitly set, +// so we're going to make it whatever the thing already +// is, or "File" +// +// Until then, collect all events. + +module.exports = ProxyWriter + +var Writer = require("./writer.js") + , getType = require("./get-type.js") + , inherits = require("inherits") + , collect = require("./collect.js") + , fs = require("fs") + +inherits(ProxyWriter, Writer) + +function ProxyWriter (props) { + var me = this + if (!(me instanceof ProxyWriter)) throw new Error( + "ProxyWriter must be called as constructor.") + + me.props = props + me._needDrain = false + + Writer.call(me, props) +} + +ProxyWriter.prototype._stat = function () { + var me = this + , props = me.props + // stat the thing to see what the proxy should be. + , stat = props.follow ? "stat" : "lstat" + + fs[stat](props.path, function (er, current) { + var type + if (er || !current) { + type = "File" + } else { + type = getType(current) + } + + props[type] = true + props.type = me.type = type + + me._old = current + me._addProxy(Writer(props, current)) + }) +} + +ProxyWriter.prototype._addProxy = function (proxy) { + // console.error("~~ set proxy", this.path) + var me = this + if (me._proxy) { + return me.error("proxy already set") + } + + me._proxy = proxy + ; [ "ready" + , "error" + , "close" + , "pipe" + , "drain" + , "warn" + ].forEach(function (ev) { + proxy.on(ev, me.emit.bind(me, ev)) + }) + + me.emit("proxy", proxy) + + var calls = me._buffer + calls.forEach(function (c) { + // console.error("~~ ~~ proxy buffered call", c[0], c[1]) + proxy[c[0]].call(proxy, c[1]) + }) + me._buffer.length = 0 + if (me._needsDrain) me.emit("drain") +} + +ProxyWriter.prototype.add = function (entry) { + // console.error("~~ proxy add") + collect(entry) + + if (!this._proxy) { + this._buffer.push(["add", [entry]]) + this._needDrain = true + return false + } + return this._proxy.add(entry) +} + +ProxyWriter.prototype.write = function (c) { + // console.error("~~ proxy write") + if (!this._proxy) { + this._buffer.push(["write", [c]]) + this._needDrain = true + return false + } + return this._proxy.write(c) +} + +ProxyWriter.prototype.end = function (c) { + // console.error("~~ proxy end") + if (!this._proxy) { + this._buffer.push(["end", c]) + return false + } + return this._proxy.end(c) +} diff --git a/deps/npm/node_modules/fstream/lib/reader.js b/deps/npm/node_modules/fstream/lib/reader.js new file mode 100644 index 0000000000..592ec6e854 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/reader.js @@ -0,0 +1,234 @@ + +module.exports = Reader + +var fs = require("graceful-fs") + , Stream = require("stream").Stream + , inherits = require("inherits") + , path = require("path") + , getType = require("./get-type.js") + , hardLinks = Reader.hardLinks = {} + , Abstract = require("./abstract.js") + +// Must do this *before* loading the child classes +inherits(Reader, Abstract) + +var DirReader = require("./dir-reader.js") + , FileReader = require("./file-reader.js") + , LinkReader = require("./link-reader.js") + , ProxyReader = require("./proxy-reader.js") + +function Reader (props, currentStat) { + var me = this + if (!(me instanceof Reader)) return new Reader(props, currentStat) + + if (typeof props === "string") { + props = { path: props } + } + + if (!props.path) { + me.error("Must provide a path", null, true) + } + + // polymorphism. + // call fstream.Reader(dir) to get a DirReader object, etc. + // Note that, unlike in the Writer case, ProxyReader is going + // to be the *normal* state of affairs, since we rarely know + // the type of a file prior to reading it. + + + var type + , ClassType + + if (props.type && typeof props.type === "function") { + type = props.type + ClassType = type + } else { + type = getType(props) + ClassType = Reader + } + + if (currentStat && !type) { + type = getType(currentStat) + props[type] = true + props.type = type + } + + switch (type) { + case "Directory": + ClassType = DirReader + break + + case "Link": + // XXX hard links are just files. + // However, it would be good to keep track of files' dev+inode + // and nlink values, and create a HardLinkReader that emits + // a linkpath value of the original copy, so that the tar + // writer can preserve them. + // ClassType = HardLinkReader + // break + + case "File": + ClassType = FileReader + break + + case "SymbolicLink": + ClassType = LinkReader + break + + case null: + ClassType = ProxyReader + break + } + + if (!(me instanceof ClassType)) { + return new ClassType(props) + } + + Abstract.call(me) + + me.readable = true + me.writable = false + + me.type = type + me.props = props + me.depth = props.depth = props.depth || 0 + me.parent = props.parent || null + me.root = props.root || (props.parent && props.parent.root) || me + + me.basename = props.basename = path.basename(me.path) + me.dirname = props.dirname = path.dirname(me.path) + me._path = me.path = path.resolve(props.path) + if (process.platform === "win32") { + me.path = me._path = me.path.replace(/\?/g, "_") + if (me._path.length >= 260) { + // how DOES one create files on the moon? + // if the path has spaces in it, then UNC will fail. + me._swallowErrors = true + //if (me._path.indexOf(" ") === -1) { + me._path = "\\\\?\\" + me.path.replace(/\//g, "\\") + //} + } + } + + // these have served their purpose, and are now just noisy clutter + props.parent = props.root = null + + // console.error("\n\n\n%s setting size to", props.path, props.size) + me.size = props.size + me.filter = typeof props.filter === "function" ? props.filter : null + if (props.sort === "alpha") props.sort = alphasort + + // start the ball rolling. + // this will stat the thing, and then call me._read() + // to start reading whatever it is. + // console.error("calling stat", props.path, currentStat) + me._stat(currentStat) +} + +function alphasort (a, b) { + return a === b ? 0 + : a.toLowerCase() > b.toLowerCase() ? 1 + : a.toLowerCase() < b.toLowerCase() ? -1 + : a > b ? 1 + : -1 +} + +Reader.prototype._stat = function (currentStat) { + var me = this + , props = me.props + , stat = props.follow ? "stat" : "lstat" + + // console.error("Reader._stat", me._path, currentStat) + if (currentStat) process.nextTick(statCb.bind(null, null, currentStat)) + else fs[stat](me._path, statCb) + + + function statCb (er, props_) { + // console.error("Reader._stat, statCb", me._path, props_, props_.nlink) + if (er) return me.error(er) + + Object.keys(props_).forEach(function (k) { + props[k] = props_[k] + }) + + // if it's not the expected size, then abort here. + if (undefined !== me.size && props.size !== me.size) { + return me.error("incorrect size") + } + me.size = props.size + + var type = getType(props) + // special little thing for handling hardlinks. + if (type !== "Directory" && props.nlink && props.nlink > 1) { + var k = props.dev + ":" + props.ino + // console.error("Reader has nlink", me._path, k) + if (hardLinks[k] === me._path || !hardLinks[k]) hardLinks[k] = me._path + else { + // switch into hardlink mode. + type = me.type = me.props.type = "Link" + me.Link = me.props.Link = true + me.linkpath = me.props.linkpath = hardLinks[k] + // console.error("Hardlink detected, switching mode", me._path, me.linkpath) + // Setting __proto__ would arguably be the "correct" + // approach here, but that just seems too wrong. + me._stat = me._read = LinkReader.prototype._read + } + } + + if (me.type && me.type !== type) { + me.error("Unexpected type: " + type) + } + + // if the filter doesn't pass, then just skip over this one. + // still have to emit end so that dir-walking can move on. + if (me.filter) { + if (!me.filter()) { + me._aborted = true + me.emit("end") + me.emit("close") + return + } + } + + me.emit("ready", props) + + // if it's a directory, then we'll be emitting "entry" events. + me._read() + } +} + +Reader.prototype.pipe = function (dest, opts) { + var me = this + if (typeof dest.add === "function") { + // piping to a multi-compatible, and we've got directory entries. + me.on("entry", function (entry) { + var ret = dest.add(entry) + if (false === ret) { + me.pause() + } + }) + } + + // console.error("R Pipe apply Stream Pipe") + return Stream.prototype.pipe.apply(this, arguments) +} + +Reader.prototype.pause = function (who) { + this._paused = true + who = who || this + this.emit("pause", who) + if (this._stream) this._stream.pause(who) +} + +Reader.prototype.resume = function (who) { + this._paused = false + who = who || this + this.emit("resume", who) + if (this._stream) this._stream.resume(who) + this._read() +} + +Reader.prototype._read = function () { + me.warn("Cannot read unknown type: "+me.type) +} + diff --git a/deps/npm/node_modules/fstream/lib/writer.js b/deps/npm/node_modules/fstream/lib/writer.js new file mode 100644 index 0000000000..f280963aa1 --- /dev/null +++ b/deps/npm/node_modules/fstream/lib/writer.js @@ -0,0 +1,324 @@ + +module.exports = Writer + +var fs = require("graceful-fs") + , inherits = require("inherits") + , rimraf = require("rimraf") + , mkdir = require("mkdirp") + , path = require("path") + , umask = process.umask() + , getType = require("./get-type.js") + , Abstract = require("./abstract.js") + +// Must do this *before* loading the child classes +inherits(Writer, Abstract) + +Writer.dirmode = 0777 & (~umask) +Writer.filemode = 0666 & (~umask) + +var DirWriter = require("./dir-writer.js") + , LinkWriter = require("./link-writer.js") + , FileWriter = require("./file-writer.js") + , ProxyWriter = require("./proxy-writer.js") + +// props is the desired state. current is optionally the current stat, +// provided here so that subclasses can avoid statting the target +// more than necessary. +function Writer (props, current) { + var me = this + + if (typeof props === "string") { + props = { path: props } + } + + if (!props.path) me.error("Must provide a path", null, true) + + // polymorphism. + // call fstream.Writer(dir) to get a DirWriter object, etc. + var type = getType(props) + , ClassType = Writer + + switch (type) { + case "Directory": + ClassType = DirWriter + break + case "File": + ClassType = FileWriter + break + case "Link": + case "SymbolicLink": + ClassType = LinkWriter + break + case null: + // Don't know yet what type to create, so we wrap in a proxy. + ClassType = ProxyWriter + break + } + + if (!(me instanceof ClassType)) return new ClassType(props) + + // now get down to business. + + Abstract.call(me) + + // props is what we want to set. + // set some convenience properties as well. + me.type = props.type + me.props = props + me.depth = props.depth || 0 + me.clobber = false === props.clobber ? props.clobber : true + me.parent = props.parent || null + me.root = props.root || (props.parent && props.parent.root) || me + + me.basename = path.basename(props.path) + me.dirname = path.dirname(props.path) + me.linkpath = props.linkpath || null + me._path = me.path = path.resolve(props.path) + if (process.platform === "win32") { + me.path = me._path = me.path.replace(/\?/g, "_") + if (me._path.length >= 260) { + me._swallowErrors = true + //if (me._path.indexOf(" ") === -1) { + me._path = "\\\\?\\" + me.path.replace(/\//g, "\\") + //} + } + } + + props.parent = props.root = null + + // console.error("\n\n\n%s setting size to", props.path, props.size) + me.size = props.size + + if (typeof props.mode === "string") { + props.mode = parseInt(props.mode, 8) + } + + me.readable = false + me.writable = true + + // buffer until ready, or while handling another entry + me._buffer = [] + me.ready = false + + // start the ball rolling. + // this checks what's there already, and then calls + // me._create() to call the impl-specific creation stuff. + me._stat(current) +} + +// Calling this means that it's something we can't create. +// Just assert that it's already there, otherwise raise a warning. +Writer.prototype._create = function () { + var me = this + fs[me.props.follow ? "stat" : "lstat"](me._path, function (er, current) { + if (er) { + return me.warn("Cannot create " + me._path + "\n" + + "Unsupported type: "+me.type, "ENOTSUP") + } + me._finish() + }) +} + +Writer.prototype._stat = function (current) { + var me = this + , props = me.props + , stat = props.follow ? "stat" : "lstat" + + if (current) statCb(null, current) + else fs[stat](me._path, statCb) + + function statCb (er, current) { + // if it's not there, great. We'll just create it. + // if it is there, then we'll need to change whatever differs + if (er || !current) { + return create(me) + } + + me._old = current + var currentType = getType(current) + + // if it's a type change, then we need to clobber or error. + // if it's not a type change, then let the impl take care of it. + if (currentType !== me.type) { + return rimraf(me._path, function (er) { + if (er) return me.error(er) + me._old = null + create(me) + }) + } + + // otherwise, just handle in the app-specific way + // this creates a fs.WriteStream, or mkdir's, or whatever + me._create() + } +} + +function create (me) { + // XXX Need to clobber non-dirs that are in the way, + // unless { clobber: false } in the props. + mkdir(path.dirname(me._path), Writer.dirmode, function (er) { + if (er) return me.error(er) + me._create() + }) +} + +Writer.prototype._finish = function () { + var me = this + + // console.error(" W Finish", me._path, me.size) + + // set up all the things. + // At this point, we're already done writing whatever we've gotta write, + // adding files to the dir, etc. + var todo = 0 + var errState = null + var done = false + + if (me._old) { + // the times will almost *certainly* have changed. + // adds the utimes syscall, but remove another stat. + me._old.atime = new Date(0) + me._old.mtime = new Date(0) + // console.error(" W Finish Stale Stat", me._path, me.size) + setProps(me._old) + } else { + var stat = me.props.follow ? "stat" : "lstat" + // console.error(" W Finish Stating", me._path, me.size) + fs[stat](me._path, function (er, current) { + // console.error(" W Finish Stated", me._path, me.size, current) + if (er) { + // if we're in the process of writing out a + // directory, it's very possible that the thing we're linking to + // doesn't exist yet (especially if it was intended as a symlink), + // so swallow ENOENT errors here and just soldier in. + if (er.code === "ENOENT" && + (me.type === "Link" || me.type === "SymbolicLink") && + process.platform === "win32") { + me.ready = true + me.emit("ready") + me.emit("end") + me.emit("close") + me.end = me._finish = function () {} + return + } else return me.error(er) + } + setProps(me._old = current) + }) + } + + return + + function setProps (current) { + // console.error(" W setprops", me._path) + // mode + var wantMode = me.props.mode + , chmod = me.props.follow || me.type === "Directory" + ? "chmod" : "lchmod" + + if (fs[chmod] && typeof wantMode === "number") { + wantMode = wantMode & 0777 + todo ++ + // console.error(" W chmod", wantMode.toString(8), me.basename) + fs[chmod](me._path, wantMode, next(chmod)) + } + + // uid, gid + // Don't even try it unless root. Too easy to EPERM. + if (process.platform !== "win32" && + process.getuid && + process.getuid() === 0 && + fs.chown && + ( typeof me.props.uid === "number" || + typeof me.props.gid === "number")) { + if (typeof me.props.uid !== "number") me.props.uid = current.uid + if (typeof me.props.gid !== "number") me.props.gid = current.gid + if (me.props.uid !== current.uid || me.props.gid !== current.gid) { + todo ++ + // console.error(" W chown", me.props.uid, me.props.gid, me.basename) + fs.chown(me._path, me.props.uid, me.props.gid, next("chown")) + } + } + + // atime, mtime. + if (fs.utimes && + !(process.platform === "win32" && me.type === "Directory")) { + var utimes = (me.props.follow || me.type !== "SymbolicLink") + ? "utimes" : "lutimes" + + if (utimes === "lutimes" && !fs[utimes]) { + if (!fs.futimes) fs.ltimes = function (a, b, c, cb) { return cb() } + else fs.lutimes = function (path, atime, mtime, cb) { + var c = require("constants") + fs.open(path, c.O_SYMLINK, function (er, fd) { + if (er) return cb(er) + fs.futimes(fd, atime, mtime, function (er) { + if (er) return cb(er) + fs.close(fd, cb) + }) + }) + } + } + + var curA = current.atime + , curM = current.mtime + , meA = me.props.atime + , meM = me.props.mtime + + if (meA === undefined) meA = curA + if (meM === undefined) meM = curM + + if (!isDate(meA)) meA = new Date(meA) + if (!isDate(meM)) meA = new Date(meM) + + if (meA.getTime() !== curA.getTime() || + meM.getTime() !== curM.getTime()) { + todo ++ + // console.error(" W utimes", meA, meM, me.basename) + fs[utimes](me._path, meA, meM, next("utimes")) + } + } + + // finally, handle the case if there was nothing to do. + if (todo === 0) { + // console.error(" W nothing to do", me.basename) + next("nothing to do")() + } + } + + function next (what) { return function (er) { + // console.error(" W Finish", what, todo) + if (errState) return + if (er) { + er.fstream_finish_call = what + return me.error(errState = er) + } + if (--todo > 0) return + if (done) return + done = true + + // all the props have been set, so we're completely done. + me.emit("end") + me.emit("close") + }} +} + +Writer.prototype.pipe = function () { + this.error("Can't pipe from writable stream") +} + +Writer.prototype.add = function () { + this.error("Cannot add to non-Directory type") +} + +Writer.prototype.write = function () { + return true +} + +function objectToString (d) { + return Object.prototype.toString.call(d) +} + +function isDate(d) { + return typeof d === 'object' && objectToString(d) === '[object Date]'; +} + diff --git a/deps/npm/node_modules/fstream/package.json b/deps/npm/node_modules/fstream/package.json new file mode 100644 index 0000000000..22d5821f62 --- /dev/null +++ b/deps/npm/node_modules/fstream/package.json @@ -0,0 +1,21 @@ +{ + "author": "Isaac Z. Schlueter (http://blog.izs.me/)", + "name": "fstream", + "description": "Advanced file system stream things", + "version": "0.0.1", + "repository": { + "type": "git", + "url": "git://github.com/isaacs/fstream.git" + }, + "main": "fstream.js", + "engines": { + "node": "0.5 || 0.6" + }, + "dependencies": { + "rimraf": "~1.0.8", + "mkdirp": "~0.1.0", + "graceful-fs": "~1.0.1", + "inherits": "~1.0.0" + }, + "devDependencies": {} +} diff --git a/deps/npm/node_modules/graceful-fs/LICENSE b/deps/npm/node_modules/graceful-fs/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/graceful-fs/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/graceful-fs/README.md b/deps/npm/node_modules/graceful-fs/README.md new file mode 100644 index 0000000000..7d2e681e32 --- /dev/null +++ b/deps/npm/node_modules/graceful-fs/README.md @@ -0,0 +1,5 @@ +Just like node's `fs` module, but it does an incremental back-off when +EMFILE is encountered. + +Useful in asynchronous situations where one needs to try to open lots +and lots of files. diff --git a/deps/npm/node_modules/graceful-fs/graceful-fs.js b/deps/npm/node_modules/graceful-fs/graceful-fs.js new file mode 100644 index 0000000000..08e99b24b0 --- /dev/null +++ b/deps/npm/node_modules/graceful-fs/graceful-fs.js @@ -0,0 +1,39 @@ +// wrapper around the non-sync fs functions to gracefully handle +// having too many file descriptors open. Note that this is +// *only* possible because async patterns let one interject timeouts +// and other cleverness anywhere in the process without disrupting +// anything else. +var fs = require("fs") + , timeout = 0 + +Object.keys(fs) + .forEach(function (i) { + exports[i] = (typeof fs[i] !== "function") ? fs[i] + : (i.match(/^[A-Z]|^create|Sync$/)) ? function () { + return fs[i].apply(fs, arguments) + } + : graceful(fs[i]) + }) + +if (process.platform === "win32" + && !process.binding("fs").lstat) { + exports.lstat = exports.stat + exports.lstatSync = exports.statSync +} + +function graceful (fn) { return function GRACEFUL () { + var args = Array.prototype.slice.call(arguments) + , cb_ = args.pop() + args.push(cb) + function cb (er) { + if (er && er.message.match(/^EMFILE, Too many open files/)) { + setTimeout(function () { + GRACEFUL.apply(fs, args) + }, timeout ++) + return + } + timeout = 0 + cb_.apply(null, arguments) + } + fn.apply(fs, args) +}} diff --git a/deps/npm/node_modules/graceful-fs/package.json b/deps/npm/node_modules/graceful-fs/package.json new file mode 100644 index 0000000000..934829c5cc --- /dev/null +++ b/deps/npm/node_modules/graceful-fs/package.json @@ -0,0 +1,16 @@ +{ + "author": "Isaac Z. Schlueter (http://blog.izs.me)", + "name": "graceful-fs", + "description": "fs with incremental backoff on EMFILE", + "version": "1.0.1", + "repository": { + "type": "git", + "url": "git://github.com/isaacs/node-graceful-fs.git" + }, + "main": "graceful-fs.js", + "engines": { + "node": "0.4 || 0.5 || 0.6" + }, + "dependencies": {}, + "devDependencies": {} +} diff --git a/deps/npm/node_modules/inherits/LICENSE b/deps/npm/node_modules/inherits/LICENSE new file mode 100644 index 0000000000..c78c4f6618 --- /dev/null +++ b/deps/npm/node_modules/inherits/LICENSE @@ -0,0 +1,26 @@ +Copyright 2011 Isaac Z. Schlueter (the "Author") +All rights reserved. + +General Public Obviousness License + +The Author asserts that this software and associated documentation +files (the "Software"), while the Author's original creation, is +nonetheless obvious, trivial, unpatentable, and implied by the +context in which the software was created. If you sat down and +thought about the problem for an hour or less, you'd probably +come up with exactly this solution. + +Permission is granted to use this software in any way +whatsoever, with the following restriction: + +You may not release the Software under a more restrictive license +than this one. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/inherits/README.md b/deps/npm/node_modules/inherits/README.md new file mode 100644 index 0000000000..b2beaed93a --- /dev/null +++ b/deps/npm/node_modules/inherits/README.md @@ -0,0 +1,51 @@ +A dead simple way to do inheritance in JS. + + var inherits = require("inherits") + + function Animal () { + this.alive = true + } + Animal.prototype.say = function (what) { + console.log(what) + } + + inherits(Dog, Animal) + function Dog () { + Dog.super.apply(this) + } + Dog.prototype.sniff = function () { + this.say("sniff sniff") + } + Dog.prototype.bark = function () { + this.say("woof woof") + } + + inherits(Chihuahua, Dog) + function Chihuahua () { + Chihuahua.super.apply(this) + } + Chihuahua.prototype.bark = function () { + this.say("yip yip") + } + + // also works + function Cat () { + Cat.super.apply(this) + } + Cat.prototype.hiss = function () { + this.say("CHSKKSS!!") + } + inherits(Cat, Animal, { + meow: function () { this.say("miao miao") } + }) + Cat.prototype.purr = function () { + this.say("purr purr") + } + + + var c = new Chihuahua + assert(c instanceof Chihuahua) + assert(c instanceof Dog) + assert(c instanceof Animal) + +The actual function is laughably small. 10-lines small. diff --git a/deps/npm/node_modules/inherits/inherits-old.js b/deps/npm/node_modules/inherits/inherits-old.js new file mode 100644 index 0000000000..ef39252dd1 --- /dev/null +++ b/deps/npm/node_modules/inherits/inherits-old.js @@ -0,0 +1,40 @@ +// This is a less perfect implementation of the inherits function, +// designed to work in cases where ES5 is not available. +// +// Note that it is a bit longer, and doesn't properly deal with +// getter/setters or property descriptor flags (enumerable, etc.) + +module.exports = inheritsOld + +function inheritsOld (c, p, proto) { + function F () { this.constructor = c } + F.prototype = p.prototype + var e = {} + for (var i in c.prototype) if (c.prototype.hasOwnProperty(i)) { + e[i] = c.prototype[i] + } + if (proto) for (var i in proto) if (proto.hasOwnProperty(i)) { + e[i] = proto[i] + } + c.prototype = new F() + for (var i in e) if (e.hasOwnProperty(i)) { + c.prototype[i] = e[i] + } + c.super = p +} + +// function Child () { +// Child.super.call(this) +// console.error([this +// ,this.constructor +// ,this.constructor === Child +// ,this.constructor.super === Parent +// ,Object.getPrototypeOf(this) === Child.prototype +// ,Object.getPrototypeOf(Object.getPrototypeOf(this)) +// === Parent.prototype +// ,this instanceof Child +// ,this instanceof Parent]) +// } +// function Parent () {} +// inheritsOld(Child, Parent) +// new Child diff --git a/deps/npm/node_modules/inherits/inherits.js b/deps/npm/node_modules/inherits/inherits.js new file mode 100644 index 0000000000..061b396207 --- /dev/null +++ b/deps/npm/node_modules/inherits/inherits.js @@ -0,0 +1,29 @@ +module.exports = inherits + +function inherits (c, p, proto) { + proto = proto || {} + var e = {} + ;[c.prototype, proto].forEach(function (s) { + Object.getOwnPropertyNames(s).forEach(function (k) { + e[k] = Object.getOwnPropertyDescriptor(s, k) + }) + }) + c.prototype = Object.create(p.prototype, e) + c.super = p +} + +//function Child () { +// Child.super.call(this) +// console.error([this +// ,this.constructor +// ,this.constructor === Child +// ,this.constructor.super === Parent +// ,Object.getPrototypeOf(this) === Child.prototype +// ,Object.getPrototypeOf(Object.getPrototypeOf(this)) +// === Parent.prototype +// ,this instanceof Child +// ,this instanceof Parent]) +//} +//function Parent () {} +//inherits(Child, Parent) +//new Child diff --git a/deps/npm/node_modules/inherits/package.json b/deps/npm/node_modules/inherits/package.json new file mode 100644 index 0000000000..7dc32771b9 --- /dev/null +++ b/deps/npm/node_modules/inherits/package.json @@ -0,0 +1,8 @@ +{ "name" : "inherits" +, "description": "A tiny simple way to do classic inheritance in js" +, "version" : "1.0.0" +, "keywords" : ["inheritance", "class", "klass", "oop", "object-oriented"] +, "main" : "./inherits.js" +, "repository" : "https://github.com/isaacs/inherits" +, "license": { "type": "GPOL", "url": "https://raw.github.com/isaacs/inherits/master/LICENSE" } +, "author" : "Isaac Z. Schlueter (http://blog.izs.me/)" } diff --git a/deps/npm/node_modules/ini/LICENSE b/deps/npm/node_modules/ini/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/ini/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/ini/README.md b/deps/npm/node_modules/ini/README.md new file mode 100644 index 0000000000..cd434b7ae7 --- /dev/null +++ b/deps/npm/node_modules/ini/README.md @@ -0,0 +1,71 @@ +An ini format parser and serializer for node. + +Sections are treated as nested objects. Items before the first heading +are saved on the object directly. + +## Usage + +Consider an ini-file `config.ini` that looks like this: + + ; this comment is beeing ignored + scope = global + + [database] + user = dbuser + password = dbpassword + database = use_this_database + + [paths.default] + datadir = /var/lib/data + +You can read, manipulate and write the ini-file like so: + + var fs = require('fs') + , ini = require('ini') + + var config = ini.parse(fs.readFileSync('./config.ini', 'utf-8')) + + config.scope = 'local' + config.database.database = 'use_another_database' + config.paths.default.tmpdir = '/tmp' + delete config.paths.default.datadir + + fs.writeFileSync('./config_modified.ini', ini.stringify(config, 'section')) + +This will result in a file called `config_modified.ini` being written to the filesystem with the following content: + + [section] + scope = local + [section.database] + user = dbuser + password = dbpassword + database = use_another_database + [section.paths.default] + tmpdir = /tmp + +## API + +### decode(inistring) +Decode the ini-style formatted `inistring` into a nested object. + +### parse(inistring) +Alias for `decode(inistring)` + +### encode(object, [section]) +Encode the object `object` into an ini-style formatted string. If the optional parameter `section` is given, then all top-level properties of the object are put into this section and the `section`-string is prepended to all sub-sections, see the usage example above. + +### stringify(object, [section]) +Alias for `encode(object, [section])` + +### safe(val) +Escapes the string `val` such that it is safe to be used as a key or value in an ini-file. Basically escapes quotes. For example + + ini.safe('"unsafe string"') + +would result in + + "\"unsafe string\"" + +### unsafe(val) +Unescapes the string `val` + diff --git a/deps/npm/node_modules/ini/ini.js b/deps/npm/node_modules/ini/ini.js new file mode 100644 index 0000000000..468511a014 --- /dev/null +++ b/deps/npm/node_modules/ini/ini.js @@ -0,0 +1,97 @@ + +exports.parse = exports.decode = decode +exports.stringify = exports.encode = encode + +exports.safe = safe +exports.unsafe = unsafe + +function encode (obj, section) { + var children = [] + , out = "" + + Object.keys(obj).forEach(function (k, _, __) { + var val = obj[k] + if (val && typeof val === "object") { + children.push(k) + } else { + out += safe(k) + " = " + safe(val) + "\n" + } + }) + + if (section && out.length) { + out = "[" + safe(section) + "]" + "\n" + out + } + + children.forEach(function (k, _, __) { + out += encode(obj[k], (section ? section + "." : "") + k) + }) + + return out +} + +function decode (str) { + var out = {} + , p = out + , section = null + , state = "START" + // section |key = value + , re = /^\[([^\]]*)\]$|^([^=]+)(=(.*))?$/i + , lines = str.split(/[\r\n]+/g) + , section = null + + lines.forEach(function (line, _, __) { + //line = line + var rem = line.indexOf(";") + if (rem !== -1) line = line.substr(0, rem)//.trim() + if (!line) return + var match = line.match(re) + if (!match) return + if (match[1] !== undefined) { + section = unsafe(match[1]) + p = out[section] = out[section] || {} + return + } + var key = unsafe(match[2]) + , value = match[3] ? unsafe((match[4] || "")) : true + p[key] = value + }) + + // {a:{y:1},"a.b":{x:2}} --> {a:{y:1,b:{x:2}}} + // use a filter to return the keys that have to be deleted. + Object.keys(out).filter(function (k, _, __) { + if (!out[k] || typeof out[k] !== "object") return false + // see if the parent section is also an object. + // if so, add it to that, and mark this one for deletion + var parts = k.split(".") + , p = out + , l = parts.pop() + parts.forEach(function (part, _, __) { + if (!p[part] || typeof p[part] !== "object") p[part] = {} + p = p[part] + }) + if (p === out) return false + p[l] = out[k] + return true + }).forEach(function (del, _, __) { + delete out[del] + }) + + return out +} + +function safe (val) { + return ( typeof val !== "string" + || val.match(/[\r\n]/) + || (val.length > 1 + && val.charAt(0) === "\"" + && val.slice(-1) === "\"") + || val !== val.trim() ) ? JSON.stringify(val) : val +} + +function unsafe (val) { + val = (val || "").trim() + if (val.charAt(0) === "\"" && val.slice(-1) === "\"") { + try { val = JSON.parse(val) } catch (_) {} + } + return val +} diff --git a/deps/npm/node_modules/ini/package.json b/deps/npm/node_modules/ini/package.json new file mode 100644 index 0000000000..6bf81b8fbc --- /dev/null +++ b/deps/npm/node_modules/ini/package.json @@ -0,0 +1,24 @@ +{ + "author": "Isaac Z. Schlueter (http://blog.izs.me/)", + "name": "ini", + "description": "An ini encoder/decoder for node", + "version": "1.0.1", + "repository": { + "type": "git", + "url": "git://github.com/isaacs/ini.git" + }, + "main": "ini.js", + "scripts": { + "test": "node ini.js" + }, + "engines": { + "node": "*" + }, + "dependencies": {}, + "devDependencies": { + "tap": "~0.0.9" + }, + "scripts": { + "test": "tap test/*.js" + } +} diff --git a/deps/npm/node_modules/ini/test/fixtures/foo.ini b/deps/npm/node_modules/ini/test/fixtures/foo.ini new file mode 100644 index 0000000000..aa4b177a01 --- /dev/null +++ b/deps/npm/node_modules/ini/test/fixtures/foo.ini @@ -0,0 +1,18 @@ +o = p + + a with spaces = b c + +; wrap in quotes to JSON-decode and preserve spaces +" xa n p " = "\"\r\nyoyoyo\r\r\n" + +; a section +[a] +av = a val +e = { o: p, a: { av: a val, b: { c: { e: "this value" } } } } +j = "{ o: "p", a: { av: "a val", b: { c: { e: "this value" } } } }" + +; nested child without middle parent +; should create otherwise-empty a.b +[a.b.c] +e = 1 +j = 2 diff --git a/deps/npm/node_modules/ini/test/foo.js b/deps/npm/node_modules/ini/test/foo.js new file mode 100644 index 0000000000..31af10320f --- /dev/null +++ b/deps/npm/node_modules/ini/test/foo.js @@ -0,0 +1,40 @@ +var i = require("../") + , tap = require("tap") + , test = tap.test + , fs = require("fs") + , path = require("path") + , fixture = path.resolve(__dirname, "./fixtures/foo.ini") + , data = fs.readFileSync(fixture, "utf8") + , d + , expectE = 'o = p\n' + + 'a with spaces = b c\n' + + '" xa n p " = "\\"\\r\\nyoyoyo\\r\\r\\n"\n' + + '[a]\n' + + 'av = a val\n' + + 'e = { o: p, a: ' + + '{ av: a val, b: { c: { e: "this value" ' + + '} } } }\nj = "\\"{ o: \\"p\\", a: { av:' + + ' \\"a val\\", b: { c: { e: \\"this value' + + '\\" } } } }\\""\n[a.b.c]\ne = 1\nj = 2\n' + , expectD = + { o: 'p', + 'a with spaces': 'b c', + " xa n p ":'"\r\nyoyoyo\r\r\n', + a: + { av: 'a val', + e: '{ o: p, a: { av: a val, b: { c: { e: "this value" } } } }', + j: '"{ o: "p", a: { av: "a val", b: { c: { e: "this value" } } } }"', + b: { c: { e: '1', j: '2' } } } + } + +test("decode from file", function (t) { + d = i.decode(data) + t.deepEqual(d, expectD) + t.end() +}) + +test("encode from data", function (t) { + e = i.encode(expectD) + t.deepEqual(e, expectE) + t.end() +}) diff --git a/deps/npm/node_modules/minimatch/.gitmodules b/deps/npm/node_modules/minimatch/.gitmodules new file mode 100644 index 0000000000..7e29c80651 --- /dev/null +++ b/deps/npm/node_modules/minimatch/.gitmodules @@ -0,0 +1,3 @@ +[submodule "node_modules/lru-cache"] + path = node_modules/lru-cache + url = https://github.com/isaacs/node-lru-cache.git diff --git a/deps/npm/node_modules/minimatch/LICENSE b/deps/npm/node_modules/minimatch/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/minimatch/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/minimatch/README.md b/deps/npm/node_modules/minimatch/README.md new file mode 100644 index 0000000000..f17ec599d5 --- /dev/null +++ b/deps/npm/node_modules/minimatch/README.md @@ -0,0 +1,107 @@ +This is the matching library used internally by npm. + +Eventually, it will replace the C binding in node-glob. + +It works by converting glob expressions into JavaScript `RegExp` +objects. + +## Usage + +```javascript +var minimatch = require("minimatch") + +minimatch("bar.foo", "*.foo") // true! +minimatch("bar.foo", "*.bar") // false! +``` + +## Features + +Supports all glob features. + +See: + +* `man sh` +* `man fnmatch` +* `man 5 gitignore` + +### Departures from zsh/bash/ksh/sh + +If the pattern starts with a `!` character, then it is negated. + +If a pattern starts with `#`, then it is treated as a comment, and +will not match anything. (Use `\#` to match a literal `#` at the +start of a line.) + +The double-star `**` is always supported, instead of requiring a special +flag. + +If an escaped pattern has no matches, and the `null` flag is not set, +then minimatch.match returns the pattern as-provided, rather than +interpreting the character escapes. For example, +`minimatch.match([], "\\*a\\?")` will return `"\\*a\\?"` rather than +`"*a?"`. + +## Functions + +### minimatch(path, pattern, options) + +Main export. Tests a path against +the pattern using the options. + +### minimatch.filter(pattern, options) + +Returns a function that tests its +supplied argument, suitable for use with `Array.filter`. + +### minimatch.match(list, pattern, options) + +Match against the list of +files, in the style of fnmatch or glob. If nothing is matched, then +return the pattern (unless `{ null: true }` in the options.) + +### minimatch.makeRe(pattern, options) + +Make a regular expression object +from the pattern. + +## Options + +All options are `false` by default. + +### debug + +Dump a ton of stuff to stderr. + +### null + +Return an empty list from minimatch.match, instead of a list +containing the pattern itself. + +### nocase + +Perform a case-insensitive match. + +### cache + +An LRU cache with `.get(k)` and `.set(k,v)` methods. By +default, an instance of `node-lru-cache` is used, with 1000 max +entries. + +### slash + +If set, then `a/*` will match `a/` as well as `a/b`. + +### matchBase + +If set, then patterns without slashes will be matched +against the basename of the path if it contains slashes. For example, +`a?b` would match `xyz/123/acb`. + +### partial + +Internal. Used by `minimatch.makeRe`. + +### dot + +Allow patterns to match paths starting with a period, even if +the pattern does not explicitly start with a period. diff --git a/deps/npm/node_modules/minimatch/minimatch.js b/deps/npm/node_modules/minimatch/minimatch.js new file mode 100644 index 0000000000..76e7d6def0 --- /dev/null +++ b/deps/npm/node_modules/minimatch/minimatch.js @@ -0,0 +1,399 @@ +// This is a JavaScript implementation of the fnmatch-like +// stuff that git uses in its .gitignore files. +// See `man 5 gitignore`. + +module.exports = minimatch + +var path = require("path") + , LRU = require("lru-cache") + +minimatch.filter = function (pattern, options) { + options = options || {} + return function (p, i, list) { + return minimatch(p, pattern, options) + } +} + +minimatch.match = function (list, pattern, options) { + if (!options) options = {} + var ret = list.filter(minimatch.filter(pattern, options)) + + // set the null flag to allow empty match sets + // Note that minimatch itself, and filter(), do not + // respect this flag, only minimatch.match(list, pattern) does. + if (!options.null && !ret.length) { + return [pattern] + } + + return ret +} + +function minimatch (p, pattern, options) { + if (typeof pattern !== "string") { + throw new TypeError("glob pattern string required") + } + + options = options || {} + + // to set the cache, just replace with a different obj + // supporting set(k,v) and v=get(k) methods. + var cache = options.cache || minimatch.cache + if (!cache) cache = minimatch.cache = new LRU(1000) + + // "" only matches "" + if (!pattern) return p === "" + + // comments. + if (pattern.trim().charAt(0) === "#") return false + + // check the cache + var re = cache.get(pattern) + if (!re && re !== false) { + cache.set(pattern, re = minimatch.makeRe(pattern, options)) + } + + if (options.debug) { + console.error(pattern + "\t" + re, JSON.stringify(p)) + } + + // some kind of invalid thing + if (!re) return false + + + // patterns that end in / can only match dirs + // however, dirs also match the same thing that *doesn't* + // end in a slash. + var match = + // a/ should not match a/*, but will match */ + // accomplish this by not applying the regexp + // directly, unless the pattern would match + // trailing slash'ed things, or the thing isn't + // a trailing slash, or slashes are opted-in + ( ( options.slash || + p.substr(-1) !== "/" || + pattern.substr(-1) === "/" ) + && !!p.match(re) ) + + // a/ should match * or a + || ( p.substr(-1) === "/" && + !!p.slice(0, -1).match(re) ) + + // a pattern with *no* slashes will match against + // either the full path, or just the basename. + || ( options.matchBase && + pattern.indexOf("/") === -1 && + path.basename(p).match(re) ) + + //console.error(" MINIMATCH: %j %j %j %j", + // re.toString(), pattern, p, match) + return match +} + +minimatch.makeRe = makeRe +function makeRe (pattern, options) { + options = options || {} + + var braceDepth = 0 + , re = "" + , escaping = false + , oneStar = "[^\\/]*?" + , twoStar = ".*?" + , reSpecials = "().*{}+?[]^$/\\" + , patternListStack = [] + , stateChar + , negate = false + , negating = false + , inClass = false + , reClassStart = [] + + for ( var i = 0, len = pattern.length, c + ; (i < len) && (c = pattern.charAt(i)) + ; i ++ ) { + + if (options.debug) { + console.error("%s\t%s %s %j", pattern, i, re, c) + } + + switch (c) { + case "\\": + if (stateChar) { + if (stateChar === "*") re += oneStar + else re += "\\" + stateChar + stateChar = false + } + if (escaping) { + re += "\\\\" // must match literal \ + escaping = false + } else { + escaping = true + } + continue + + case "!": + if (i === 0 || negating) { + negate = !negate + negating = true + break + } else { + negating = false + } + // fallthrough + case "+": + case "@": + case "*": + case "?": + if (options.debug) { + console.error("%s\t%s %s %j <-- stateChar", pattern, i, re, c) + } + + negating = false + if (escaping) { + re += "\\" + c + escaping = false + } else if (inClass) { + re += c + } else if (c === "*" && stateChar === "*") { // ** + re += twoStar + stateChar = false + } else { + if (stateChar) { + if (stateChar === "*") re += oneStar + else if (stateChar === "?") re += "." + else re += "\\" + stateChar + } + stateChar = c + } + continue + + case "(": + if (escaping) { + re += "\\(" + escaping = false + } else if (inClass) { + re += "(" + } else if (stateChar) { + plType = stateChar + patternListStack.push(plType) + re += stateChar === "!" ? "(?!" : "(?:" + stateChar = false + } else { + re += "\\(" + } + continue + + case ")": + if (escaping || inClass) { + re += "\\)" + escaping = false + } else if (patternListStack.length) { + re += ")" + plType = patternListStack.pop() + switch (plType) { + case "?": + case "+": + case "*": re += plType + case "!": + case "@": break + } + } else { + re += "\\)" + } + continue + + case "|": + if (escaping || inClass) { + re += "\\|" + escaping = false + } else if (patternListStack.length) { + re += "|" + } else { + re += "\\|" + } + continue + + // these are mostly the same in regexp and glob :) + case "[": + if (stateChar) { + // some state-tracking char was before the [ + switch (stateChar) { + case "*": + re += oneStar + break + case "?": + re += "." + break + default: + re += "\\"+stateChar + break + } + stateChar = false + } + + if (escaping || inClass) { + re += "\\" + c + escaping = false + } else { + inClass = true + classStart = i + reClassStart = re.length + re += c + } + continue + + case "]": + // a right bracket shall lose its special + // meaning and represent itself in + // a bracket expression if it occurs + // first in the list. -- POSIX.2 2.8.3.2 + if (i === classStart + 1) escaping = true + + if (escaping || !inClass) { + re += "\\" + c + escaping = false + } else { + inClass = false + re += c + } + continue + + case "{": + if (escaping || inClass) { + re += "\\{" + escaping = false + } else { + re += "(?:" + braceDepth ++ + } + continue + + case "}": + if (escaping || inClass || braceDepth === 0) { + re += "\\}" + escaping = false + } else { + re += ")" + braceDepth -- + } + continue + + case ",": + if (escaping || inClass || braceDepth === 0) { + re += "," + escaping = false + } else { + re += "|" + } + continue + + default: + if (stateChar) { + // we had some state-tracking character + // that wasn't consumed by this pass. + switch (stateChar) { + case "*": + re += oneStar + break + case "?": + re += "." + break + default: + re += "\\"+stateChar + break + } + stateChar = false + } + + if (escaping) { + // no need + escaping = false + } else if (reSpecials.indexOf(c) !== -1 + && !(c === "^" && inClass)) { + re += "\\" + } + + re += c + + } // switch + + if (negating && c !== "!") negating = false + + } // for + + // handle trailing things that only matter at the very end. + if (stateChar) { + // we had some state-tracking character + // that wasn't consumed by this pass. + switch (stateChar) { + case "*": + re += oneStar + break + case "?": + re += "." + break + default: + re += "\\"+stateChar + break + } + stateChar = false + } else if (escaping) { + re += "\\\\" + } + + // "[abc" is valid, equivalent to "\[abc" + if (inClass) { + // split where the last [ was, and escape it + // this is a huge pita. We now have to re-walk + // the contents of the would-be class to re-translate + // any characters that were passed through as-is + var cs = re.substr(reClassStart + 1) + , csOpts = Object.create(options) + csOpts.partial = true + + re = re.substr(0, reClassStart) + "\\[" + + makeRe(cs, csOpts) + } + + if (options.partial) return re + + // don't match "." files unless pattern starts with "." + if (!options.dot && pattern.charAt(0) !== ".") { + re = "(?!\\.)" + re + } + + // must match entire pattern + // ending in a * or ** will make it less strict. + re = "^" + re + "$" + + // fail on the pattern, but allow anything otherwise. + if (negate) re = "^(?!" + re + ").*$" + + // really insane glob patterns can cause bad things. + var flags = "" + if (options.nocase) flags += "i" + + if (options.debug) { + console.error("/%s/%s", re, flags) + } + + try { + return new RegExp(re, flags) + } catch(ex) { + return false + } +} + +if (require.main === module) { + // more tests in test/*.js + var tests = ["{a,b{c,d}}" + ,"a.*$?" + ,"\\{a,b{c,d}}" + ,"a/{c/,}d/{e/,f/{g,h,i}/}k" + ,"!*.bak" + ,"!!*.bak" + ,"!!!*.bak" + ,"\\a\\b\\c\\d" + ] + tests.forEach(function (t) { + console.log([t,makeRe(t)]) + }) +} diff --git a/deps/npm/node_modules/minimatch/node_modules/lru-cache/LICENSE b/deps/npm/node_modules/minimatch/node_modules/lru-cache/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/minimatch/node_modules/lru-cache/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/minimatch/node_modules/lru-cache/README.md b/deps/npm/node_modules/minimatch/node_modules/lru-cache/README.md new file mode 100644 index 0000000000..1f5f155b7f --- /dev/null +++ b/deps/npm/node_modules/minimatch/node_modules/lru-cache/README.md @@ -0,0 +1,12 @@ +# lru cache + +A cache object that deletes the least-recently-used items. + +Usage: + + var LRU = require("lru-cache") + , cache = LRU(10) // max 10 items. default = Infinity + cache.set("key", "value") + cache.get("key") // "value" + +RTFS for more info. diff --git a/deps/npm/node_modules/minimatch/node_modules/lru-cache/lib/lru-cache.js b/deps/npm/node_modules/minimatch/node_modules/lru-cache/lib/lru-cache.js new file mode 100644 index 0000000000..39b6c2d88b --- /dev/null +++ b/deps/npm/node_modules/minimatch/node_modules/lru-cache/lib/lru-cache.js @@ -0,0 +1,151 @@ + +module.exports = LRUCache + +function hOP (obj, key) { + return Object.prototype.hasOwnProperty.call(obj, key) +} + +function LRUCache (maxLength) { + if (!(this instanceof LRUCache)) { + return new LRUCache(maxLength) + } + var cache = {} // hash of items by key + , lruList = {} // list of items in order of use recency + , lru = 0 // least recently used + , mru = 0 // most recently used + , length = 0 // number of items in the list + + // resize the cache when the maxLength changes. + Object.defineProperty(this, "maxLength", + { set : function (mL) { + if (!mL || !(typeof mL === "number") || mL <= 0 ) mL = Infinity + maxLength = mL + // if it gets above double maxLength, trim right away. + // otherwise, do it whenever it's convenient. + if (length > maxLength) trim() + } + , get : function () { return maxLength } + , enumerable : true + }) + this.maxLength = maxLength + Object.defineProperty(this, "length", + { get : function () { return length } + , enumerable : true + }) + + this.set = function (key, value) { + if (hOP(cache, key)) { + this.get(key) + cache[key].value = value + return undefined + } + var hit = {key:key, value:value, lu:mru++} + lruList[hit.lu] = cache[key] = hit + length ++ + if (length > maxLength) trim() + } + this.get = function (key) { + if (!hOP(cache, key)) return undefined + var hit = cache[key] + delete lruList[hit.lu] + if (hit.lu === lru) lruWalk() + hit.lu = mru ++ + lruList[hit.lu] = hit + return hit.value + } + this.del = function (key) { + if (!hOP(cache, key)) return undefined + var hit = cache[key] + delete cache[key] + delete lruList[hit.lu] + if (hit.lu === lru) lruWalk() + length -- + } + function lruWalk () { + // lru has been deleted, hop up to the next hit. + lru = Object.keys(lruList).shift() + } + function trim () { + if (length <= maxLength) return undefined + var prune = Object.keys(lruList).slice(0, length - maxLength) + for (var i = 0, l = (length - maxLength); i < l; i ++) { + delete cache[ lruList[prune[i]].key ] + delete lruList[prune[i]] + } + length = maxLength + lruWalk() + } +} + +if (!process || !module || module !== process.mainModule) return undefined + +var l = LRUCache(3) + , assert = require("assert") + +l.set(1, 1) +l.set(2, 1) +l.set(3, 1) +l.set(4, 1) +l.set(5, 1) +l.set(6, 1) + +assert.equal(l.get(1), undefined) +assert.equal(l.get(2), undefined) +assert.equal(l.get(3), undefined) +assert.equal(l.get(4), 1) +assert.equal(l.get(5), 1) +assert.equal(l.get(6), 1) + +// now keep re-getting the 6 so it remains the most recently used. +// in this case, we'll have 6, 7, 8, 9, 10, 11, so the ending length = 5 +l.set(7, 1) +l.get(6) +l.set(8, 1) +l.get(6) +l.set(9, 1) +l.get(6) +l.set(10, 1) +l.get(6) +l.set(11, 1) +assert.equal(l.length, 3) +assert.equal(l.get(4), undefined) +assert.equal(l.get(5), undefined) +assert.equal(l.get(6), 1) +assert.equal(l.get(7), undefined) +assert.equal(l.get(8), undefined) +assert.equal(l.get(9), undefined) +assert.equal(l.get(10), 1) +assert.equal(l.get(11), 1) + +// test changing the maxLength, verify that the LRU items get dropped. +l.maxLength = 100 +for (var i = 0; i < 100; i ++) l.set(i, i) +assert.equal(l.length, 100) +for (var i = 0; i < 100; i ++) { + assert.equal(l.get(i), i) +} +l.maxLength = 3 +assert.equal(l.length, 3) +for (var i = 0; i < 97; i ++) { + assert.equal(l.get(i), undefined) +} +for (var i = 98; i < 100; i ++) { + assert.equal(l.get(i), i) +} + +// now remove the maxLength restriction, and try again. +l.maxLength = "hello" +for (var i = 0; i < 100; i ++) l.set(i, i) +assert.equal(l.length, 100) +for (var i = 0; i < 100; i ++) { + assert.equal(l.get(i), i) +} +// should trigger an immediate resize +l.maxLength = 3 +assert.equal(l.length, 3) +for (var i = 0; i < 97; i ++) { + assert.equal(l.get(i), undefined) +} +for (var i = 98; i < 100; i ++) { + assert.equal(l.get(i), i) +} diff --git a/deps/npm/node_modules/minimatch/node_modules/lru-cache/package.json b/deps/npm/node_modules/minimatch/node_modules/lru-cache/package.json new file mode 100644 index 0000000000..289f839d31 --- /dev/null +++ b/deps/npm/node_modules/minimatch/node_modules/lru-cache/package.json @@ -0,0 +1,13 @@ +{ "name" : "lru-cache" +, "description" : "A cache object that deletes the least-recently-used items." +, "version" : "1.0.4" +, "author" : "Isaac Z. Schlueter " +, "scripts" : { "test" : "node lib/lru-cache.js" } +, "main" : "lib/lru-cache" +, "repository" : "git://github.com/isaacs/node-lru-cache.git" +, "licenses" : + [ { "type" : "MIT" + , "url" : "http://github.com/isaacs/node-lru-cache/raw/master/LICENSE" + } + ] +} diff --git a/deps/npm/node_modules/minimatch/package.json b/deps/npm/node_modules/minimatch/package.json new file mode 100644 index 0000000000..b83bebb91b --- /dev/null +++ b/deps/npm/node_modules/minimatch/package.json @@ -0,0 +1,32 @@ +{ + "author": "Isaac Z. Schlueter (http://blog.izs.me)", + "name": "minimatch", + "description": "a glob matcher in javascript", + "version": "0.0.4", + "repository": { + "type": "git", + "url": "git://github.com/isaacs/minimatch.git" + }, + "main": "minimatch.js", + "scripts": { + "test": "tap test" + }, + "engines": { + "node": "*" + }, + "bundleDependencies": [ + "lru-cache" + ], + "dependencies": { + "lru-cache": "~1.0.2" + }, + "devDependencies": { + "tap": "~0.0.5" + }, + "licenses" : [ + { + "type" : "MIT", + "url" : "http://github.com/isaacs/minimatch/raw/master/LICENSE" + } + ] +} diff --git a/deps/npm/node_modules/minimatch/test/basic.js b/deps/npm/node_modules/minimatch/test/basic.js new file mode 100644 index 0000000000..e4c5437e1c --- /dev/null +++ b/deps/npm/node_modules/minimatch/test/basic.js @@ -0,0 +1,141 @@ +// http://www.bashcookbook.com/bashinfo/source/bash-1.14.7/tests/glob-test + +var tap = require("tap") + , mm = require("../") + , files = [ "a", "b", "c", "d", "abc" + , "abd", "abe", "bb", "bcd" + , "ca", "cb", "dd", "de" + , "bdir/", "bdir/cfile"] + , next = files.concat([ "a-b", "aXb" + , ".x", ".y" ]) + +tap.test("basic tests", function (t) { + // [ pattern, [matches], MM opts, files, TAP opts] + ; [ "http://www.bashcookbook.com/bashinfo" + + "/source/bash-1.14.7/tests/glob-test" + , ["a*", ["a", "abc", "abd", "abe"]] + , ["X*", ["X*"]] + // allow null glob expansion + , ["X*", [], { null: true }] + + // isaacs: Slightly different than bash/sh/ksh + // \\* is not un-escaped to literal "*" in a failed match, + // but it does make it get treated as a literal star + , ["\\*", ["\\*"]] + , ["\\**", ["\\**"]] + + , ["b*/", ["bdir/"]] + , ["c*", ["c", "ca", "cb"]] + , ["**", files] + + + , ["\\.\\./*/", ["\\.\\./*/"]] + , ["s/\\..*//", ["s/\\..*//"]] + + // legendary larry crashes bashes + , ["/^root:/{s/^[^:]*:[^:]*:\([^:]*\).*$/\\1/" + , ["/^root:/{s/^[^:]*:[^:]*:\([^:]*\).*$/\\1/"]] + , ["/^root:/{s/^[^:]*:[^:]*:\([^:]*\).*$/\1/" + , ["/^root:/{s/^[^:]*:[^:]*:\([^:]*\).*$/\1/"]] + + // character classes + , ["[a-c]b*", ["abc", "abd", "abe", "bb", "cb"]] + , ["[a-y]*[^c]", ["abd", "abe", "bb", "bcd", + "bdir/", "ca", "cb", "dd", "de"]] + , ["a*[^c]", ["abd", "abe"]] + , function () { files.push("a-b", "aXb") } + , ["a[X-]b", ["a-b", "aXb"]] + , function () { files.push(".x", ".y") } + , ["[^a-c]*", ["d", "dd", "de"]] + , function () { files.push("a*b/", "a*b/ooo") } + , ["a\\*b/*", ["a*b/ooo"]] + , ["a\\*?/*", ["a*b/ooo"]] + , ["*\\\\!*", [], {null: true}, ["echo !7"]] + , ["*\\!*", ["echo !7"], null, ["echo !7"]] + , ["*.\\*", ["r.*"], null, ["r.*"]] + , ["a[b]c", ["abc"]] + , ["a[\\b]c", ["abc"]] + , ["a?c", ["abc"]] + , ["a\\*c", [], {null: true}, ["abc"]] + , ["", [""], { null: true }, [""]] + + , "http://www.opensource.apple.com/source/bash/bash-23/" + + "bash/tests/glob-test" + , function () { files.push("man/", "man/man1/", "man/man1/bash.1") } + , ["*/man*/bash.*", ["man/man1/bash.1"]] + , ["man/man1/bash.1", ["man/man1/bash.1"]] + , ["a***c", ["abc"], null, ["abc"]] + , ["a*****?c", ["abc"], null, ["abc"]] + , ["?*****??", ["abc"], null, ["abc"]] + , ["*****??", ["abc"], null, ["abc"]] + , ["?*****?c", ["abc"], null, ["abc"]] + , ["?***?****c", ["abc"], null, ["abc"]] + , ["?***?****?", ["abc"], null, ["abc"]] + , ["?***?****", ["abc"], null, ["abc"]] + , ["*******c", ["abc"], null, ["abc"]] + , ["*******?", ["abc"], null, ["abc"]] + , ["a*cd**?**??k", ["abcdecdhjk"], null, ["abcdecdhjk"]] + , ["a**?**cd**?**??k", ["abcdecdhjk"], null, ["abcdecdhjk"]] + , ["a**?**cd**?**??k***", ["abcdecdhjk"], null, ["abcdecdhjk"]] + , ["a**?**cd**?**??***k", ["abcdecdhjk"], null, ["abcdecdhjk"]] + , ["a**?**cd**?**??***k**", ["abcdecdhjk"], null, ["abcdecdhjk"]] + , ["a****c**?**??*****", ["abcdecdhjk"], null, ["abcdecdhjk"]] + , ["[-abc]", ["-"], null, ["-"]] + , ["[abc-]", ["-"], null, ["-"]] + , ["\\", ["\\"], null, ["\\"]] + , ["[\\\\]", ["\\"], null, ["\\"]] + , ["[[]", ["["], null, ["["]] + , ["[", ["["], null, ["["]] + , ["[*", ["[abc"], null, ["[abc"]] + , "a right bracket shall lose its special meaning and " + + "represent itself in a bracket expression if it occurs " + + "first in the list. -- POSIX.2 2.8.3.2" + , ["[]]", ["]"], null, ["]"]] + , ["[]-]", ["]"], null, ["]"]] + , ["[a-\z]", ["p"], null, ["p"]] + , ["[/\\\\]*", ["/tmp"], null, ["/tmp"]] + , ["??**********?****?", [], { null: true }, ["abc"]] + , ["??**********?****c", [], { null: true }, ["abc"]] + , ["?************c****?****", [], { null: true }, ["abc"]] + , ["*c*?**", [], { null: true }, ["abc"]] + , ["a*****c*?**", [], { null: true }, ["abc"]] + , ["a********???*******", [], { null: true }, ["abc"]] + , ["[]", [], { null: true }, ["a"]] + , ["[abc", [], { null: true }, ["["]] + + , "nocase tests" + , ["XYZ", ["xYz"], { nocase: true, null: true } + , ["xYz", "ABC", "IjK"]] + , ["ab*", ["ABC"], { nocase: true, null: true } + , ["xYz", "ABC", "IjK"]] + , ["[ia]?[ck]", ["ABC", "IjK"], { nocase: true, null: true } + , ["xYz", "ABC", "IjK"]] + + ].forEach(function (c) { + if (typeof c === "function") return c() + if (typeof c === "string") return t.comment(c) + + var pattern = c[0] + , expect = c[1].sort(alpha) + , options = c[2] || {} + , f = c[3] || files + , tapOpts = c[4] || {} + + // options.debug = true + var r = mm.makeRe(pattern, options) + tapOpts.re = String(r) || JSON.stringify(r) + tapOpts.files = JSON.stringify(f) + tapOpts.pattern = pattern + + var actual = mm.match(f, pattern, options) + + t.equivalent( actual, expect + , JSON.stringify(pattern) + " " + JSON.stringify(expect) + , c[4] ) + }) + t.end() +}) + +function alpha (a, b) { + return a > b ? 1 : -1 +} diff --git a/deps/npm/node_modules/mkdirp/LICENSE b/deps/npm/node_modules/mkdirp/LICENSE new file mode 100644 index 0000000000..432d1aeb01 --- /dev/null +++ b/deps/npm/node_modules/mkdirp/LICENSE @@ -0,0 +1,21 @@ +Copyright 2010 James Halliday (mail@substack.net) + +This project is free software released under the MIT/X11 license: + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/deps/npm/node_modules/mkdirp/README.markdown b/deps/npm/node_modules/mkdirp/README.markdown new file mode 100644 index 0000000000..0393c4ea53 --- /dev/null +++ b/deps/npm/node_modules/mkdirp/README.markdown @@ -0,0 +1,21 @@ +mkdirp +====== + +Like `mkdir -p`, but in node.js! + +Example +======= + +pow.js +------ + var mkdirp = require('mkdirp'); + + mkdirp('/tmp/foo/bar/baz', 0755, function (err) { + if (err) console.error(err) + else console.log('pow!') + }); + +Output + pow! + +And now /tmp/foo/bar/baz exists, huzzah! diff --git a/deps/npm/node_modules/mkdirp/examples/pow.js b/deps/npm/node_modules/mkdirp/examples/pow.js new file mode 100644 index 0000000000..7741462212 --- /dev/null +++ b/deps/npm/node_modules/mkdirp/examples/pow.js @@ -0,0 +1,6 @@ +var mkdirp = require('mkdirp'); + +mkdirp('/tmp/foo/bar/baz', 0755, function (err) { + if (err) console.error(err) + else console.log('pow!') +}); diff --git a/deps/npm/node_modules/mkdirp/index.js b/deps/npm/node_modules/mkdirp/index.js new file mode 100644 index 0000000000..6602801463 --- /dev/null +++ b/deps/npm/node_modules/mkdirp/index.js @@ -0,0 +1,36 @@ +var path = require('path'); +var fs = require('fs'); + +module.exports = mkdirP.mkdirp = mkdirP.mkdirP = mkdirP; + +function mkdirP (p, mode, f) { + var cb = f || function () {}; + if (typeof mode === 'string') mode = parseInt(mode, 8); + p = path.resolve(p); + + fs.mkdir(p, mode, function (er) { + if (!er) return cb(); + switch (er.code) { + case 'ENOENT': + mkdirP(path.dirname(p), mode, function (er) { + if (er) cb(er); + else mkdirP(p, mode, cb); + }); + break; + + case 'EEXIST': + fs.stat(p, function (er2, stat) { + // if the stat fails, then that's super weird. + // let the original EEXIST be the failure reason. + if (er2 || !stat.isDirectory()) cb(er) + else if ((stat.mode & 0777) !== mode) fs.chmod(p, mode, cb); + else cb(); + }); + break; + + default: + cb(er); + break; + } + }); +} diff --git a/deps/npm/node_modules/mkdirp/package.json b/deps/npm/node_modules/mkdirp/package.json new file mode 100644 index 0000000000..99149f747d --- /dev/null +++ b/deps/npm/node_modules/mkdirp/package.json @@ -0,0 +1,23 @@ +{ + "name" : "mkdirp", + "description" : "Recursively mkdir, like `mkdir -p`", + "version" : "0.1.0", + "author" : "James Halliday (http://substack.net)", + "main" : "./index", + "keywords" : [ + "mkdir", + "directory" + ], + "repository" : { + "type" : "git", + "url" : "http://github.com/substack/node-mkdirp.git" + }, + "scripts" : { + "test" : "tap test/*.js" + }, + "devDependencies" : { + "tap" : "0.0.x" + }, + "license" : "MIT/X11", + "engines": { "node": "*" } +} diff --git a/deps/npm/node_modules/mkdirp/test/chmod.js b/deps/npm/node_modules/mkdirp/test/chmod.js new file mode 100644 index 0000000000..0609694e38 --- /dev/null +++ b/deps/npm/node_modules/mkdirp/test/chmod.js @@ -0,0 +1,39 @@ +var mkdirp = require('../').mkdirp; +var path = require('path'); +var fs = require('fs'); +var test = require('tap').test; + +var ps = [ '', 'tmp' ]; + +for (var i = 0; i < 25; i++) { + var dir = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + ps.push(dir); +} + +var file = ps.join('/'); + +test('chmod-pre', function (t) { + var mode = 0744 + mkdirp(file, mode, function (er) { + t.ifError(er, 'should not error'); + fs.stat(file, function (er, stat) { + t.ifError(er, 'should exist'); + t.ok(stat && stat.isDirectory(), 'should be directory'); + t.equal(stat && stat.mode & 0777, mode, 'should be 0744'); + t.end(); + }); + }); +}); + +test('chmod', function (t) { + var mode = 0755 + mkdirp(file, mode, function (er) { + t.ifError(er, 'should not error'); + fs.stat(file, function (er, stat) { + t.ifError(er, 'should exist'); + t.ok(stat && stat.isDirectory(), 'should be directory'); + t.equal(stat && stat.mode & 0777, mode, 'should be 0755'); + t.end(); + }); + }); +}); diff --git a/deps/npm/node_modules/mkdirp/test/clobber.js b/deps/npm/node_modules/mkdirp/test/clobber.js new file mode 100644 index 0000000000..0eb7099870 --- /dev/null +++ b/deps/npm/node_modules/mkdirp/test/clobber.js @@ -0,0 +1,37 @@ +var mkdirp = require('../').mkdirp; +var path = require('path'); +var fs = require('fs'); +var test = require('tap').test; + +var ps = [ '', 'tmp' ]; + +for (var i = 0; i < 25; i++) { + var dir = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + ps.push(dir); +} + +var file = ps.join('/'); + +// a file in the way +var itw = ps.slice(0, 3).join('/'); + + +test('clobber-pre', function (t) { + console.error("about to write to "+itw) + fs.writeFileSync(itw, 'I AM IN THE WAY, THE TRUTH, AND THE LIGHT.'); + + fs.stat(itw, function (er, stat) { + t.ifError(er) + t.ok(stat && stat.isFile(), 'should be file') + t.end() + }) +}) + +test('clobber', function (t) { + t.plan(2); + mkdirp(file, 0755, function (err) { + t.ok(err); + t.equal(err.code, 'ENOTDIR'); + t.end(); + }); +}); diff --git a/deps/npm/node_modules/mkdirp/test/mkdirp.js b/deps/npm/node_modules/mkdirp/test/mkdirp.js new file mode 100644 index 0000000000..b07cd70c10 --- /dev/null +++ b/deps/npm/node_modules/mkdirp/test/mkdirp.js @@ -0,0 +1,28 @@ +var mkdirp = require('../'); +var path = require('path'); +var fs = require('fs'); +var test = require('tap').test; + +test('woo', function (t) { + t.plan(2); + var x = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + var y = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + var z = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + + var file = '/tmp/' + [x,y,z].join('/'); + + mkdirp(file, 0755, function (err) { + if (err) t.fail(err); + else path.exists(file, function (ex) { + if (!ex) t.fail('file not created') + else fs.stat(file, function (err, stat) { + if (err) t.fail(err) + else { + t.equal(stat.mode & 0777, 0755); + t.ok(stat.isDirectory(), 'target not a directory'); + t.end(); + } + }) + }) + }); +}); diff --git a/deps/npm/node_modules/mkdirp/test/race.js b/deps/npm/node_modules/mkdirp/test/race.js new file mode 100644 index 0000000000..96a0447636 --- /dev/null +++ b/deps/npm/node_modules/mkdirp/test/race.js @@ -0,0 +1,41 @@ +var mkdirp = require('../').mkdirp; +var path = require('path'); +var fs = require('fs'); +var test = require('tap').test; + +test('race', function (t) { + t.plan(4); + var ps = [ '', 'tmp' ]; + + for (var i = 0; i < 25; i++) { + var dir = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + ps.push(dir); + } + var file = ps.join('/'); + + var res = 2; + mk(file, function () { + if (--res === 0) t.end(); + }); + + mk(file, function () { + if (--res === 0) t.end(); + }); + + function mk (file, cb) { + mkdirp(file, 0755, function (err) { + if (err) t.fail(err); + else path.exists(file, function (ex) { + if (!ex) t.fail('file not created') + else fs.stat(file, function (err, stat) { + if (err) t.fail(err) + else { + t.equal(stat.mode & 0777, 0755); + t.ok(stat.isDirectory(), 'target not a directory'); + if (cb) cb(); + } + }) + }) + }); + } +}); diff --git a/deps/npm/node_modules/mkdirp/test/rel.js b/deps/npm/node_modules/mkdirp/test/rel.js new file mode 100644 index 0000000000..79858243ab --- /dev/null +++ b/deps/npm/node_modules/mkdirp/test/rel.js @@ -0,0 +1,32 @@ +var mkdirp = require('../'); +var path = require('path'); +var fs = require('fs'); +var test = require('tap').test; + +test('rel', function (t) { + t.plan(2); + var x = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + var y = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + var z = Math.floor(Math.random() * Math.pow(16,4)).toString(16); + + var cwd = process.cwd(); + process.chdir('/tmp'); + + var file = [x,y,z].join('/'); + + mkdirp(file, 0755, function (err) { + if (err) t.fail(err); + else path.exists(file, function (ex) { + if (!ex) t.fail('file not created') + else fs.stat(file, function (err, stat) { + if (err) t.fail(err) + else { + process.chdir(cwd); + t.equal(stat.mode & 0777, 0755); + t.ok(stat.isDirectory(), 'target not a directory'); + t.end(); + } + }) + }) + }); +}); diff --git a/deps/npm/node_modules/node-uuid/LICENSE.md b/deps/npm/node_modules/node-uuid/LICENSE.md new file mode 100644 index 0000000000..bcdddf9a05 --- /dev/null +++ b/deps/npm/node_modules/node-uuid/LICENSE.md @@ -0,0 +1,3 @@ +Copyright (c) 2010 Robert Kieffer + +Dual licensed under the [MIT](http://en.wikipedia.org/wiki/MIT_License) and [GPL](http://en.wikipedia.org/wiki/GNU_General_Public_License) licenses. diff --git a/deps/npm/node_modules/node-uuid/README.md b/deps/npm/node_modules/node-uuid/README.md new file mode 100644 index 0000000000..c00675066f --- /dev/null +++ b/deps/npm/node_modules/node-uuid/README.md @@ -0,0 +1,100 @@ +# node-uuid + +Simple, fast generation of RFC4122[RFC4122(v4)](http://www.ietf.org/rfc/rfc4122.txt) UUIDS. It runs in node.js and all major browsers. + +## Installation + + npm install node-uuid + +### In browser + + + +### In node.js + + var uuid = require('node-uuid'); + +## Usage + +### Generate a String UUID + + var id = uuid(); // -> '92329D39-6F5C-4520-ABFC-AAB64544E172' + +### Generate a Binary UUID + + // Simple form - allocates a Buffer/Array for you + var buf = uuid('binary'); + // node.js -> + // browser -> [8, 80, 5, 200, 156, 178, 76, 7, 172, 7, 209, 79, 185, 245, 4, 81] + + // Provide your own Buffer or Array + var buf = new Array(16); + uuid('binary', buf); // -> [8, 80, 5, 200, 156, 178, 76, 7, 172, 7, 209, 79, 185, 245, 4, 81] + var buf = new Buffer(16); + uuid('binary', buf); // -> + + // Provide your own Buffer/Array, plus specify offset + // (e.g. here we fill an array with 3 uuids) + var buf = new Buffer(16 \* 3); + uuid('binary', id, 0); + uuid('binary', id, 16); + uuid('binary', id, 32); + +## Testing + +test/test.js generates performance data (similar to test/benchmark.js). It also verifies the syntax of 100K string UUIDs, and logs the distribution of hex digits found therein. For example: + + - - - Performance Data - - - + uuid(): 1052631 uuids/second + uuid('binary'): 680272 uuids/second + uuid('binary', buffer): 2702702 uuids/second + + - - - Distribution of Hex Digits (% deviation from ideal) - - - + 0 |================================| 187705 (0.11%) + 1 |================================| 187880 (0.2%) + 2 |================================| 186875 (-0.33%) + 3 |================================| 186847 (-0.35%) + 4 |==================================================| 287433 (-0.02%) + 5 |================================| 187910 (0.22%) + 6 |================================| 188172 (0.36%) + 7 |================================| 187350 (-0.08%) + 8 |====================================| 211994 (-0.24%) + 9 |====================================| 212664 (0.08%) + A |=====================================| 213185 (0.32%) + B |=====================================| 212877 (0.18%) + C |================================| 187445 (-0.03%) + D |================================| 186737 (-0.41%) + E |================================| 187155 (-0.18%) + F |================================| 187771 (0.14%) + +Note that the increased values for 4 and 8-B are expected as part of the RFC4122 syntax (and are accounted for in the deviation calculation). BTW, if someone wants to do the calculation to determine what a statistically significant deviation would be, I'll gladly add that to the test. + +### In browser + + Open test/test.html + +### In node.js + + > node test/test.js + +node.js users can also run the node-uuid .vs. uuid.js benchmark: + + > node test/benchmark.js + +## Performance + +### In node.js + +node-uuid is designed to be fast. That said, the target platform is node.js, where it is screaming fast. Here's what I get on my 2.66GHz Macbook Pro for the test/benchmark.js script: + + nodeuuid(): 1126126 uuids/second + nodeuuid('binary'): 782472 uuids/second + nodeuuid('binary', buffer): 2688172 uuids/second + uuidjs(): 620347 uuids/second + uuidjs('binary'): 1275510 uuids/second + +The uuidjs() entries are for Nikhil Marathe's [uuidjs module](https://bitbucket.org/nikhilm/uuidjs), and are provided for comparison. uuidjs is a wrapper around the native libuuid library. + +### In browser + +node-uuid performance varies dramatically across browsers. For comprehensive test results, please [checkout the JSPerf tests](http://jsperf.com/node-uuid-performance). diff --git a/deps/npm/node_modules/node-uuid/package.json b/deps/npm/node_modules/node-uuid/package.json new file mode 100644 index 0000000000..bf70062abf --- /dev/null +++ b/deps/npm/node_modules/node-uuid/package.json @@ -0,0 +1,12 @@ +{ + "name" : "node-uuid", + "description" : "Simple, fast generation of RFC4122(v4) UUIDs.", + "url" : "http://github.com/broofa/node-uuid", + "keywords" : ["uuid", "guid", "rfc4122"], + "author" : "Robert Kieffer ", + "contributors" : [], + "dependencies" : [], + "lib" : ".", + "main" : "./uuid.js", + "version" : "1.2.0" +} diff --git a/deps/npm/node_modules/node-uuid/test/benchmark-native.c b/deps/npm/node_modules/node-uuid/test/benchmark-native.c new file mode 100644 index 0000000000..dbfc75f6d7 --- /dev/null +++ b/deps/npm/node_modules/node-uuid/test/benchmark-native.c @@ -0,0 +1,34 @@ +/* +Test performance of native C UUID generation + +To Compile: cc -luuid benchmark-native.c -o benchmark-native +*/ + +#include +#include +#include +#include + +int main() { + uuid_t myid; + char buf[36+1]; + int i; + struct timeval t; + double start, finish; + + gettimeofday(&t, NULL); + start = t.tv_sec + t.tv_usec/1e6; + + int n = 2e5; + for (i = 0; i < n; i++) { + uuid_generate(myid); + uuid_unparse(myid, buf); + } + + gettimeofday(&t, NULL); + finish = t.tv_sec + t.tv_usec/1e6; + double dur = finish - start; + + printf("%d uuids/sec", (int)(n/dur)); + return 0; +} diff --git a/deps/npm/node_modules/node-uuid/test/benchmark.js b/deps/npm/node_modules/node-uuid/test/benchmark.js new file mode 100644 index 0000000000..2505dc4a21 --- /dev/null +++ b/deps/npm/node_modules/node-uuid/test/benchmark.js @@ -0,0 +1,27 @@ +var nodeuuid = require('../uuid'), + uuidjs = require('uuid').generate, + N = 5e5; + +function rate(msg, t) { + console.log(msg + ': ' + + (N / (Date.now() - t) * 1e3 | 0) + + ' uuids/second'); +} + +// node-uuid - string form +for (var i = 0, t = Date.now(); i < N; i++) nodeuuid(); +rate('nodeuuid()', t); + +for (var i = 0, t = Date.now(); i < N; i++) nodeuuid('binary'); +rate('nodeuuid(\'binary\')', t); + +var buffer = new nodeuuid.BufferClass(16); +for (var i = 0, t = Date.now(); i < N; i++) nodeuuid('binary', buffer); +rate('nodeuuid(\'binary\', buffer)', t); + +// node-uuid - string form +for (var i = 0, t = Date.now(); i < N; i++) uuidjs(); +rate('uuidjs()', t); + +for (var i = 0, t = Date.now(); i < N; i++) uuidjs('binary'); +rate('uuidjs(\'binary\')', t); diff --git a/deps/npm/node_modules/node-uuid/test/test.html b/deps/npm/node_modules/node-uuid/test/test.html new file mode 100644 index 0000000000..89e0f2c44b --- /dev/null +++ b/deps/npm/node_modules/node-uuid/test/test.html @@ -0,0 +1,14 @@ + + + + + + + + + diff --git a/deps/npm/node_modules/node-uuid/test/test.js b/deps/npm/node_modules/node-uuid/test/test.js new file mode 100644 index 0000000000..5037566eac --- /dev/null +++ b/deps/npm/node_modules/node-uuid/test/test.js @@ -0,0 +1,83 @@ +if (typeof(uuid) == 'undefined') { + uuid = require('../uuid'); +} + +var UUID_FORMAT = /[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-4[0-9a-fA-F]{3}-[89a-fAB][0-9a-fA-F]{3}-[0-9a-fA-F]{12}/; +var N = 1e5; + +function log(msg) { + if (typeof(document) != 'undefined') { + document.write('
' + msg + '
'); + } + if (typeof(console) != 'undefined') { + console.log(msg); + } +} + +function rate(msg, t) { + log(msg + ': ' + (N / (Date.now() - t) * 1e3 | 0) + ' uuids/second'); +} + +// Perf tests +log('- - - Performance Data - - -'); +for (var i = 0, t = Date.now(); i < N; i++) uuid(); +rate('uuid()', t); +for (var i = 0, t = Date.now(); i < N; i++) uuid('binary'); +rate('uuid(\'binary\')', t); +var buf = new uuid.BufferClass(16); +for (var i = 0, t = Date.now(); i < N; i++) uuid('binary', buf); +rate('uuid(\'binary\', buffer)', t); + +var counts = {}, max = 0; + +var b = new uuid.BufferClass(16); +for (var i = 0; i < N; i++) { + id = uuid(); + if (!UUID_FORMAT.test(id)) { + throw Error(id + ' is not a valid UUID string'); + } + + if (id != uuid.unparse(uuid.parse(id))) { + throw Error(id + ' does not parse/unparse'); + } + + // Count digits for our randomness check + var digits = id.replace(/-/g, '').split(''); + for (var j = digits.length-1; j >= 0; j--) { + var c = digits[j]; + max = Math.max(max, counts[c] = (counts[c] || 0) + 1); + } +} + +// Get %'age an actual value differs from the ideal value +function divergence(actual, ideal) { + return Math.round(100*100*(actual - ideal)/ideal)/100; +} + +log('
- - - Distribution of Hex Digits (% deviation from ideal) - - -'); + +// Check randomness +for (var i = 0; i < 16; i++) { + var c = i.toString(16); + var bar = '', n = counts[c], p = Math.round(n/max*100|0); + + // 1-3,5-8, and D-F: 1:16 odds over 30 digits + var ideal = N*30/16; + if (i == 4) { + // 4: 1:1 odds on 1 digit, plus 1:16 odds on 30 digits + ideal = N*(1 + 30/16); + } else if (i >= 8 && i <= 11) { + // 8-B: 1:4 odds on 1 digit, plus 1:16 odds on 30 digits + ideal = N*(1/4 + 30/16); + } else { + // Otherwise: 1:16 odds on 30 digits + ideal = N*30/16; + } + var d = divergence(n, ideal); + + // Draw bar using UTF squares (just for grins) + var s = n/max*50 | 0; + while (s--) bar += '='; + + log(c + ' |' + bar + '| ' + counts[c] + ' (' + d + '%)'); +} diff --git a/deps/npm/node_modules/node-uuid/uuid.js b/deps/npm/node_modules/node-uuid/uuid.js new file mode 100644 index 0000000000..fdf6c54fd4 --- /dev/null +++ b/deps/npm/node_modules/node-uuid/uuid.js @@ -0,0 +1,80 @@ +(function() { + /* + * Generate a RFC4122(v4) UUID + * + * Documentation at https://github.com/broofa/node-uuid + */ + + // Use node.js Buffer class if available, otherwise use the Array class + var BufferClass = typeof(Buffer) == 'function' ? Buffer : Array; + + // Buffer used for generating string uuids + var _buf = new BufferClass(16); + + // Cache number <-> hex string for octet values + var toString = []; + var toNumber = {}; + for (var i = 0; i < 256; i++) { + toString[i] = (i + 0x100).toString(16).substr(1); + toNumber[toString[i]] = i; + } + + function parse(s) { + var buf = new BufferClass(16); + var i = 0, ton = toNumber; + s.toLowerCase().replace(/[0-9a-f][0-9a-f]/g, function(octet) { + buf[i++] = toNumber[octet]; + }); + return buf; + } + + function unparse(buf) { + var tos = toString, b = buf; + return tos[b[0]] + tos[b[1]] + tos[b[2]] + tos[b[3]] + '-' + + tos[b[4]] + tos[b[5]] + '-' + + tos[b[6]] + tos[b[7]] + '-' + + tos[b[8]] + tos[b[9]] + '-' + + tos[b[10]] + tos[b[11]] + tos[b[12]] + + tos[b[13]] + tos[b[14]] + tos[b[15]]; + } + + var b32 = 0x100000000, ff = 0xff; + function uuid(fmt, buf, offset) { + var b = fmt != 'binary' ? _buf : (buf ? buf : new BufferClass(16)); + var i = buf && offset || 0; + + var r = Math.random()*b32; + b[i++] = r & ff; + b[i++] = r>>>8 & ff; + b[i++] = r>>>16 & ff; + b[i++] = r>>>24 & ff; + r = Math.random()*b32; + b[i++] = r & ff; + b[i++] = r>>>8 & ff; + b[i++] = r>>>16 & 0x0f | 0x40; // See RFC4122 sect. 4.1.3 + b[i++] = r>>>24 & ff; + r = Math.random()*b32; + b[i++] = r & 0x3f | 0x80; // See RFC4122 sect. 4.4 + b[i++] = r>>>8 & ff; + b[i++] = r>>>16 & ff; + b[i++] = r>>>24 & ff; + r = Math.random()*b32; + b[i++] = r & ff; + b[i++] = r>>>8 & ff; + b[i++] = r>>>16 & ff; + b[i++] = r>>>24 & ff; + + return fmt === undefined ? unparse(b) : b; + }; + + uuid.parse = parse; + uuid.unparse = unparse; + uuid.BufferClass = BufferClass; + + if (typeof(module) != 'undefined') { + module.exports = uuid; + } else { + // In browser? Set as top-level function + this.uuid = uuid; + } +})(); diff --git a/deps/npm/node_modules/nopt/.gitignore b/deps/npm/node_modules/nopt/.gitignore new file mode 100644 index 0000000000..e69de29bb2 diff --git a/deps/npm/node_modules/nopt/LICENSE b/deps/npm/node_modules/nopt/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/nopt/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/nopt/README.md b/deps/npm/node_modules/nopt/README.md new file mode 100644 index 0000000000..eeddfd4fe1 --- /dev/null +++ b/deps/npm/node_modules/nopt/README.md @@ -0,0 +1,208 @@ +If you want to write an option parser, and have it be good, there are +two ways to do it. The Right Way, and the Wrong Way. + +The Wrong Way is to sit down and write an option parser. We've all done +that. + +The Right Way is to write some complex configurable program with so many +options that you go half-insane just trying to manage them all, and put +it off with duct-tape solutions until you see exactly to the core of the +problem, and finally snap and write an awesome option parser. + +If you want to write an option parser, don't write an option parser. +Write a package manager, or a source control system, or a service +restarter, or an operating system. You probably won't end up with a +good one of those, but if you don't give up, and you are relentless and +diligent enough in your procrastination, you may just end up with a very +nice option parser. + +## USAGE + + // my-program.js + var nopt = require("nopt") + , Stream = require("stream").Stream + , path = require("path") + , knownOpts = { "foo" : [String, null] + , "bar" : [Stream, Number] + , "baz" : path + , "bloo" : [ "big", "medium", "small" ] + , "flag" : Boolean + , "pick" : Boolean + , "many" : [String, Array] + } + , shortHands = { "foofoo" : ["--foo", "Mr. Foo"] + , "b7" : ["--bar", "7"] + , "m" : ["--bloo", "medium"] + , "p" : ["--pick"] + , "f" : ["--flag"] + } + // everything is optional. + // knownOpts and shorthands default to {} + // arg list defaults to process.argv + // slice defaults to 2 + , parsed = nopt(knownOpts, shortHands, process.argv, 2) + console.log(parsed) + +This would give you support for any of the following: + +```bash +$ node my-program.js --foo "blerp" --no-flag +{ "foo" : "blerp", "flag" : false } + +$ node my-program.js ---bar 7 --foo "Mr. Hand" --flag +{ bar: 7, foo: "Mr. Hand", flag: true } + +$ node my-program.js --foo "blerp" -f -----p +{ foo: "blerp", flag: true, pick: true } + +$ node my-program.js -fp --foofoo +{ foo: "Mr. Foo", flag: true, pick: true } + +$ node my-program.js --foofoo -- -fp # -- stops the flag parsing. +{ foo: "Mr. Foo", argv: { remain: ["-fp"] } } + +$ node my-program.js --blatzk 1000 -fp # unknown opts are ok. +{ blatzk: 1000, flag: true, pick: true } + +$ node my-program.js --blatzk true -fp # but they need a value +{ blatzk: true, flag: true, pick: true } + +$ node my-program.js --no-blatzk -fp # unless they start with "no-" +{ blatzk: false, flag: true, pick: true } + +$ node my-program.js --baz b/a/z # known paths are resolved. +{ baz: "/Users/isaacs/b/a/z" } + +# if Array is one of the types, then it can take many +# values, and will always be an array. The other types provided +# specify what types are allowed in the list. + +$ node my-program.js --many 1 --many null --many foo +{ many: ["1", "null", "foo"] } + +$ node my-program.js --many foo +{ many: ["foo"] } +``` + +Read the tests at the bottom of `lib/nopt.js` for more examples of +what this puppy can do. + +## Types + +The following types are supported, and defined on `nopt.typeDefs` + +* String: A normal string. No parsing is done. +* path: A file system path. Gets resolved against cwd if not absolute. +* url: A url. If it doesn't parse, it isn't accepted. +* Number: Must be numeric. +* Date: Must parse as a date. If it does, and `Date` is one of the options, + then it will return a Date object, not a string. +* Boolean: Must be either `true` or `false`. If an option is a boolean, + then it does not need a value, and its presence will imply `true` as + the value. To negate boolean flags, do `--no-whatever` or `--whatever + false` +* NaN: Means that the option is strictly not allowed. Any value will + fail. +* Stream: An object matching the "Stream" class in node. Valuable + for use when validating programmatically. (npm uses this to let you + supply any WriteStream on the `outfd` and `logfd` config options.) +* Array: If `Array` is specified as one of the types, then the value + will be parsed as a list of options. This means that multiple values + can be specified, and that the value will always be an array. + +If a type is an array of values not on this list, then those are +considered valid values. For instance, in the example above, the +`--bloo` option can only be one of `"big"`, `"medium"`, or `"small"`, +and any other value will be rejected. + +When parsing unknown fields, `"true"`, `"false"`, and `"null"` will be +interpreted as their JavaScript equivalents, and numeric values will be +interpreted as a number. + +You can also mix types and values, or multiple types, in a list. For +instance `{ blah: [Number, null] }` would allow a value to be set to +either a Number or null. + +To define a new type, add it to `nopt.typeDefs`. Each item in that +hash is an object with a `type` member and a `validate` method. The +`type` member is an object that matches what goes in the type list. The +`validate` method is a function that gets called with `validate(data, +key, val)`. Validate methods should assign `data[key]` to the valid +value of `val` if it can be handled properly, or return boolean +`false` if it cannot. + +You can also call `nopt.clean(data, types, typeDefs)` to clean up a +config object and remove its invalid properties. + +## Error Handling + +By default, nopt outputs a warning to standard error when invalid +options are found. You can change this behavior by assigning a method +to `nopt.invalidHandler`. This method will be called with +the offending `nopt.invalidHandler(key, val, types)`. + +If no `nopt.invalidHandler` is assigned, then it will console.error +its whining. If it is assigned to boolean `false` then the warning is +suppressed. + +## Abbreviations + +Yes, they are supported. If you define options like this: + +```javascript +{ "foolhardyelephants" : Boolean +, "pileofmonkeys" : Boolean } +``` + +Then this will work: + +```bash +node program.js --foolhar --pil +node program.js --no-f --pileofmon +# etc. +``` + +## Shorthands + +Shorthands are a hash of shorter option names to a snippet of args that +they expand to. + +If multiple one-character shorthands are all combined, and the +combination does not unambiguously match any other option or shorthand, +then they will be broken up into their constituent parts. For example: + +```json +{ "s" : ["--loglevel", "silent"] +, "g" : "--global" +, "f" : "--force" +, "p" : "--parseable" +, "l" : "--long" +} +``` + +```bash +npm ls -sgflp +# just like doing this: +npm ls --loglevel silent --global --force --long --parseable +``` + +## The Rest of the args + +The config object returned by nopt is given a special member called +`argv`, which is an object with the following fields: + +* `remain`: The remaining args after all the parsing has occurred. +* `original`: The args as they originally appeared. +* `cooked`: The args after flags and shorthands are expanded. + +## Slicing + +Node programs are called with more or less the exact argv as it appears +in C land, after the v8 and node-specific options have been plucked off. +As such, `argv[0]` is always `node` and `argv[1]` is always the +JavaScript program being run. + +That's usually not very useful to you. So they're sliced off by +default. If you want them, then you can pass in `0` as the last +argument, or any other number that you'd like to slice off the start of +the list. diff --git a/deps/npm/node_modules/nopt/bin/nopt.js b/deps/npm/node_modules/nopt/bin/nopt.js new file mode 100755 index 0000000000..df90c729af --- /dev/null +++ b/deps/npm/node_modules/nopt/bin/nopt.js @@ -0,0 +1,44 @@ +#!/usr/bin/env node +var nopt = require("../lib/nopt") + , types = { num: Number + , bool: Boolean + , help: Boolean + , list: Array + , "num-list": [Number, Array] + , "str-list": [String, Array] + , "bool-list": [Boolean, Array] + , str: String } + , shorthands = { s: [ "--str", "astring" ] + , b: [ "--bool" ] + , nb: [ "--no-bool" ] + , tft: [ "--bool-list", "--no-bool-list", "--bool-list", "true" ] + , "?": ["--help"] + , h: ["--help"] + , H: ["--help"] + , n: [ "--num", "125" ] } + , parsed = nopt( types + , shorthands + , process.argv + , 2 ) + +console.log("parsed", parsed) + +if (parsed.help) { + console.log("") + console.log("nopt cli tester") + console.log("") + console.log("types") + console.log(Object.keys(types).map(function M (t) { + var type = types[t] + if (Array.isArray(type)) { + return [t, type.map(function (type) { return type.name })] + } + return [t, type && type.name] + }).reduce(function (s, i) { + s[i[0]] = i[1] + return s + }, {})) + console.log("") + console.log("shorthands") + console.log(shorthands) +} diff --git a/deps/npm/node_modules/nopt/examples/my-program.js b/deps/npm/node_modules/nopt/examples/my-program.js new file mode 100755 index 0000000000..142447e18e --- /dev/null +++ b/deps/npm/node_modules/nopt/examples/my-program.js @@ -0,0 +1,30 @@ +#!/usr/bin/env node + +//process.env.DEBUG_NOPT = 1 + +// my-program.js +var nopt = require("../lib/nopt") + , Stream = require("stream").Stream + , path = require("path") + , knownOpts = { "foo" : [String, null] + , "bar" : [Stream, Number] + , "baz" : path + , "bloo" : [ "big", "medium", "small" ] + , "flag" : Boolean + , "pick" : Boolean + } + , shortHands = { "foofoo" : ["--foo", "Mr. Foo"] + , "b7" : ["--bar", "7"] + , "m" : ["--bloo", "medium"] + , "p" : ["--pick"] + , "f" : ["--flag", "true"] + , "g" : ["--flag"] + , "s" : "--flag" + } + // everything is optional. + // knownOpts and shorthands default to {} + // arg list defaults to process.argv + // slice defaults to 2 + , parsed = nopt(knownOpts, shortHands, process.argv, 2) + +console.log("parsed =\n"+ require("util").inspect(parsed)) diff --git a/deps/npm/node_modules/nopt/lib/nopt.js b/deps/npm/node_modules/nopt/lib/nopt.js new file mode 100644 index 0000000000..ff802dafe3 --- /dev/null +++ b/deps/npm/node_modules/nopt/lib/nopt.js @@ -0,0 +1,552 @@ +// info about each config option. + +var debug = process.env.DEBUG_NOPT || process.env.NOPT_DEBUG + ? function () { console.error.apply(console, arguments) } + : function () {} + +var url = require("url") + , path = require("path") + , Stream = require("stream").Stream + , abbrev = require("abbrev") + +module.exports = exports = nopt +exports.clean = clean + +exports.typeDefs = + { String : { type: String, validate: validateString } + , Boolean : { type: Boolean, validate: validateBoolean } + , url : { type: url, validate: validateUrl } + , Number : { type: Number, validate: validateNumber } + , path : { type: path, validate: validatePath } + , Stream : { type: Stream, validate: validateStream } + , Date : { type: Date, validate: validateDate } + } + +function nopt (types, shorthands, args, slice) { + args = args || process.argv + types = types || {} + shorthands = shorthands || {} + if (typeof slice !== "number") slice = 2 + + debug(types, shorthands, args, slice) + + args = args.slice(slice) + var data = {} + , key + , remain = [] + , cooked = args + , original = args.slice(0) + + parse(args, data, remain, types, shorthands) + // now data is full + clean(data, types, exports.typeDefs) + data.argv = {remain:remain,cooked:cooked,original:original} + data.argv.toString = function () { + return this.original.map(JSON.stringify).join(" ") + } + return data +} + +function clean (data, types, typeDefs) { + typeDefs = typeDefs || exports.typeDefs + var remove = {} + , typeDefault = [false, true, null, String, Number] + + Object.keys(data).forEach(function (k) { + if (k === "argv") return + var val = data[k] + , isArray = Array.isArray(val) + , type = types[k] + if (!isArray) val = [val] + if (!type) type = typeDefault + if (type === Array) type = typeDefault.concat(Array) + if (!Array.isArray(type)) type = [type] + + debug("val=%j", val) + debug("types=", type) + val = val.map(function (val) { + // if it's an unknown value, then parse false/true/null/numbers/dates + if (typeof val === "string") { + debug("string %j", val) + val = val.trim() + if ((val === "null" && ~type.indexOf(null)) + || (val === "true" && + (~type.indexOf(true) || ~type.indexOf(Boolean))) + || (val === "false" && + (~type.indexOf(false) || ~type.indexOf(Boolean)))) { + val = JSON.parse(val) + debug("jsonable %j", val) + } else if (~type.indexOf(Number) && !isNaN(val)) { + debug("convert to number", val) + val = +val + } else if (~type.indexOf(Date) && !isNaN(Date.parse(val))) { + debug("convert to date", val) + val = new Date(val) + } + } + + if (!types.hasOwnProperty(k)) { + return val + } + + // allow `--no-blah` to set 'blah' to null if null is allowed + if (val === false && ~type.indexOf(null) && + !(~type.indexOf(false) || ~type.indexOf(Boolean))) { + val = null + } + + var d = {} + d[k] = val + debug("prevalidated val", d, val, types[k]) + if (!validate(d, k, val, types[k], typeDefs)) { + if (exports.invalidHandler) { + exports.invalidHandler(k, val, types[k], data) + } else if (exports.invalidHandler !== false) { + debug("invalid: "+k+"="+val, types[k]) + } + return remove + } + debug("validated val", d, val, types[k]) + return d[k] + }).filter(function (val) { return val !== remove }) + + if (!val.length) delete data[k] + else if (isArray) { + debug(isArray, data[k], val) + data[k] = val + } else data[k] = val[0] + + debug("k=%s val=%j", k, val, data[k]) + }) +} + +function validateString (data, k, val) { + data[k] = String(val) +} + +function validatePath (data, k, val) { + data[k] = path.resolve(String(val)) + return true +} + +function validateNumber (data, k, val) { + debug("validate Number %j %j %j", k, val, isNaN(val)) + if (isNaN(val)) return false + data[k] = +val +} + +function validateDate (data, k, val) { + debug("validate Date %j %j %j", k, val, Date.parse(val)) + var s = Date.parse(val) + if (isNaN(s)) return false + data[k] = new Date(val) +} + +function validateBoolean (data, k, val) { + if (val instanceof Boolean) val = val.valueOf() + else if (typeof val === "string") { + if (!isNaN(val)) val = !!(+val) + else if (val === "null" || val === "false") val = false + else val = true + } else val = !!val + data[k] = val +} + +function validateUrl (data, k, val) { + val = url.parse(String(val)) + if (!val.host) return false + data[k] = val.href +} + +function validateStream (data, k, val) { + if (!(val instanceof Stream)) return false + data[k] = val +} + +function validate (data, k, val, type, typeDefs) { + // arrays are lists of types. + if (Array.isArray(type)) { + for (var i = 0, l = type.length; i < l; i ++) { + if (type[i] === Array) continue + if (validate(data, k, val, type[i], typeDefs)) return true + } + delete data[k] + return false + } + + // an array of anything? + if (type === Array) return true + + // NaN is poisonous. Means that something is not allowed. + if (type !== type) { + debug("Poison NaN", k, val, type) + delete data[k] + return false + } + + // explicit list of values + if (val === type) { + debug("Explicitly allowed %j", val) + // if (isArray) (data[k] = data[k] || []).push(val) + // else data[k] = val + data[k] = val + return true + } + + // now go through the list of typeDefs, validate against each one. + var ok = false + , types = Object.keys(typeDefs) + for (var i = 0, l = types.length; i < l; i ++) { + debug("test type %j %j %j", k, val, types[i]) + var t = typeDefs[types[i]] + if (t && type === t.type) { + var d = {} + ok = false !== t.validate(d, k, val) + val = d[k] + if (ok) { + // if (isArray) (data[k] = data[k] || []).push(val) + // else data[k] = val + data[k] = val + break + } + } + } + debug("OK? %j (%j %j %j)", ok, k, val, types[i]) + + if (!ok) delete data[k] + return ok +} + +function parse (args, data, remain, types, shorthands) { + debug("parse", args, data, remain) + + var key = null + , abbrevs = abbrev(Object.keys(types)) + , shortAbbr = abbrev(Object.keys(shorthands)) + + for (var i = 0; i < args.length; i ++) { + var arg = args[i] + debug("arg", arg) + + if (arg.match(/^-{2,}$/)) { + // done with keys. + // the rest are args. + remain.push.apply(remain, args.slice(i + 1)) + args[i] = "--" + break + } + if (arg.charAt(0) === "-") { + if (arg.indexOf("=") !== -1) { + var v = arg.split("=") + arg = v.shift() + v = v.join("=") + args.splice.apply(args, [i, 1].concat([arg, v])) + } + // see if it's a shorthand + // if so, splice and back up to re-parse it. + var shRes = resolveShort(arg, shorthands, shortAbbr, abbrevs) + debug("arg=%j shRes=%j", arg, shRes) + if (shRes) { + debug(arg, shRes) + args.splice.apply(args, [i, 1].concat(shRes)) + if (arg !== shRes[0]) { + i -- + continue + } + } + arg = arg.replace(/^-+/, "") + var no = false + while (arg.toLowerCase().indexOf("no-") === 0) { + no = !no + arg = arg.substr(3) + } + + if (abbrevs[arg]) arg = abbrevs[arg] + + var isArray = types[arg] === Array || + Array.isArray(types[arg]) && types[arg].indexOf(Array) !== -1 + + var val + , la = args[i + 1] + + var isBool = no || + types[arg] === Boolean || + Array.isArray(types[arg]) && types[arg].indexOf(Boolean) !== -1 || + (la === "false" && + (types[arg] === null || + Array.isArray(types[arg]) && ~types[arg].indexOf(null))) + + if (isBool) { + // just set and move along + val = !no + // however, also support --bool true or --bool false + if (la === "true" || la === "false") { + val = JSON.parse(la) + la = null + if (no) val = !val + i ++ + } + + // also support "foo":[Boolean, "bar"] and "--foo bar" + if (Array.isArray(types[arg]) && la) { + if (~types[arg].indexOf(la)) { + // an explicit type + val = la + i ++ + } else if ( la === "null" && ~types[arg].indexOf(null) ) { + // null allowed + val = null + i ++ + } else if ( !la.match(/^-{2,}[^-]/) && + !isNaN(la) && + ~types[arg].indexOf(Number) ) { + // number + val = +la + i ++ + } else if ( !la.match(/^-[^-]/) && ~types[arg].indexOf(String) ) { + // string + val = la + i ++ + } + } + + if (isArray) (data[arg] = data[arg] || []).push(val) + else data[arg] = val + + continue + } + + if (la && la.match(/^-{2,}$/)) { + la = undefined + i -- + } + + val = la === undefined ? true : la + if (isArray) (data[arg] = data[arg] || []).push(val) + else data[arg] = val + + i ++ + continue + } + remain.push(arg) + } +} + +function resolveShort (arg, shorthands, shortAbbr, abbrevs) { + // handle single-char shorthands glommed together, like + // npm ls -glp, but only if there is one dash, and only if + // all of the chars are single-char shorthands, and it's + // not a match to some other abbrev. + arg = arg.replace(/^-+/, '') + if (abbrevs[arg] && !shorthands[arg]) { + return null + } + if (shortAbbr[arg]) { + arg = shortAbbr[arg] + } else { + var singles = shorthands.___singles + if (!singles) { + singles = Object.keys(shorthands).filter(function (s) { + return s.length === 1 + }).reduce(function (l,r) { l[r] = true ; return l }, {}) + shorthands.___singles = singles + } + var chrs = arg.split("").filter(function (c) { + return singles[c] + }) + if (chrs.join("") === arg) return chrs.map(function (c) { + return shorthands[c] + }).reduce(function (l, r) { + return l.concat(r) + }, []) + } + + if (shorthands[arg] && !Array.isArray(shorthands[arg])) { + shorthands[arg] = shorthands[arg].split(/\s+/) + } + return shorthands[arg] +} + +if (module === require.main) { +var assert = require("assert") + , util = require("util") + + , shorthands = + { s : ["--loglevel", "silent"] + , d : ["--loglevel", "info"] + , dd : ["--loglevel", "verbose"] + , ddd : ["--loglevel", "silly"] + , noreg : ["--no-registry"] + , reg : ["--registry"] + , "no-reg" : ["--no-registry"] + , silent : ["--loglevel", "silent"] + , verbose : ["--loglevel", "verbose"] + , h : ["--usage"] + , H : ["--usage"] + , "?" : ["--usage"] + , help : ["--usage"] + , v : ["--version"] + , f : ["--force"] + , desc : ["--description"] + , "no-desc" : ["--no-description"] + , "local" : ["--no-global"] + , l : ["--long"] + , p : ["--parseable"] + , porcelain : ["--parseable"] + , g : ["--global"] + } + + , types = + { aoa: Array + , nullstream: [null, Stream] + , date: Date + , str: String + , browser : String + , cache : path + , color : ["always", Boolean] + , depth : Number + , description : Boolean + , dev : Boolean + , editor : path + , force : Boolean + , global : Boolean + , globalconfig : path + , group : [String, Number] + , gzipbin : String + , logfd : [Number, Stream] + , loglevel : ["silent","win","error","warn","info","verbose","silly"] + , long : Boolean + , "node-version" : [false, String] + , npaturl : url + , npat : Boolean + , "onload-script" : [false, String] + , outfd : [Number, Stream] + , parseable : Boolean + , pre: Boolean + , prefix: path + , proxy : url + , "rebuild-bundle" : Boolean + , registry : url + , searchopts : String + , searchexclude: [null, String] + , shell : path + , t: [Array, String] + , tag : String + , tar : String + , tmp : path + , "unsafe-perm" : Boolean + , usage : Boolean + , user : String + , username : String + , userconfig : path + , version : Boolean + , viewer: path + , _exit : Boolean + } + +; [["-v", {version:true}, []] + ,["---v", {version:true}, []] + ,["ls -s --no-reg connect -d", + {loglevel:"info",registry:null},["ls","connect"]] + ,["ls ---s foo",{loglevel:"silent"},["ls","foo"]] + ,["ls --registry blargle", {}, ["ls"]] + ,["--no-registry", {registry:null}, []] + ,["--no-color true", {color:false}, []] + ,["--no-color false", {color:true}, []] + ,["--no-color", {color:false}, []] + ,["--color false", {color:false}, []] + ,["--color --logfd 7", {logfd:7,color:true}, []] + ,["--color=true", {color:true}, []] + ,["--logfd=10", {logfd:10}, []] + ,["--tmp=/tmp -tar=gtar",{tmp:"/tmp",tar:"gtar"},[]] + ,["--tmp=tmp -tar=gtar", + {tmp:path.resolve(process.cwd(), "tmp"),tar:"gtar"},[]] + ,["--logfd x", {}, []] + ,["a -true -- -no-false", {true:true},["a","-no-false"]] + ,["a -no-false", {false:false},["a"]] + ,["a -no-no-true", {true:true}, ["a"]] + ,["a -no-no-no-false", {false:false}, ["a"]] + ,["---NO-no-No-no-no-no-nO-no-no"+ + "-No-no-no-no-no-no-no-no-no"+ + "-no-no-no-no-NO-NO-no-no-no-no-no-no"+ + "-no-body-can-do-the-boogaloo-like-I-do" + ,{"body-can-do-the-boogaloo-like-I-do":false}, []] + ,["we are -no-strangers-to-love "+ + "--you-know the-rules --and so-do-i "+ + "---im-thinking-of=a-full-commitment "+ + "--no-you-would-get-this-from-any-other-guy "+ + "--no-gonna-give-you-up "+ + "-no-gonna-let-you-down=true "+ + "--no-no-gonna-run-around false "+ + "--desert-you=false "+ + "--make-you-cry false "+ + "--no-tell-a-lie "+ + "--no-no-and-hurt-you false" + ,{"strangers-to-love":false + ,"you-know":"the-rules" + ,"and":"so-do-i" + ,"you-would-get-this-from-any-other-guy":false + ,"gonna-give-you-up":false + ,"gonna-let-you-down":false + ,"gonna-run-around":false + ,"desert-you":false + ,"make-you-cry":false + ,"tell-a-lie":false + ,"and-hurt-you":false + },["we", "are"]] + ,["-t one -t two -t three" + ,{t: ["one", "two", "three"]} + ,[]] + ,["-t one -t null -t three four five null" + ,{t: ["one", "null", "three"]} + ,["four", "five", "null"]] + ,["-t foo" + ,{t:["foo"]} + ,[]] + ,["--no-t" + ,{t:["false"]} + ,[]] + ,["-no-no-t" + ,{t:["true"]} + ,[]] + ,["-aoa one -aoa null -aoa 100" + ,{aoa:["one", null, 100]} + ,[]] + ,["-str 100" + ,{str:"100"} + ,[]] + ,["--color always" + ,{color:"always"} + ,[]] + ,["--no-nullstream" + ,{nullstream:null} + ,[]] + ,["--nullstream false" + ,{nullstream:null} + ,[]] + ,["--notadate 2011-01-25" + ,{notadate: "2011-01-25"} + ,[]] + ,["--date 2011-01-25" + ,{date: new Date("2011-01-25")} + ,[]] + ].forEach(function (test) { + var argv = test[0].split(/\s+/) + , opts = test[1] + , rem = test[2] + , actual = nopt(types, shorthands, argv, 0) + , parsed = actual.argv + delete actual.argv + console.log(util.inspect(actual, false, 2, true), parsed.remain) + for (var i in opts) { + var e = JSON.stringify(opts[i]) + , a = JSON.stringify(actual[i] === undefined ? null : actual[i]) + if (e && typeof e === "object") { + assert.deepEqual(e, a) + } else { + assert.equal(e, a) + } + } + assert.deepEqual(rem, parsed.remain) + }) +} diff --git a/deps/npm/node_modules/nopt/package.json b/deps/npm/node_modules/nopt/package.json new file mode 100644 index 0000000000..d1118e3999 --- /dev/null +++ b/deps/npm/node_modules/nopt/package.json @@ -0,0 +1,12 @@ +{ "name" : "nopt" +, "version" : "1.0.10" +, "description" : "Option parsing for Node, supporting types, shorthands, etc. Used by npm." +, "author" : "Isaac Z. Schlueter (http://blog.izs.me/)" +, "main" : "lib/nopt.js" +, "scripts" : { "test" : "node lib/nopt.js" } +, "repository" : "http://github.com/isaacs/nopt" +, "bin" : "./bin/nopt.js" +, "license" : + { "type" : "MIT" + , "url" : "https://github.com/isaacs/nopt/raw/master/LICENSE" } +, "dependencies" : { "abbrev" : "1" }} diff --git a/deps/npm/node_modules/proto-list/LICENSE b/deps/npm/node_modules/proto-list/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/proto-list/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/proto-list/README.md b/deps/npm/node_modules/proto-list/README.md new file mode 100644 index 0000000000..43cfa35893 --- /dev/null +++ b/deps/npm/node_modules/proto-list/README.md @@ -0,0 +1,3 @@ +A list of objects, bound by their prototype chain. + +Used in npm's config stuff. diff --git a/deps/npm/node_modules/proto-list/package.json b/deps/npm/node_modules/proto-list/package.json new file mode 100644 index 0000000000..5cab34befe --- /dev/null +++ b/deps/npm/node_modules/proto-list/package.json @@ -0,0 +1,9 @@ +{ "name" : "proto-list" +, "version" : "1.0.0" +, "description" : "A utility for managing a prototype chain" +, "main" : "./proto-list.js" +, "author" : "Isaac Z. Schlueter (http://blog.izs.me/)" +, "scripts" : { "test" : "node proto-list.js" } +, "repository": { "type": "git", "url": "https://github.com/isaacs/proto-list" } +, "license": { "type": "MIT", "url": "https://github.com/isaacs/proto-list/blob/master/LICENSE" } +, "devDependencies" : { "tap" : "0" } } diff --git a/deps/npm/node_modules/proto-list/proto-list.js b/deps/npm/node_modules/proto-list/proto-list.js new file mode 100644 index 0000000000..759d827382 --- /dev/null +++ b/deps/npm/node_modules/proto-list/proto-list.js @@ -0,0 +1,94 @@ + +module.exports = ProtoList + +function ProtoList () { this.list = [] } +ProtoList.prototype = + { get length () { return this.list.length } + , get keys () { + var k = [] + for (var i in this.list[0]) k.push(i) + return k + } + , get snapshot () { + var o = {} + this.keys.forEach(function (k) { o[k] = this.get(k) }, this) + return o + } + , push : function (obj) { + if (typeof obj !== "object") obj = {valueOf:obj} + if (this.list.length >= 1) { + this.list[this.list.length - 1].__proto__ = obj + } + obj.__proto__ = Object.prototype + return this.list.push(obj) + } + , pop : function () { + if (this.list.length >= 2) { + this.list[this.list.length - 2].__proto__ = Object.prototype + } + return this.list.pop() + } + , unshift : function (obj) { + obj.__proto__ = this.list[0] || Object.prototype + return this.list.unshift(obj) + } + , shift : function () { + if (this.list.length >= 1) { + this.list[0].__proto__ = Object.prototype + } + return this.list.shift() + } + , get : function (key) { + return this.list[0][key] + } + , set : function (key, val, save) { + if (!this.length) this.push({}) + if (save && this.list[0].hasOwnProperty(key)) this.push({}) + return this.list[0][key] = val + } + , forEach : function (fn, thisp) { + for (var key in this.list[0]) fn.call(thisp, key, this.list[0][key]) + } + , slice : function () { + return this.list.slice.apply(this.list, arguments) + } + , splice : function () { + return this.list.splice.apply(this.list, arguments) + } + } + +if (module === require.main) { + +var tap = require("tap") + , test = tap.test + +tap.plan(1) + +tap.test("protoList tests", function (t) { + var p = new ProtoList + p.push({foo:"bar"}) + p.push({}) + p.set("foo", "baz") + t.equal(p.get("foo"), "baz") + + var p = new ProtoList + p.push({foo:"bar"}) + p.set("foo", "baz") + t.equal(p.get("foo"), "baz") + t.equal(p.length, 1) + p.pop() + t.equal(p.length, 0) + p.set("foo", "asdf") + t.equal(p.length, 1) + t.equal(p.get("foo"), "asdf") + p.push({bar:"baz"}) + t.equal(p.length, 2) + t.equal(p.get("foo"), "asdf") + p.shift() + t.equal(p.length, 1) + t.equal(p.get("foo"), undefined) + t.end() +}) + + +} diff --git a/deps/npm/node_modules/request/LICENSE b/deps/npm/node_modules/request/LICENSE new file mode 100644 index 0000000000..a4a9aee0c2 --- /dev/null +++ b/deps/npm/node_modules/request/LICENSE @@ -0,0 +1,55 @@ +Apache License + +Version 2.0, January 2004 + +http://www.apache.org/licenses/ + +TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + +1. Definitions. + +"License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. + +"Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. + +"Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. + +"You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. + +"Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. + +"Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. + +"Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). + +"Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. + +"Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." + +"Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. + +2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. + +3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. + +4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: + +You must give any other recipients of the Work or Derivative Works a copy of this License; and + +You must cause any modified files to carry prominent notices stating that You changed the files; and + +You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and + +If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. + +5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. + +6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. + +7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. + +8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. + +9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. + +END OF TERMS AND CONDITIONS \ No newline at end of file diff --git a/deps/npm/node_modules/request/README.md b/deps/npm/node_modules/request/README.md new file mode 100644 index 0000000000..315bdbaf4a --- /dev/null +++ b/deps/npm/node_modules/request/README.md @@ -0,0 +1,193 @@ +# Request -- Simplified HTTP request method + +## Install + +
+  npm install request
+
+ +Or from source: + +
+  git clone git://github.com/mikeal/request.git 
+  cd request
+  npm link
+
+ +## Super simple to use + +Request is designed to be the simplest way possible to make http calls. It support HTTPS and follows redirects by default. + +```javascript +var request = require('request'); +request('http://www.google.com', function (error, response, body) { + if (!error && response.statusCode == 200) { + console.log(body) // Print the google web page. + } +}) +``` + +## Streaming + +You can stream any response to a file stream. + +```javascript +request('http://google.com/doodle.png').pipe(fs.createWriteStream('doodle.png')) +``` + +You can also stream a file to a PUT or POST request. This method will also check the file extension against a mapping of file extensions to content-types, in this case `application/json`, and use the proper content-type in the PUT request if one is not already provided in the headers. + +```javascript +fs.readStream('file.json').pipe(request.put('http://mysite.com/obj.json')) +``` + +Request can also pipe to itself. When doing so the content-type and content-length will be preserved in the PUT headers. + +```javascript +request.get('http://google.com/img.png').pipe(request.put('http://mysite.com/img.png')) +``` + +Now let's get fancy. + +```javascript +http.createServer(function (req, resp) { + if (req.url === '/doodle.png') { + if (req.method === 'PUT') { + req.pipe(request.put('http://mysite.com/doodle.png')) + } else if (req.method === 'GET' || req.method === 'HEAD') { + request.get('http://mysite.com/doodle.png').pipe(resp) + } + } +}) +``` + +You can also pipe() from a http.ServerRequest instance and to a http.ServerResponse instance. The HTTP method and headers will be sent as well as the entity-body data. Which means that, if you don't really care about security, you can do: + +```javascript +http.createServer(function (req, resp) { + if (req.url === '/doodle.png') { + var x = request('http://mysite.com/doodle.png') + req.pipe(x) + x.pipe(resp) + } +}) +``` + +And since pipe() returns the destination stream in node 0.5.x you can do one line proxying :) + +```javascript +req.pipe(request('http://mysite.com/doodle.png')).pipe(resp) +``` + +Also, none of this new functionality conflicts with requests previous features, it just expands them. + +```javascript +var r = request.defaults({'proxy':'http://localproxy.com'}) + +http.createServer(function (req, resp) { + if (req.url === '/doodle.png') { + r.get('http://google.com/doodle.png').pipe(resp) + } +}) +``` + +You can still use intermediate proxies, the requests will still follow HTTP forwards, etc. + +### request(options, callback) + +The first argument can be either a url or an options object. The only required option is uri, all others are optional. + +* `uri` || `url` - fully qualified uri or a parsed url object from url.parse() +* `method` - http method, defaults to GET +* `headers` - http headers, defaults to {} +* `body` - entity body for POST and PUT requests. Must be buffer or string. +* `json` - sets `body` but to JSON representation of value and adds `Content-type: application/json` header. +* `multipart` - (experimental) array of objects which contains their own headers and `body` attribute. Sends `multipart/related` request. See example below. +* `followRedirect` - follow HTTP 3xx responses as redirects. defaults to true. +* `maxRedirects` - the maximum number of redirects to follow, defaults to 10. +* `onResponse` - If true the callback will be fired on the "response" event instead of "end". If a function it will be called on "response" and not effect the regular semantics of the main callback on "end". +* `encoding` - Encoding to be used on response.setEncoding when buffering the response data. +* `pool` - A hash object containing the agents for these requests. If omitted this request will use the global pool which is set to node's default maxSockets. +* `pool.maxSockets` - Integer containing the maximum amount of sockets in the pool. +* `timeout` - Integer containing the number of milliseconds to wait for a request to respond before aborting the request +* `proxy` - An HTTP proxy to be used. Support proxy Auth with Basic Auth the same way it's supported with the `url` parameter by embedding the auth info in the uri. +* `strictSSL` - Set to `true` to require that SSL certificates be valid. Note: to use your own certificate authority, you need to specify an agent that was created with that ca as an option. + + +The callback argument gets 3 arguments. The first is an error when applicable (usually from the http.Client option not the http.ClientRequest object). The second in an http.ClientResponse object. The third is the response body buffer. + +## Convenience methods + +There are also shorthand methods for different HTTP METHODs and some other conveniences. + +### request.defaults(options) + +This method returns a wrapper around the normal request API that defaults to whatever options you pass in to it. + +### request.put + +Same as request() but defaults to `method: "PUT"`. + +```javascript +request.put(url) +``` + +### request.post + +Same as request() but defaults to `method: "POST"`. + +```javascript +request.post(url) +``` + +### request.head + +Same as request() but defaults to `method: "HEAD"`. + +```javascript +request.head(url) +``` + +### request.del + +Same as request() but defaults to `method: "DELETE"`. + +```javascript +request.del(url) +``` + +### request.get + +Alias to normal request method for uniformity. + +```javascript +request.get(url) +``` + + +## Examples: + +```javascript + var request = require('request') + , rand = Math.floor(Math.random()*100000000).toString() + ; + request( + { method: 'PUT' + , uri: 'http://mikeal.couchone.com/testjs/' + rand + , multipart: + [ { 'content-type': 'application/json' + , body: JSON.stringify({foo: 'bar', _attachments: {'message.txt': {follows: true, length: 18, 'content_type': 'text/plain' }}}) + } + , { body: 'I am an attachment' } + ] + } + , function (error, response, body) { + if(response.statusCode == 201){ + console.log('document saved as: http://mikeal.couchone.com/testjs/'+ rand) + } else { + console.log('error: '+ response.statusCode) + console.log(body) + } + } + ) +``` diff --git a/deps/npm/node_modules/request/main.js b/deps/npm/node_modules/request/main.js new file mode 100644 index 0000000000..e328e9b8b6 --- /dev/null +++ b/deps/npm/node_modules/request/main.js @@ -0,0 +1,506 @@ +// Copyright 2010-2011 Mikeal Rogers +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +var http = require('http') + , https = false + , tls = false + , url = require('url') + , util = require('util') + , stream = require('stream') + , qs = require('querystring') + , mimetypes = require('./mimetypes') + ; + +try { + https = require('https') +} catch (e) {} + +try { + tls = require('tls') +} catch (e) {} + +function toBase64 (str) { + return (new Buffer(str || "", "ascii")).toString("base64") +} + +// Hacky fix for pre-0.4.4 https +if (https && !https.Agent) { + https.Agent = function (options) { + http.Agent.call(this, options) + } + util.inherits(https.Agent, http.Agent) + https.Agent.prototype._getConnection = function(host, port, cb) { + var s = tls.connect(port, host, this.options, function() { + // do other checks here? + if (cb) cb() + }) + return s + } +} + +function isReadStream (rs) { + if (rs.readable && rs.path && rs.mode) { + return true + } +} + +function copy (obj) { + var o = {} + for (var i in obj) o[i] = obj[i] + return o +} + +var isUrl = /^https?:/ + +var globalPool = {} + +function Request (options) { + stream.Stream.call(this) + this.readable = true + this.writable = true + + if (typeof options === 'string') { + options = {uri:options} + } + + for (var i in options) { + this[i] = options[i] + } + if (!this.pool) this.pool = globalPool + this.dests = [] + this.__isRequestRequest = true +} +util.inherits(Request, stream.Stream) +Request.prototype.getAgent = function (host, port) { + if (!this.pool[host+':'+port]) { + this.pool[host+':'+port] = new this.httpModule.Agent({host:host, port:port}) + } + return this.pool[host+':'+port] +} +Request.prototype.request = function () { + var self = this + + // Protect against double callback + if (!self._callback && self.callback) { + self._callback = self.callback + self.callback = function () { + if (self._callbackCalled) return // Print a warning maybe? + self._callback.apply(self, arguments) + self._callbackCalled = true + } + } + + if (self.url) { + // People use this property instead all the time so why not just support it. + self.uri = self.url + delete self.url + } + + if (!self.uri) { + throw new Error("options.uri is a required argument") + } else { + if (typeof self.uri == "string") self.uri = url.parse(self.uri) + } + if (self.proxy) { + if (typeof self.proxy == 'string') self.proxy = url.parse(self.proxy) + } + + self._redirectsFollowed = self._redirectsFollowed || 0 + self.maxRedirects = (self.maxRedirects !== undefined) ? self.maxRedirects : 10 + self.followRedirect = (self.followRedirect !== undefined) ? self.followRedirect : true + if (self.followRedirect) + self.redirects = self.redirects || [] + + self.headers = self.headers ? copy(self.headers) : {} + + var setHost = false + if (!self.headers.host) { + self.headers.host = self.uri.hostname + if (self.uri.port) { + if ( !(self.uri.port === 80 && self.uri.protocol === 'http:') && + !(self.uri.port === 443 && self.uri.protocol === 'https:') ) + self.headers.host += (':'+self.uri.port) + } + setHost = true + } + + if (!self.uri.pathname) {self.uri.pathname = '/'} + if (!self.uri.port) { + if (self.uri.protocol == 'http:') {self.uri.port = 80} + else if (self.uri.protocol == 'https:') {self.uri.port = 443} + } + + if (self.proxy) { + self.port = self.proxy.port + self.host = self.proxy.hostname + } else { + self.port = self.uri.port + self.host = self.uri.hostname + } + + if (self.onResponse === true) { + self.onResponse = self.callback + delete self.callback + } + + var clientErrorHandler = function (error) { + if (setHost) delete self.headers.host + if (self.timeout && self.timeoutTimer) clearTimeout(self.timeoutTimer) + self.emit('error', error) + } + if (self.onResponse) self.on('error', function (e) {self.onResponse(e)}) + if (self.callback) self.on('error', function (e) {self.callback(e)}) + + + if (self.uri.auth && !self.headers.authorization) { + self.headers.authorization = "Basic " + toBase64(self.uri.auth.split(':').map(function(item){ return qs.unescape(item)}).join(':')) + } + if (self.proxy && self.proxy.auth && !self.headers['proxy-authorization']) { + self.headers['proxy-authorization'] = "Basic " + toBase64(self.proxy.auth.split(':').map(function(item){ return qs.unescape(item)}).join(':')) + } + + if (self.uri.path) { + self.path = self.uri.path + } else { + self.path = self.uri.pathname + (self.uri.search || "") + } + + if (self.path.length === 0) self.path = '/' + + if (self.proxy) self.path = (self.uri.protocol + '//' + self.uri.host + self.path) + + if (self.json) { + self.headers['content-type'] = 'application/json' + if (typeof self.json === 'boolean') { + if (typeof self.body === 'object') self.body = JSON.stringify(self.body) + } else { + self.body = JSON.stringify(self.json) + } + + } else if (self.multipart) { + self.body = '' + self.headers['content-type'] = 'multipart/related;boundary="frontier"' + if (!self.multipart.forEach) throw new Error('Argument error, options.multipart.') + + self.multipart.forEach(function (part) { + var body = part.body + if(!body) throw Error('Body attribute missing in multipart.') + delete part.body + self.body += '--frontier\r\n' + Object.keys(part).forEach(function(key){ + self.body += key + ': ' + part[key] + '\r\n' + }) + self.body += '\r\n' + body + '\r\n' + }) + self.body += '--frontier--' + } + + if (self.body) { + if (!Buffer.isBuffer(self.body)) { + self.body = new Buffer(self.body) + } + if (self.body.length) { + self.headers['content-length'] = self.body.length + } else { + throw new Error('Argument error, options.body.') + } + } + + self.httpModule = + {"http:":http, "https:":https}[self.proxy ? self.proxy.protocol : self.uri.protocol] + + if (!self.httpModule) throw new Error("Invalid protocol") + + if (self.pool === false) { + self.agent = false + } else { + if (self.maxSockets) { + // Don't use our pooling if node has the refactored client + self.agent = self.httpModule.globalAgent || self.getAgent(self.host, self.port) + self.agent.maxSockets = self.maxSockets + } + if (self.pool.maxSockets) { + // Don't use our pooling if node has the refactored client + self.agent = self.httpModule.globalAgent || self.getAgent(self.host, self.port) + self.agent.maxSockets = self.pool.maxSockets + } + } + + self.start = function () { + self._started = true + self.method = self.method || 'GET' + + self.req = self.httpModule.request(self, function (response) { + self.response = response + response.request = self + + if (self.httpModule === https && + self.strictSSL && + !response.client.authorized) { + var sslErr = response.client.authorizationError + self.emit('error', new Error('SSL Error: '+ sslErr)) + return + } + + if (setHost) delete self.headers.host + if (self.timeout && self.timeoutTimer) clearTimeout(self.timeoutTimer) + + if (response.statusCode >= 300 && + response.statusCode < 400 && + self.followRedirect && + self.method !== 'PUT' && + self.method !== 'POST' && + response.headers.location) { + if (self._redirectsFollowed >= self.maxRedirects) { + self.emit('error', new Error("Exceeded maxRedirects. Probably stuck in a redirect loop.")) + return + } + self._redirectsFollowed += 1 + + if (!isUrl.test(response.headers.location)) { + response.headers.location = url.resolve(self.uri.href, response.headers.location) + } + self.uri = response.headers.location + self.redirects.push( { statusCode : response.statusCode, + redirectUri: response.headers.location }) + delete self.req + delete self.agent + delete self._started + if (self.headers) { + delete self.headers.host + } + request(self, self.callback) + return // Ignore the rest of the response + } else { + self._redirectsFollowed = self._redirectsFollowed || 0 + // Be a good stream and emit end when the response is finished. + // Hack to emit end on close because of a core bug that never fires end + response.on('close', function () { + if (!self._ended) self.response.emit('end') + }) + + if (self.encoding) { + if (self.dests.length !== 0) { + console.error("Ingoring encoding parameter as this stream is being piped to another stream which makes the encoding option invalid.") + } else { + response.setEncoding(self.encoding) + } + } + + self.pipeDest = function (dest) { + if (dest.headers) { + dest.headers['content-type'] = response.headers['content-type'] + if (response.headers['content-length']) { + dest.headers['content-length'] = response.headers['content-length'] + } + } + if (dest.setHeader) { + for (var i in response.headers) { + dest.setHeader(i, response.headers[i]) + } + dest.statusCode = response.statusCode + } + if (self.pipefilter) self.pipefilter(response, dest) + } + + self.dests.forEach(function (dest) { + self.pipeDest(dest) + }) + + response.on("data", function (chunk) { + self._destdata = true + self.emit("data", chunk) + }) + response.on("end", function (chunk) { + self._ended = true + self.emit("end", chunk) + }) + response.on("close", function () {self.emit("close")}) + + self.emit('response', response) + + if (self.onResponse) { + self.onResponse(null, response) + } + if (self.callback) { + var buffer = [] + var bodyLen = 0 + self.on("data", function (chunk) { + buffer.push(chunk) + bodyLen += chunk.length + }) + self.on("end", function () { + if (buffer.length && Buffer.isBuffer(buffer[0])) { + var body = new Buffer(bodyLen) + var i = 0 + buffer.forEach(function (chunk) { + chunk.copy(body, i, 0, chunk.length) + i += chunk.length + }) + response.body = body.toString() + } else if (buffer.length) { + response.body = buffer.join('') + } + + if (self.json) { + try { + response.body = JSON.parse(response.body) + } catch (e) {} + } + self.callback(null, response, response.body) + }) + } + } + }) + + if (self.timeout) { + self.timeoutTimer = setTimeout(function() { + self.req.abort() + var e = new Error("ETIMEDOUT") + e.code = "ETIMEDOUT" + self.emit("error", e) + }, self.timeout) + } + + self.req.on('error', clientErrorHandler) + } + + self.once('pipe', function (src) { + if (self.ntick) throw new Error("You cannot pipe to this stream after the first nextTick() after creation of the request stream.") + self.src = src + if (isReadStream(src)) { + if (!self.headers['content-type'] && !self.headers['Content-Type']) + self.headers['content-type'] = mimetypes.lookup(src.path.slice(src.path.lastIndexOf('.')+1)) + } else { + if (src.headers) { + for (var i in src.headers) { + if (!self.headers[i]) { + self.headers[i] = src.headers[i] + } + } + } + if (src.method && !self.method) { + self.method = src.method + } + } + + self.on('pipe', function () { + console.error("You have already piped to this stream. Pipeing twice is likely to break the request.") + }) + }) + + process.nextTick(function () { + if (self.body) { + self.write(self.body) + self.end() + } else if (self.requestBodyStream) { + console.warn("options.requestBodyStream is deprecated, please pass the request object to stream.pipe.") + self.requestBodyStream.pipe(self) + } else if (!self.src) { + self.headers['content-length'] = 0 + self.end() + } + self.ntick = true + }) +} +Request.prototype.pipe = function (dest) { + if (this.response) { + if (this._destdata) { + throw new Error("You cannot pipe after data has been emitted from the response.") + } else if (this._ended) { + throw new Error("You cannot pipe after the response has been ended.") + } else { + stream.Stream.prototype.pipe.call(this, dest) + this.pipeDest(dest) + return dest + } + } else { + this.dests.push(dest) + stream.Stream.prototype.pipe.call(this, dest) + return dest + } +} +Request.prototype.write = function () { + if (!this._started) this.start() + if (!this.req) throw new Error("This request has been piped before http.request() was called.") + this.req.write.apply(this.req, arguments) +} +Request.prototype.end = function () { + if (!this._started) this.start() + if (!this.req) throw new Error("This request has been piped before http.request() was called.") + this.req.end.apply(this.req, arguments) +} +Request.prototype.pause = function () { + if (!this.response) throw new Error("This request has been piped before http.request() was called.") + this.response.pause.apply(this.response, arguments) +} +Request.prototype.resume = function () { + if (!this.response) throw new Error("This request has been piped before http.request() was called.") + this.response.resume.apply(this.response, arguments) +} + +function request (options, callback) { + if (typeof options === 'string') options = {uri:options} + if (callback) options.callback = callback + var r = new Request(options) + r.request() + return r +} + +module.exports = request + +request.defaults = function (options) { + var def = function (method) { + var d = function (opts, callback) { + if (typeof opts === 'string') opts = {uri:opts} + for (var i in options) { + if (opts[i] === undefined) opts[i] = options[i] + } + return method(opts, callback) + } + return d + } + var de = def(request) + de.get = def(request.get) + de.post = def(request.post) + de.put = def(request.put) + de.head = def(request.head) + de.del = def(request.del) + return de +} + +request.get = request +request.post = function (options, callback) { + if (typeof options === 'string') options = {uri:options} + options.method = 'POST' + return request(options, callback) +} +request.put = function (options, callback) { + if (typeof options === 'string') options = {uri:options} + options.method = 'PUT' + return request(options, callback) +} +request.head = function (options, callback) { + if (typeof options === 'string') options = {uri:options} + options.method = 'HEAD' + if (options.body || options.requestBodyStream || options.json || options.multipart) { + throw new Error("HTTP HEAD requests MUST NOT include a request body.") + } + return request(options, callback) +} +request.del = function (options, callback) { + if (typeof options === 'string') options = {uri:options} + options.method = 'DELETE' + return request(options, callback) +} diff --git a/deps/npm/node_modules/request/mimetypes.js b/deps/npm/node_modules/request/mimetypes.js new file mode 100644 index 0000000000..86910064c9 --- /dev/null +++ b/deps/npm/node_modules/request/mimetypes.js @@ -0,0 +1,146 @@ +// from http://github.com/felixge/node-paperboy +exports.types = { + "aiff":"audio/x-aiff", + "arj":"application/x-arj-compressed", + "asf":"video/x-ms-asf", + "asx":"video/x-ms-asx", + "au":"audio/ulaw", + "avi":"video/x-msvideo", + "bcpio":"application/x-bcpio", + "ccad":"application/clariscad", + "cod":"application/vnd.rim.cod", + "com":"application/x-msdos-program", + "cpio":"application/x-cpio", + "cpt":"application/mac-compactpro", + "csh":"application/x-csh", + "css":"text/css", + "deb":"application/x-debian-package", + "dl":"video/dl", + "doc":"application/msword", + "drw":"application/drafting", + "dvi":"application/x-dvi", + "dwg":"application/acad", + "dxf":"application/dxf", + "dxr":"application/x-director", + "etx":"text/x-setext", + "ez":"application/andrew-inset", + "fli":"video/x-fli", + "flv":"video/x-flv", + "gif":"image/gif", + "gl":"video/gl", + "gtar":"application/x-gtar", + "gz":"application/x-gzip", + "hdf":"application/x-hdf", + "hqx":"application/mac-binhex40", + "html":"text/html", + "ice":"x-conference/x-cooltalk", + "ico":"image/x-icon", + "ief":"image/ief", + "igs":"model/iges", + "ips":"application/x-ipscript", + "ipx":"application/x-ipix", + "jad":"text/vnd.sun.j2me.app-descriptor", + "jar":"application/java-archive", + "jpeg":"image/jpeg", + "jpg":"image/jpeg", + "js":"text/javascript", + "json":"application/json", + "latex":"application/x-latex", + "lsp":"application/x-lisp", + "lzh":"application/octet-stream", + "m":"text/plain", + "m3u":"audio/x-mpegurl", + "man":"application/x-troff-man", + "me":"application/x-troff-me", + "midi":"audio/midi", + "mif":"application/x-mif", + "mime":"www/mime", + "movie":"video/x-sgi-movie", + "mustache":"text/plain", + "mp4":"video/mp4", + "mpg":"video/mpeg", + "mpga":"audio/mpeg", + "ms":"application/x-troff-ms", + "nc":"application/x-netcdf", + "oda":"application/oda", + "ogm":"application/ogg", + "pbm":"image/x-portable-bitmap", + "pdf":"application/pdf", + "pgm":"image/x-portable-graymap", + "pgn":"application/x-chess-pgn", + "pgp":"application/pgp", + "pm":"application/x-perl", + "png":"image/png", + "pnm":"image/x-portable-anymap", + "ppm":"image/x-portable-pixmap", + "ppz":"application/vnd.ms-powerpoint", + "pre":"application/x-freelance", + "prt":"application/pro_eng", + "ps":"application/postscript", + "qt":"video/quicktime", + "ra":"audio/x-realaudio", + "rar":"application/x-rar-compressed", + "ras":"image/x-cmu-raster", + "rgb":"image/x-rgb", + "rm":"audio/x-pn-realaudio", + "rpm":"audio/x-pn-realaudio-plugin", + "rtf":"text/rtf", + "rtx":"text/richtext", + "scm":"application/x-lotusscreencam", + "set":"application/set", + "sgml":"text/sgml", + "sh":"application/x-sh", + "shar":"application/x-shar", + "silo":"model/mesh", + "sit":"application/x-stuffit", + "skt":"application/x-koan", + "smil":"application/smil", + "snd":"audio/basic", + "sol":"application/solids", + "spl":"application/x-futuresplash", + "src":"application/x-wais-source", + "stl":"application/SLA", + "stp":"application/STEP", + "sv4cpio":"application/x-sv4cpio", + "sv4crc":"application/x-sv4crc", + "svg":"image/svg+xml", + "swf":"application/x-shockwave-flash", + "tar":"application/x-tar", + "tcl":"application/x-tcl", + "tex":"application/x-tex", + "texinfo":"application/x-texinfo", + "tgz":"application/x-tar-gz", + "tiff":"image/tiff", + "tr":"application/x-troff", + "tsi":"audio/TSP-audio", + "tsp":"application/dsptype", + "tsv":"text/tab-separated-values", + "unv":"application/i-deas", + "ustar":"application/x-ustar", + "vcd":"application/x-cdlink", + "vda":"application/vda", + "vivo":"video/vnd.vivo", + "vrm":"x-world/x-vrml", + "wav":"audio/x-wav", + "wax":"audio/x-ms-wax", + "wma":"audio/x-ms-wma", + "wmv":"video/x-ms-wmv", + "wmx":"video/x-ms-wmx", + "wrl":"model/vrml", + "wvx":"video/x-ms-wvx", + "xbm":"image/x-xbitmap", + "xlw":"application/vnd.ms-excel", + "xml":"text/xml", + "xpm":"image/x-xpixmap", + "xwd":"image/x-xwindowdump", + "xyz":"chemical/x-pdb", + "zip":"application/zip", +}; + +exports.lookup = function(ext, defaultType) { + defaultType = defaultType || 'application/octet-stream'; + + return (ext in exports.types) + ? exports.types[ext] + : defaultType; +}; \ No newline at end of file diff --git a/deps/npm/node_modules/request/package.json b/deps/npm/node_modules/request/package.json new file mode 100644 index 0000000000..2d98aa2510 --- /dev/null +++ b/deps/npm/node_modules/request/package.json @@ -0,0 +1,15 @@ +{ "name" : "request" +, "description" : "Simplified HTTP request client." +, "tags" : ["http", "simple", "util", "utility"] +, "version" : "2.1.1" +, "author" : "Mikeal Rogers " +, "repository" : + { "type" : "git" + , "url" : "http://github.com/mikeal/request.git" + } +, "bugs" : + { "url" : "http://github.com/mikeal/request/issues" } +, "engines" : ["node >= 0.3.6"] +, "main" : "./main" +, "scripts": { "test": "bash tests/run.sh" } +} diff --git a/deps/npm/node_modules/request/tests/googledoodle.png b/deps/npm/node_modules/request/tests/googledoodle.png new file mode 100644 index 0000000000000000000000000000000000000000..f80c9c52d3c507996535a19ee0bcfe3821de322d GIT binary patch literal 38510 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 literal 0 HcmV?d00001 diff --git a/deps/npm/node_modules/request/tests/run.sh b/deps/npm/node_modules/request/tests/run.sh new file mode 100755 index 0000000000..57d0f64966 --- /dev/null +++ b/deps/npm/node_modules/request/tests/run.sh @@ -0,0 +1,6 @@ +FAILS=0 +for i in tests/test-*.js; do + echo $i + node $i || let FAILS++ +done +exit $FAILS diff --git a/deps/npm/node_modules/request/tests/server.js b/deps/npm/node_modules/request/tests/server.js new file mode 100644 index 0000000000..bad1e50305 --- /dev/null +++ b/deps/npm/node_modules/request/tests/server.js @@ -0,0 +1,57 @@ +var http = require('http') + , events = require('events') + , stream = require('stream') + , assert = require('assert') + ; + +exports.createServer = function (port) { + port = port || 6767 + var s = http.createServer(function (req, resp) { + s.emit(req.url, req, resp); + }) + s.port = port + s.url = 'http://localhost:'+port + return s; +} + +exports.createPostStream = function (text) { + var postStream = new stream.Stream(); + postStream.writeable = true; + postStream.readable = true; + setTimeout(function () {postStream.emit('data', new Buffer(text)); postStream.emit('end')}, 0); + return postStream; +} +exports.createPostValidator = function (text) { + var l = function (req, resp) { + var r = ''; + req.on('data', function (chunk) {r += chunk}) + req.on('end', function () { + if (r !== text) console.log(r, text); + assert.equal(r, text) + resp.writeHead(200, {'content-type':'text/plain'}) + resp.write('OK') + resp.end() + }) + } + return l; +} +exports.createGetResponse = function (text, contentType) { + var l = function (req, resp) { + contentType = contentType || 'text/plain' + resp.writeHead(200, {'content-type':contentType}) + resp.write(text) + resp.end() + } + return l; +} +exports.createChunkResponse = function (chunks, contentType) { + var l = function (req, resp) { + contentType = contentType || 'text/plain' + resp.writeHead(200, {'content-type':contentType}) + chunks.forEach(function (chunk) { + resp.write(chunk) + }) + resp.end() + } + return l; +} diff --git a/deps/npm/node_modules/request/tests/test-body.js b/deps/npm/node_modules/request/tests/test-body.js new file mode 100644 index 0000000000..18ad5b9c6f --- /dev/null +++ b/deps/npm/node_modules/request/tests/test-body.js @@ -0,0 +1,90 @@ +var server = require('./server') + , events = require('events') + , stream = require('stream') + , assert = require('assert') + , request = require('../main.js') + ; + +var s = server.createServer(); + +var tests = + { testGet : + { resp : server.createGetResponse("TESTING!") + , expectBody: "TESTING!" + } + , testGetChunkBreak : + { resp : server.createChunkResponse( + [ new Buffer([239]) + , new Buffer([163]) + , new Buffer([191]) + , new Buffer([206]) + , new Buffer([169]) + , new Buffer([226]) + , new Buffer([152]) + , new Buffer([131]) + ]) + , expectBody: "Ω☃" + } + , testGetJSON : + { resp : server.createGetResponse('{"test":true}', 'application/json') + , json : true + , expectBody: {"test":true} + } + , testPutString : + { resp : server.createPostValidator("PUTTINGDATA") + , method : "PUT" + , body : "PUTTINGDATA" + } + , testPutBuffer : + { resp : server.createPostValidator("PUTTINGDATA") + , method : "PUT" + , body : new Buffer("PUTTINGDATA") + } + , testPutJSON : + { resp : server.createPostValidator(JSON.stringify({foo: 'bar'})) + , method: "PUT" + , json: {foo: 'bar'} + } + , testPutMultipart : + { resp: server.createPostValidator( + '--frontier\r\n' + + 'content-type: text/html\r\n' + + '\r\n' + + 'Oh hi.' + + '\r\n--frontier\r\n\r\n' + + 'Oh hi.' + + '\r\n--frontier--' + ) + , method: "PUT" + , multipart: + [ {'content-type': 'text/html', 'body': 'Oh hi.'} + , {'body': 'Oh hi.'} + ] + } + } + +s.listen(s.port, function () { + + var counter = 0 + + for (i in tests) { + (function () { + var test = tests[i] + s.on('/'+i, test.resp) + test.uri = s.url + '/' + i + request(test, function (err, resp, body) { + if (err) throw err + if (test.expectBody) { + assert.deepEqual(test.expectBody, body) + } + counter = counter - 1; + if (counter === 0) { + console.log(Object.keys(tests).length+" tests passed.") + s.close() + } + }) + counter++ + })() + } +}) + diff --git a/deps/npm/node_modules/request/tests/test-errors.js b/deps/npm/node_modules/request/tests/test-errors.js new file mode 100644 index 0000000000..e5df87b56c --- /dev/null +++ b/deps/npm/node_modules/request/tests/test-errors.js @@ -0,0 +1,30 @@ +var server = require('./server') + , events = require('events') + , assert = require('assert') + , request = require('../main.js') + ; + +var local = 'http://localhost:8888/asdf' + +try { + request({uri:local, body:{}}) + assert.fail("Should have throw") +} catch(e) { + assert.equal(e.message, 'Argument error, options.body.') +} + +try { + request({uri:local, multipart: 'foo'}) + assert.fail("Should have throw") +} catch(e) { + assert.equal(e.message, 'Argument error, options.multipart.') +} + +try { + request({uri:local, multipart: [{}]}) + assert.fail("Should have throw") +} catch(e) { + assert.equal(e.message, 'Body attribute missing in multipart.') +} + +console.log("All tests passed.") \ No newline at end of file diff --git a/deps/npm/node_modules/request/tests/test-pipes.js b/deps/npm/node_modules/request/tests/test-pipes.js new file mode 100644 index 0000000000..0774647a17 --- /dev/null +++ b/deps/npm/node_modules/request/tests/test-pipes.js @@ -0,0 +1,167 @@ +var server = require('./server') + , events = require('events') + , stream = require('stream') + , assert = require('assert') + , fs = require('fs') + , request = require('../main.js') + , path = require('path') + , util = require('util') + ; + +var s = server.createServer(3453); + +function ValidationStream(str) { + this.str = str + this.buf = '' + this.on('data', function (data) { + this.buf += data + }) + this.on('end', function () { + assert.equal(this.str, this.buf) + }) + this.writable = true +} +util.inherits(ValidationStream, stream.Stream) +ValidationStream.prototype.write = function (chunk) { + this.emit('data', chunk) +} +ValidationStream.prototype.end = function (chunk) { + if (chunk) emit('data', chunk) + this.emit('end') +} + +s.listen(s.port, function () { + counter = 0; + + var check = function () { + counter = counter - 1 + if (counter === 0) { + console.log('All tests passed.') + setTimeout(function () { + process.exit(); + }, 500) + } + } + + // Test pipeing to a request object + s.once('/push', server.createPostValidator("mydata")); + + var mydata = new stream.Stream(); + mydata.readable = true + + counter++ + var r1 = request.put({url:'http://localhost:3453/push'}, function () { + check(); + }) + mydata.pipe(r1) + + mydata.emit('data', 'mydata'); + mydata.emit('end'); + + + // Test pipeing from a request object. + s.once('/pull', server.createGetResponse("mypulldata")); + + var mypulldata = new stream.Stream(); + mypulldata.writable = true + + counter++ + request({url:'http://localhost:3453/pull'}).pipe(mypulldata) + + var d = ''; + + mypulldata.write = function (chunk) { + d += chunk; + } + mypulldata.end = function () { + assert.equal(d, 'mypulldata'); + check(); + }; + + + s.on('/cat', function (req, resp) { + if (req.method === "GET") { + resp.writeHead(200, {'content-type':'text/plain-test', 'content-length':4}); + resp.end('asdf') + } else if (req.method === "PUT") { + assert.equal(req.headers['content-type'], 'text/plain-test'); + assert.equal(req.headers['content-length'], 4) + var validate = ''; + + req.on('data', function (chunk) {validate += chunk}) + req.on('end', function () { + resp.writeHead(201); + resp.end(); + assert.equal(validate, 'asdf'); + check(); + }) + } + }) + s.on('/pushjs', function (req, resp) { + if (req.method === "PUT") { + assert.equal(req.headers['content-type'], 'text/javascript'); + check(); + } + }) + s.on('/catresp', function (req, resp) { + request.get('http://localhost:3453/cat').pipe(resp) + }) + s.on('/doodle', function (req, resp) { + if (req.headers['x-oneline-proxy']) { + resp.setHeader('x-oneline-proxy', 'yup') + } + resp.writeHead('200', {'content-type':'image/png'}) + fs.createReadStream(path.join(__dirname, 'googledoodle.png')).pipe(resp) + }) + s.on('/onelineproxy', function (req, resp) { + var x = request('http://localhost:3453/doodle') + req.pipe(x) + x.pipe(resp) + }) + + counter++ + fs.createReadStream(__filename).pipe(request.put('http://localhost:3453/pushjs')) + + counter++ + request.get('http://localhost:3453/cat').pipe(request.put('http://localhost:3453/cat')) + + counter++ + request.get('http://localhost:3453/catresp', function (e, resp, body) { + assert.equal(resp.headers['content-type'], 'text/plain-test'); + assert.equal(resp.headers['content-length'], 4) + check(); + }) + + var doodleWrite = fs.createWriteStream(path.join(__dirname, 'test.png')) + + counter++ + request.get('http://localhost:3453/doodle').pipe(doodleWrite) + + doodleWrite.on('close', function () { + assert.deepEqual(fs.readFileSync(path.join(__dirname, 'googledoodle.png')), fs.readFileSync(path.join(__dirname, 'test.png'))) + check() + }) + + process.on('exit', function () { + fs.unlinkSync(path.join(__dirname, 'test.png')) + }) + + counter++ + request.get({uri:'http://localhost:3453/onelineproxy', headers:{'x-oneline-proxy':'nope'}}, function (err, resp, body) { + assert.equal(resp.headers['x-oneline-proxy'], 'yup') + check() + }) + + s.on('/afterresponse', function (req, resp) { + resp.write('d') + resp.end() + }) + + counter++ + var afterresp = request.post('http://localhost:3453/afterresponse').on('response', function () { + var v = new ValidationStream('d') + afterresp.pipe(v) + v.on('end', check) + }) + +}) diff --git a/deps/npm/node_modules/request/tests/test-timeout.js b/deps/npm/node_modules/request/tests/test-timeout.js new file mode 100644 index 0000000000..673f8ad86f --- /dev/null +++ b/deps/npm/node_modules/request/tests/test-timeout.js @@ -0,0 +1,87 @@ +var server = require('./server') + , events = require('events') + , stream = require('stream') + , assert = require('assert') + , request = require('../main.js') + ; + +var s = server.createServer(); +var expectedBody = "waited"; +var remainingTests = 5; + +s.listen(s.port, function () { + // Request that waits for 200ms + s.on('/timeout', function (req, resp) { + setTimeout(function(){ + resp.writeHead(200, {'content-type':'text/plain'}) + resp.write(expectedBody) + resp.end() + }, 200); + }); + + // Scenario that should timeout + var shouldTimeout = { + url: s.url + "/timeout", + timeout:100 + } + + + request(shouldTimeout, function (err, resp, body) { + assert.equal(err.code, "ETIMEDOUT"); + checkDone(); + }) + + + // Scenario that shouldn't timeout + var shouldntTimeout = { + url: s.url + "/timeout", + timeout:300 + } + + request(shouldntTimeout, function (err, resp, body) { + assert.equal(err, null); + assert.equal(expectedBody, body) + checkDone(); + }) + + // Scenario with no timeout set, so shouldn't timeout + var noTimeout = { + url: s.url + "/timeout" + } + + request(noTimeout, function (err, resp, body) { + assert.equal(err); + assert.equal(expectedBody, body) + checkDone(); + }) + + // Scenario with a negative timeout value, should be treated a zero or the minimum delay + var negativeTimeout = { + url: s.url + "/timeout", + timeout:-1000 + } + + request(negativeTimeout, function (err, resp, body) { + assert.equal(err.code, "ETIMEDOUT"); + checkDone(); + }) + + // Scenario with a float timeout value, should be rounded by setTimeout anyway + var floatTimeout = { + url: s.url + "/timeout", + timeout: 100.76 + } + + request(floatTimeout, function (err, resp, body) { + assert.equal(err.code, "ETIMEDOUT"); + checkDone(); + }) + + function checkDone() { + if(--remainingTests == 0) { + s.close(); + console.log("All tests passed."); + } + } +}) + diff --git a/deps/npm/node_modules/rimraf/AUTHORS b/deps/npm/node_modules/rimraf/AUTHORS new file mode 100644 index 0000000000..008cbe7dd9 --- /dev/null +++ b/deps/npm/node_modules/rimraf/AUTHORS @@ -0,0 +1,5 @@ +# Authors sorted by whether or not they're me. +Isaac Z. Schlueter (http://blog.izs.me) +Wayne Larsen (http://github.com/wvl) +ritch +Marcel Laverdet diff --git a/deps/npm/node_modules/rimraf/LICENSE b/deps/npm/node_modules/rimraf/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/rimraf/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/rimraf/README.md b/deps/npm/node_modules/rimraf/README.md new file mode 100644 index 0000000000..99983dc437 --- /dev/null +++ b/deps/npm/node_modules/rimraf/README.md @@ -0,0 +1,32 @@ +A `rm -rf` for node. + +Install with `npm install rimraf`, or just drop rimraf.js somewhere. + +## API + +`rimraf(f, [options,] callback)` + +The callback will be called with an error if there is one. Certain +errors are handled for you: + +* `EBUSY` - rimraf will back off a maximum of opts.maxBusyTries times + before giving up. +* `EMFILE` - If too many file descriptors get opened, rimraf will + patiently wait until more become available. + +## Options + +The options object is optional. These fields are respected: + +* `maxBusyTries` - The number of times to retry a file or folder in the + event of an `EBUSY` error. The default is 3. +* `gently` - If provided a `gently` path, then rimraf will only delete + files and folders that are beneath this path, and only delete symbolic + links that point to a place within this path. (This is very important + to npm's use-case, and shows rimraf's pedigree.) + + +## rimraf.sync + +It can remove stuff synchronously, too. But that's not so good. Use +the async API. It's better. diff --git a/deps/npm/node_modules/rimraf/fiber.js b/deps/npm/node_modules/rimraf/fiber.js new file mode 100644 index 0000000000..8812a6b449 --- /dev/null +++ b/deps/npm/node_modules/rimraf/fiber.js @@ -0,0 +1,86 @@ +// fiber/future port originally written by Marcel Laverdet +// https://gist.github.com/1131093 +// I updated it to bring to feature parity with cb version. +// The bugs are probably mine, not Marcel's. +// -- isaacs + +var path = require('path') + , fs = require('fs') + , Future = require('fibers/future') + +// Create future-returning fs functions +var fs2 = {} +for (var ii in fs) { + fs2[ii] = Future.wrap(fs[ii]) +} + +// Return a future which just pauses for a certain amount of time + +function timer (ms) { + var future = new Future + setTimeout(function () { + future.return() + }, ms) + return future +} + +function realish (p) { + return path.resolve(path.dirname(fs2.readlink(p))) +} + +// for EMFILE backoff. +var timeout = 0 + , EMFILE_MAX = 1000 + +function rimraf_ (p, opts) { + opts = opts || {} + opts.maxBusyTries = opts.maxBusyTries || 3 + if (opts.gently) opts.gently = path.resolve(opts.gently) + var busyTries = 0 + + // exits by throwing or returning. + // loops on handled errors. + while (true) { + try { + var stat = fs2.lstat(p).wait() + + // check to make sure that symlinks are ours. + if (opts.gently) { + var rp = stat.isSymbolicLink() ? realish(p) : path.resolve(p) + if (rp.indexOf(opts.gently) !== 0) { + var er = new Error("Refusing to delete: "+p+" not in "+opts.gently) + er.errno = require("constants").EEXIST + er.code = "EEXIST" + er.path = p + throw er + } + } + + if (!stat.isDirectory()) return fs2.unlink(p).wait() + + var rimrafs = fs2.readdir(p).wait().map(function (file) { + return rimraf(path.join(p, file), opts) + }) + + Future.wait(rimrafs) + fs2.rmdir(p).wait() + timeout = 0 + return + + } catch (er) { + if (er.message.match(/^EMFILE/) && timeout < EMFILE_MAX) { + timer(timeout++).wait() + } else if (er.message.match(/^EBUSY/) + && busyTries < opt.maxBusyTries) { + timer(++busyTries * 100).wait() + } else if (er.message.match(/^ENOENT/)) { + // already gone + return + } else { + throw er + } + } + } +} + +var rimraf = module.exports = rimraf_.future() diff --git a/deps/npm/node_modules/rimraf/package.json b/deps/npm/node_modules/rimraf/package.json new file mode 100644 index 0000000000..5b48d5bb2e --- /dev/null +++ b/deps/npm/node_modules/rimraf/package.json @@ -0,0 +1,9 @@ +{"name":"rimraf" +,"version":"1.0.8" +,"main":"rimraf.js" +,"description":"A deep deletion module for node (like `rm -rf`)" +,"author":"Isaac Z. Schlueter (http://blog.izs.me/)" +,"license": + {"type":"MIT", "url": "https://github.com/isaacs/rimraf/raw/master/LICENSE"} +,"repository":"git://github.com/isaacs/rimraf.git" +,"scripts":{"test":"cd test && bash run.sh"}} diff --git a/deps/npm/node_modules/rimraf/rimraf.js b/deps/npm/node_modules/rimraf/rimraf.js new file mode 100644 index 0000000000..27e22669e3 --- /dev/null +++ b/deps/npm/node_modules/rimraf/rimraf.js @@ -0,0 +1,140 @@ +module.exports = rimraf +rimraf.sync = rimrafSync + +var path = require("path") + , fs + +try { + // optional dependency + fs = require("graceful-fs") +} catch (er) { + fs = require("fs") +} + +var lstat = process.platform === "win32" ? "stat" : "lstat" + , lstatSync = lstat + "Sync" + +// for EMFILE handling +var timeout = 0 + , EMFILE_MAX = 1000 + +function rimraf (p, opts, cb) { + if (typeof opts === "function") cb = opts, opts = {} + + if (!cb) throw new Error("No callback passed to rimraf()") + if (!opts) opts = {} + + var busyTries = 0 + opts.maxBusyTries = opts.maxBusyTries || 3 + + if (opts.gently) opts.gently = path.resolve(opts.gently) + + rimraf_(p, opts, function CB (er) { + if (er) { + if (er.message.match(/^EBUSY/) && busyTries < opts.maxBusyTries) { + var time = (opts.maxBusyTries - busyTries) * 100 + busyTries ++ + // try again, with the same exact callback as this one. + return setTimeout(function () { + rimraf_(p, opts, CB) + }) + } + + // this one won't happen if graceful-fs is used. + if (er.message.match(/^EMFILE/) && timeout < EMFILE_MAX) { + return setTimeout(function () { + rimraf_(p, opts, CB) + }, timeout ++) + } + + // already gone + if (er.message.match(/^ENOENT/)) er = null + } + + timeout = 0 + cb(er) + }) +} + +function rimraf_ (p, opts, cb) { + fs[lstat](p, function (er, s) { + // if the stat fails, then assume it's already gone. + if (er) { + // already gone + if (er.message.match(/^ENOENT/)) return cb() + // some other kind of error, permissions, etc. + return cb(er) + } + + // don't delete that don't point actually live in the "gently" path + if (opts.gently) return clobberTest(p, s, opts, cb) + return rm_(p, s, opts, cb) + }) +} + +function rm_ (p, s, opts, cb) { + if (!s.isDirectory()) return fs.unlink(p, cb) + fs.readdir(p, function (er, files) { + if (er) return cb(er) + asyncForEach(files.map(function (f) { + return path.join(p, f) + }), function (file, cb) { + rimraf(file, opts, cb) + }, function (er) { + if (er) return cb(er) + fs.rmdir(p, cb) + }) + }) +} + +function clobberTest (p, s, opts, cb) { + var gently = opts.gently + if (!s.isSymbolicLink()) next(null, path.resolve(p)) + else realish(p, next) + + function next (er, rp) { + if (er) return rm_(p, s, cb) + if (rp.indexOf(gently) !== 0) return clobberFail(p, gently, cb) + else return rm_(p, s, opts, cb) + } +} + +function realish (p, cb) { + fs.readlink(p, function (er, r) { + if (er) return cb(er) + return cb(null, path.resolve(path.dirname(p), r)) + }) +} + +function clobberFail (p, g, cb) { + var er = new Error("Refusing to delete: "+p+" not in "+g) + , constants = require("constants") + er.errno = constants.EEXIST + er.code = "EEXIST" + er.path = p + return cb(er) +} + +function asyncForEach (list, fn, cb) { + if (!list.length) cb() + var c = list.length + , errState = null + list.forEach(function (item, i, list) { + fn(item, function (er) { + if (errState) return + if (er) return cb(errState = er) + if (-- c === 0) return cb() + }) + }) +} + +// this looks simpler, but it will fail with big directory trees, +// or on slow stupid awful cygwin filesystems +function rimrafSync (p) { + var s = fs[lstatSync](p) + if (!s.isDirectory()) return fs.unlinkSync(p) + fs.readdirSync(p).forEach(function (f) { + rimrafSync(path.join(p, f)) + }) + fs.rmdirSync(p) +} diff --git a/deps/npm/node_modules/rimraf/test/run.sh b/deps/npm/node_modules/rimraf/test/run.sh new file mode 100644 index 0000000000..598f0163b2 --- /dev/null +++ b/deps/npm/node_modules/rimraf/test/run.sh @@ -0,0 +1,10 @@ +#!/bin/bash +set -e +for i in test-*.js; do + echo -n $i ... + bash setup.sh + node $i + ! [ -d target ] + echo "pass" +done +rm -rf target diff --git a/deps/npm/node_modules/rimraf/test/setup.sh b/deps/npm/node_modules/rimraf/test/setup.sh new file mode 100644 index 0000000000..2602e63160 --- /dev/null +++ b/deps/npm/node_modules/rimraf/test/setup.sh @@ -0,0 +1,47 @@ +#!/bin/bash + +set -e + +files=10 +folders=2 +depth=4 +target="$PWD/target" + +rm -rf target + +fill () { + local depth=$1 + local files=$2 + local folders=$3 + local target=$4 + + if ! [ -d $target ]; then + mkdir -p $target + fi + + local f + + f=$files + while [ $f -gt 0 ]; do + touch "$target/f-$depth-$f" + let f-- + done + + let depth-- + + if [ $depth -le 0 ]; then + return 0 + fi + + f=$folders + while [ $f -gt 0 ]; do + mkdir "$target/folder-$depth-$f" + fill $depth $files $folders "$target/d-$depth-$f" + let f-- + done +} + +fill $depth $files $folders $target + +# sanity assert +[ -d $target ] diff --git a/deps/npm/node_modules/rimraf/test/test-async.js b/deps/npm/node_modules/rimraf/test/test-async.js new file mode 100644 index 0000000000..9c2e0b7be0 --- /dev/null +++ b/deps/npm/node_modules/rimraf/test/test-async.js @@ -0,0 +1,5 @@ +var rimraf = require("../rimraf") + , path = require("path") +rimraf(path.join(__dirname, "target"), function (er) { + if (er) throw er +}) diff --git a/deps/npm/node_modules/rimraf/test/test-fiber.js b/deps/npm/node_modules/rimraf/test/test-fiber.js new file mode 100644 index 0000000000..20d61a1099 --- /dev/null +++ b/deps/npm/node_modules/rimraf/test/test-fiber.js @@ -0,0 +1,15 @@ +var rimraf + , path = require("path") + +try { + rimraf = require("../fiber") +} catch (er) { + console.error("skipping fiber test") +} + +if (rimraf) { + Fiber(function () { + rimraf(path.join(__dirname, "target")).wait() + }).run() +} + diff --git a/deps/npm/node_modules/rimraf/test/test-sync.js b/deps/npm/node_modules/rimraf/test/test-sync.js new file mode 100644 index 0000000000..eb71f10476 --- /dev/null +++ b/deps/npm/node_modules/rimraf/test/test-sync.js @@ -0,0 +1,3 @@ +var rimraf = require("../rimraf") + , path = require("path") +rimraf.sync(path.join(__dirname, "target")) diff --git a/deps/npm/node_modules/semver/LICENSE b/deps/npm/node_modules/semver/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/semver/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/semver/README.md b/deps/npm/node_modules/semver/README.md new file mode 100644 index 0000000000..6fa37a3d89 --- /dev/null +++ b/deps/npm/node_modules/semver/README.md @@ -0,0 +1,119 @@ +semver(1) -- The semantic versioner for npm +=========================================== + +## Usage + + $ npm install semver + + semver.valid('1.2.3') // true + semver.valid('a.b.c') // false + semver.clean(' =v1.2.3 ') // '1.2.3' + semver.satisfies('1.2.3', '1.x || >=2.5.0 || 5.0.0 - 7.2.3') // true + semver.gt('1.2.3', '9.8.7') // false + semver.lt('1.2.3', '9.8.7') // true + +As a command-line utility: + + $ semver -h + + Usage: semver -v [-r ] + Test if version(s) satisfy the supplied range(s), + and sort them. + + Multiple versions or ranges may be supplied. + + Program exits successfully if any valid version satisfies + all supplied ranges, and prints all satisfying versions. + + If no versions are valid, or ranges are not satisfied, + then exits failure. + + Versions are printed in ascending order, so supplying + multiple versions to the utility will just sort them. + +## Versions + +A version is the following things, in this order: + +* a number (Major) +* a period +* a number (minor) +* a period +* a number (patch) +* OPTIONAL: a hyphen, followed by a number (build) +* OPTIONAL: a collection of pretty much any non-whitespace characters + (tag) + +A leading `"="` or `"v"` character is stripped off and ignored. + +## Comparisons + +The ordering of versions is done using the following algorithm, given +two versions and asked to find the greater of the two: + +* If the majors are numerically different, then take the one + with a bigger major number. `2.3.4 > 1.3.4` +* If the minors are numerically different, then take the one + with the bigger minor number. `2.3.4 > 2.2.4` +* If the patches are numerically different, then take the one with the + bigger patch number. `2.3.4 > 2.3.3` +* If only one of them has a build number, then take the one with the + build number. `2.3.4-0 > 2.3.4` +* If they both have build numbers, and the build numbers are numerically + different, then take the one with the bigger build number. + `2.3.4-10 > 2.3.4-9` +* If only one of them has a tag, then take the one without the tag. + `2.3.4 > 2.3.4-beta` +* If they both have tags, then take the one with the lexicographically + larger tag. `2.3.4-beta > 2.3.4-alpha` +* At this point, they're equal. + +## Ranges + +The following range styles are supported: + +* `>1.2.3` Greater than a specific version. +* `<1.2.3` Less than +* `1.2.3 - 2.3.4` := `>=1.2.3 <=2.3.4` +* `~1.2.3` := `>=1.2.3 <1.3.0` +* `~1.2` := `>=1.2.0 <2.0.0` +* `~1` := `>=1.0.0 <2.0.0` +* `1.2.x` := `>=1.2.0 <1.3.0` +* `1.x` := `>=1.0.0 <2.0.0` + +Ranges can be joined with either a space (which implies "and") or a +`||` (which implies "or"). + +## Functions + +* valid(v): Return the parsed version, or null if it's not valid. +* inc(v, release): Return the version incremented by the release type + (major, minor, patch, or build), or null if it's not valid. + +### Comparison + +* gt(v1, v2): `v1 > v2` +* gte(v1, v2): `v1 >= v2` +* lt(v1, v2): `v1 < v2` +* lte(v1, v2): `v1 <= v2` +* eq(v1, v2): `v1 == v2` This is true if they're logically equivalent, + even if they're not the exact same string. You already know how to + compare strings. +* neq(v1, v2): `v1 != v2` The opposite of eq. +* cmp(v1, comparator, v2): Pass in a comparison string, and it'll call + the corresponding function above. `"==="` and `"!=="` do simple + string comparison, but are included for completeness. Throws if an + invalid comparison string is provided. +* compare(v1, v2): Return 0 if v1 == v2, or 1 if v1 is greater, or -1 if + v2 is greater. Sorts in ascending order if passed to Array.sort(). +* rcompare(v1, v2): The reverse of compare. Sorts an array of versions + in descending order when passed to Array.sort(). + + +### Ranges + +* validRange(range): Return the valid range or null if it's not valid +* satisfies(version, range): Return true if the version satisfies the + range. +* maxSatisfying(versions, range): Return the highest version in the list + that satisfies the range, or null if none of them do. diff --git a/deps/npm/node_modules/semver/bin/semver b/deps/npm/node_modules/semver/bin/semver new file mode 100755 index 0000000000..3e6afb40d8 --- /dev/null +++ b/deps/npm/node_modules/semver/bin/semver @@ -0,0 +1,71 @@ +#!/usr/bin/env node +// Standalone semver comparison program. +// Exits successfully and prints matching version(s) if +// any supplied version is valid and passes all tests. + +var argv = process.argv.slice(2) + , versions = [] + , range = [] + , gt = [] + , lt = [] + , eq = [] + , semver = require("../semver") + +main() + +function main () { + if (!argv.length) return help() + while (argv.length) { + var a + switch (a = argv.shift()) { + case "-v": case "--version": + versions.push(argv.shift()) + break + case "-r": case "--range": + range.push(argv.shift()) + break + case "-h": case "--help": case "-?": + return help() + default: + versions.push(a) + break + } + } + + versions = versions.filter(semver.valid) + for (var i = 0, l = range.length; i < l ; i ++) { + versions = versions.filter(function (v) { + return semver.satisfies(v, range[i]) + }) + if (!versions.length) return fail() + } + return success(versions) +} + +function fail () { process.exit(1) } + +function success () { + versions.sort(semver.compare) + .map(semver.clean) + .forEach(function (v,i,_) { console.log(v) }) +} + +function help () { + console.log(["Usage: semver -v [-r ]" + ,"Test if version(s) satisfy the supplied range(s)," + ,"and sort them." + ,"" + ,"Multiple versions or ranges may be supplied." + ,"" + ,"Program exits successfully if any valid version satisfies" + ,"all supplied ranges, and prints all satisfying versions." + ,"" + ,"If no versions are valid, or ranges are not satisfied," + ,"then exits failure." + ,"" + ,"Versions are printed in ascending order, so supplying" + ,"multiple versions to the utility will just sort them." + ].join("\n")) +} + + diff --git a/deps/npm/node_modules/semver/package.json b/deps/npm/node_modules/semver/package.json new file mode 100644 index 0000000000..cd10e341ce --- /dev/null +++ b/deps/npm/node_modules/semver/package.json @@ -0,0 +1,11 @@ +{ "name" : "semver" +, "version" : "1.0.12" +, "description" : "The semantic version parser used by npm." +, "main" : "semver.js" +, "scripts" : { "test" : "tap test.js" } +, "devDependencies": { "tap" : "0.x >=0.0.4" } +, "license" : + { "type" : "MIT" + , "url" : "https://github.com/isaacs/semver/raw/master/LICENSE" } +, "repository" : "git://github.com/isaacs/node-semver.git" +, "bin" : { "semver" : "./bin/semver" } } diff --git a/deps/npm/node_modules/semver/semver.js b/deps/npm/node_modules/semver/semver.js new file mode 100644 index 0000000000..789b118b9f --- /dev/null +++ b/deps/npm/node_modules/semver/semver.js @@ -0,0 +1,303 @@ + +// See http://semver.org/ +// This implementation is a *hair* less strict in that it allows +// v1.2.3 things, and also tags that don't begin with a char. + +var semver = "\\s*[v=]*\\s*([0-9]+)" // major + + "\\.([0-9]+)" // minor + + "\\.([0-9]+)" // patch + + "(-[0-9]+-?)?" // build + + "([a-zA-Z-][a-zA-Z0-9-\.:]*)?" // tag + , exprComparator = "^((<|>)?=?)\s*("+semver+")$|^$" + , xRangePlain = "[v=]*([0-9]+|x|X|\\*)" + + "(?:\\.([0-9]+|x|X|\\*)" + + "(?:\\.([0-9]+|x|X|\\*)" + + "([a-zA-Z-][a-zA-Z0-9-\.:]*)?)?)?" + , xRange = "((?:<|>)?=?)?\\s*" + xRangePlain + , exprSpermy = "(?:~>?)"+xRange + , expressions = exports.expressions = + { parse : new RegExp("^\\s*"+semver+"\\s*$") + , parsePackage : new RegExp("^\\s*([^\/]+)[-@](" +semver+")\\s*$") + , parseRange : new RegExp( + "^\\s*(" + semver + ")\\s+-\\s+(" + semver + ")\\s*$") + , validComparator : new RegExp("^"+exprComparator+"$") + , parseXRange : new RegExp("^"+xRange+"$") + , parseSpermy : new RegExp("^"+exprSpermy+"$") + } + + +Object.getOwnPropertyNames(expressions).forEach(function (i) { + exports[i] = function (str) { + return ("" + (str || "")).match(expressions[i]) + } +}) + +exports.rangeReplace = ">=$1 <=$7" +exports.clean = clean +exports.compare = compare +exports.satisfies = satisfies +exports.gt = gt +exports.gte = gte +exports.lt = lt +exports.lte = lte +exports.eq = eq +exports.neq = neq +exports.cmp = cmp +exports.inc = inc + +exports.valid = valid +exports.validPackage = validPackage +exports.validRange = validRange +exports.maxSatisfying = maxSatisfying + +exports.replaceStars = replaceStars +exports.toComparators = toComparators + +function stringify (version) { + var v = version + return [v[1]||'', v[2]||'', v[3]||''].join(".") + (v[4]||'') + (v[5]||'') +} + +function clean (version) { + version = exports.parse(version) + if (!version) return version + return stringify(version) +} + +function valid (version) { + if (typeof version !== "string") return null + return exports.parse(version) && version.trim().replace(/^[v=]+/, '') +} + +function validPackage (version) { + if (typeof version !== "string") return null + return version.match(expressions.parsePackage) && version.trim() +} + +// range can be one of: +// "1.0.3 - 2.0.0" range, inclusive, like ">=1.0.3 <=2.0.0" +// ">1.0.2" like 1.0.3 - 9999.9999.9999 +// ">=1.0.2" like 1.0.2 - 9999.9999.9999 +// "<2.0.0" like 0.0.0 - 1.9999.9999 +// ">1.0.2 <2.0.0" like 1.0.3 - 1.9999.9999 +var starExpression = /(<|>)?=?\s*\*/g + , starReplace = "" + , compTrimExpression = new RegExp("((<|>)?=?)\\s*(" + +semver+"|"+xRangePlain+")", "g") + , compTrimReplace = "$1$3" + +function toComparators (range) { + var ret = (range || "").trim() + .replace(expressions.parseRange, exports.rangeReplace) + .replace(compTrimExpression, compTrimReplace) + .split(/\s+/) + .join(" ") + .split("||") + .map(function (orchunk) { + return orchunk + .split(" ") + .map(replaceXRanges) + .map(replaceSpermies) + .map(replaceStars) + .join(" ").trim() + }) + .map(function (orchunk) { + return orchunk + .trim() + .split(/\s+/) + .filter(function (c) { return c.match(expressions.validComparator) }) + }) + .filter(function (c) { return c.length }) + return ret +} + +function replaceStars (stars) { + return stars.trim().replace(starExpression, starReplace) +} + +// "2.x","2.x.x" --> ">=2.0.0- <2.1.0-" +// "2.3.x" --> ">=2.3.0- <2.4.0-" +function replaceXRanges (ranges) { + return ranges.split(/\s+/) + .map(replaceXRange) + .join(" ") +} + +function replaceXRange (version) { + return version.trim().replace(expressions.parseXRange, + function (v, gtlt, M, m, p, t) { + var anyX = !M || M.toLowerCase() === "x" || M === "*" + || !m || m.toLowerCase() === "x" || m === "*" + || !p || p.toLowerCase() === "x" || p === "*" + , ret = v + + if (gtlt && anyX) { + // just replace x'es with zeroes + ;(!M || M === "*" || M.toLowerCase() === "x") && (M = 0) + ;(!m || m === "*" || m.toLowerCase() === "x") && (m = 0) + ;(!p || p === "*" || p.toLowerCase() === "x") && (p = 0) + ret = gtlt + M+"."+m+"."+p+"-" + } else if (!M || M === "*" || M.toLowerCase() === "x") { + ret = "*" // allow any + } else if (!m || m === "*" || m.toLowerCase() === "x") { + // append "-" onto the version, otherwise + // "1.x.x" matches "2.0.0beta", since the tag + // *lowers* the version value + ret = ">="+M+".0.0- <"+(+M+1)+".0.0-" + } else if (!p || p === "*" || p.toLowerCase() === "x") { + ret = ">="+M+"."+m+".0- <"+M+"."+(+m+1)+".0-" + } + //console.error("parseXRange", [].slice.call(arguments), ret) + return ret + }) +} + +// ~, ~> --> * (any, kinda silly) +// ~2, ~2.x, ~2.x.x, ~>2, ~>2.x ~>2.x.x --> >=2.0.0 <3.0.0 +// ~2.0, ~2.0.x, ~>2.0, ~>2.0.x --> >=2.0.0 <2.1.0 +// ~1.2, ~1.2.x, ~>1.2, ~>1.2.x --> >=1.2.0 <1.3.0 +// ~1.2.3, ~>1.2.3 --> >=1.2.3 <1.3.0 +// ~1.2.0, ~>1.2.0 --> >=1.2.0 <1.3.0 +function replaceSpermies (version) { + return version.trim().replace(expressions.parseSpermy, + function (v, gtlt, M, m, p, t) { + if (gtlt) throw new Error( + "Using '"+gtlt+"' with ~ makes no sense. Don't do it.") + + if (!M || M.toLowerCase() === "x") { + return "" + } + // ~1 == >=1.0.0- <2.0.0- + if (!m || m.toLowerCase() === "x") { + return ">="+M+".0.0- <"+(+M+1)+".0.0-" + } + // ~1.2 == >=1.2.0- <1.3.0- + if (!p || p.toLowerCase() === "x") { + return ">="+M+"."+m+".0- <"+M+"."+(+m+1)+".0-" + } + // ~1.2.3 == >=1.2.3- <1.3.0- + t = t || "-" + return ">="+M+"."+m+"."+p+t+" <"+M+"."+(+m+1)+".0-" + }) +} + +function validRange (range) { + range = replaceStars(range) + var c = toComparators(range) + return (c.length === 0) + ? null + : c.map(function (c) { return c.join(" ") }).join("||") +} + +// returns the highest satisfying version in the list, or undefined +function maxSatisfying (versions, range) { + return versions + .filter(function (v) { return satisfies(v, range) }) + .sort(compare) + .pop() +} +function satisfies (version, range) { + version = valid(version) + if (!version) return false + range = toComparators(range) + for (var i = 0, l = range.length ; i < l ; i ++) { + var ok = false + for (var j = 0, ll = range[i].length ; j < ll ; j ++) { + var r = range[i][j] + , gtlt = r.charAt(0) === ">" ? gt + : r.charAt(0) === "<" ? lt + : false + , eq = r.charAt(!!gtlt) === "=" + , sub = (!!eq) + (!!gtlt) + if (!gtlt) eq = true + r = r.substr(sub) + r = (r === "") ? r : valid(r) + ok = (r === "") || (eq && r === version) || (gtlt && gtlt(version, r)) + if (!ok) break + } + if (ok) return true + } + return false +} + +// return v1 > v2 ? 1 : -1 +function compare (v1, v2) { + var g = gt(v1, v2) + return g === null ? 0 : g ? 1 : -1 +} + +function rcompare (v1, v2) { + return compare(v2, v1) +} + +function lt (v1, v2) { return gt(v2, v1) } +function gte (v1, v2) { return !lt(v1, v2) } +function lte (v1, v2) { return !gt(v1, v2) } +function eq (v1, v2) { return gt(v1, v2) === null } +function neq (v1, v2) { return gt(v1, v2) !== null } +function cmp (v1, c, v2) { + switch (c) { + case ">": return gt(v1, v2) + case "<": return lt(v1, v2) + case ">=": return gte(v1, v2) + case "<=": return lte(v1, v2) + case "==": return eq(v1, v2) + case "!=": return neq(v1, v2) + case "===": return v1 === v2 + case "!==": return v1 !== v2 + default: throw new Error("Y U NO USE VALID COMPARATOR!? "+c) + } +} + +// return v1 > v2 +function num (v) { + return v === undefined ? -1 : parseInt((v||"0").replace(/[^0-9]+/g, ''), 10) +} +function gt (v1, v2) { + v1 = exports.parse(v1) + v2 = exports.parse(v2) + if (!v1 || !v2) return false + + for (var i = 1; i < 5; i ++) { + v1[i] = num(v1[i]) + v2[i] = num(v2[i]) + if (v1[i] > v2[i]) return true + else if (v1[i] !== v2[i]) return false + } + // no tag is > than any tag, or use lexicographical order. + var tag1 = v1[5] || "" + , tag2 = v2[5] || "" + + // kludge: null means they were equal. falsey, and detectable. + // embarrassingly overclever, though, I know. + return tag1 === tag2 ? null + : !tag1 ? true + : !tag2 ? false + : tag1 > tag2 +} + +function inc (version, release) { + version = exports.parse(version) + if (!version) return null + + var parsedIndexLookup = + { 'major': 1 + , 'minor': 2 + , 'patch': 3 + , 'build': 4 } + var incIndex = parsedIndexLookup[release] + if (incIndex === undefined) return null + + var current = num(version[incIndex]) + version[incIndex] = current === -1 ? 1 : current + 1 + + for (var i = incIndex + 1; i < 5; i ++) { + if (num(version[i]) !== -1) version[i] = "0" + } + + if (version[4]) version[4] = "-" + version[4] + version[5] = "" + + return stringify(version) +} + diff --git a/deps/npm/node_modules/semver/test.js b/deps/npm/node_modules/semver/test.js new file mode 100644 index 0000000000..c28fe39701 --- /dev/null +++ b/deps/npm/node_modules/semver/test.js @@ -0,0 +1,397 @@ +var tap = require("tap") + , test = tap.test + , semver = require("./semver.js") + , eq = semver.eq + , gt = semver.gt + , lt = semver.lt + , neq = semver.neq + , cmp = semver.cmp + , gte = semver.gte + , lte = semver.lte + , satisfies = semver.satisfies + , validRange = semver.validRange + , inc = semver.inc + , replaceStars = semver.replaceStars + , toComparators = semver.toComparators + +tap.plan(8) + +test("\ncomparison tests", function (t) { +; [ ["0.0.0", "0.0.0foo"] + , ["0.0.1", "0.0.0"] + , ["1.0.0", "0.9.9"] + , ["0.10.0", "0.9.0"] + , ["0.99.0", "0.10.0"] + , ["2.0.0", "1.2.3"] + , ["v0.0.0", "0.0.0foo"] + , ["v0.0.1", "0.0.0"] + , ["v1.0.0", "0.9.9"] + , ["v0.10.0", "0.9.0"] + , ["v0.99.0", "0.10.0"] + , ["v2.0.0", "1.2.3"] + , ["0.0.0", "v0.0.0foo"] + , ["0.0.1", "v0.0.0"] + , ["1.0.0", "v0.9.9"] + , ["0.10.0", "v0.9.0"] + , ["0.99.0", "v0.10.0"] + , ["2.0.0", "v1.2.3"] + , ["1.2.3", "1.2.3-asdf"] + , ["1.2.3-4", "1.2.3"] + , ["1.2.3-4-foo", "1.2.3"] + , ["1.2.3-5", "1.2.3-5-foo"] + , ["1.2.3-5", "1.2.3-4"] + , ["1.2.3-5-foo", "1.2.3-5-Foo"] + ].forEach(function (v) { + var v0 = v[0] + , v1 = v[1] + t.ok(gt(v0, v1), "gt('"+v0+"', '"+v1+"')") + t.ok(lt(v1, v0), "lt('"+v1+"', '"+v0+"')") + t.ok(!gt(v1, v0), "!gt('"+v1+"', '"+v0+"')") + t.ok(!lt(v0, v1), "!lt('"+v0+"', '"+v1+"')") + t.ok(eq(v0, v0), "eq('"+v0+"', '"+v0+"')") + t.ok(eq(v1, v1), "eq('"+v1+"', '"+v1+"')") + t.ok(neq(v0, v1), "neq('"+v0+"', '"+v1+"')") + t.ok(cmp(v1, "==", v1), "cmp('"+v1+"' == '"+v1+"')") + t.ok(cmp(v0, ">=", v1), "cmp('"+v0+"' >= '"+v1+"')") + t.ok(cmp(v1, "<=", v0), "cmp('"+v1+"' <= '"+v0+"')") + t.ok(cmp(v0, "!=", v1), "cmp('"+v0+"' != '"+v1+"')") + }) + t.end() +}) + +test("\nequality tests", function (t) { +; [ ["1.2.3", "v1.2.3"] + , ["1.2.3", "=1.2.3"] + , ["1.2.3", "v 1.2.3"] + , ["1.2.3", "= 1.2.3"] + , ["1.2.3", " v1.2.3"] + , ["1.2.3", " =1.2.3"] + , ["1.2.3", " v 1.2.3"] + , ["1.2.3", " = 1.2.3"] + , ["1.2.3-0", "v1.2.3-0"] + , ["1.2.3-0", "=1.2.3-0"] + , ["1.2.3-0", "v 1.2.3-0"] + , ["1.2.3-0", "= 1.2.3-0"] + , ["1.2.3-0", " v1.2.3-0"] + , ["1.2.3-0", " =1.2.3-0"] + , ["1.2.3-0", " v 1.2.3-0"] + , ["1.2.3-0", " = 1.2.3-0"] + , ["1.2.3-01", "v1.2.3-1"] + , ["1.2.3-01", "=1.2.3-1"] + , ["1.2.3-01", "v 1.2.3-1"] + , ["1.2.3-01", "= 1.2.3-1"] + , ["1.2.3-01", " v1.2.3-1"] + , ["1.2.3-01", " =1.2.3-1"] + , ["1.2.3-01", " v 1.2.3-1"] + , ["1.2.3-01", " = 1.2.3-1"] + , ["1.2.3beta", "v1.2.3beta"] + , ["1.2.3beta", "=1.2.3beta"] + , ["1.2.3beta", "v 1.2.3beta"] + , ["1.2.3beta", "= 1.2.3beta"] + , ["1.2.3beta", " v1.2.3beta"] + , ["1.2.3beta", " =1.2.3beta"] + , ["1.2.3beta", " v 1.2.3beta"] + , ["1.2.3beta", " = 1.2.3beta"] + ].forEach(function (v) { + var v0 = v[0] + , v1 = v[1] + t.ok(eq(v0, v1), "eq('"+v0+"', '"+v1+"')") + t.ok(!neq(v0, v1), "!neq('"+v0+"', '"+v1+"')") + t.ok(cmp(v0, "==", v1), "cmp("+v0+"=="+v1+")") + t.ok(!cmp(v0, "!=", v1), "!cmp("+v0+"!="+v1+")") + t.ok(!cmp(v0, "===", v1), "!cmp("+v0+"==="+v1+")") + t.ok(cmp(v0, "!==", v1), "cmp("+v0+"!=="+v1+")") + t.ok(!gt(v0, v1), "!gt('"+v0+"', '"+v1+"')") + t.ok(gte(v0, v1), "gte('"+v0+"', '"+v1+"')") + t.ok(!lt(v0, v1), "!lt('"+v0+"', '"+v1+"')") + t.ok(lte(v0, v1), "lte('"+v0+"', '"+v1+"')") + }) + t.end() +}) + + +test("\nrange tests", function (t) { +; [ ["1.0.0 - 2.0.0", "1.2.3"] + , ["1.0.0", "1.0.0"] + , [">=*", "0.2.4"] + , ["", "1.0.0"] + , ["*", "1.2.3"] + , ["*", "v1.2.3-foo"] + , [">=1.0.0", "1.0.0"] + , [">=1.0.0", "1.0.1"] + , [">=1.0.0", "1.1.0"] + , [">1.0.0", "1.0.1"] + , [">1.0.0", "1.1.0"] + , ["<=2.0.0", "2.0.0"] + , ["<=2.0.0", "1.9999.9999"] + , ["<=2.0.0", "0.2.9"] + , ["<2.0.0", "1.9999.9999"] + , ["<2.0.0", "0.2.9"] + , [">= 1.0.0", "1.0.0"] + , [">= 1.0.0", "1.0.1"] + , [">= 1.0.0", "1.1.0"] + , ["> 1.0.0", "1.0.1"] + , ["> 1.0.0", "1.1.0"] + , ["<= 2.0.0", "2.0.0"] + , ["<= 2.0.0", "1.9999.9999"] + , ["<= 2.0.0", "0.2.9"] + , ["< 2.0.0", "1.9999.9999"] + , ["<\t2.0.0", "0.2.9"] + , [">=0.1.97", "v0.1.97"] + , [">=0.1.97", "0.1.97"] + , ["0.1.20 || 1.2.4", "1.2.4"] + , [">=0.2.3 || <0.0.1", "0.0.0"] + , [">=0.2.3 || <0.0.1", "0.2.3"] + , [">=0.2.3 || <0.0.1", "0.2.4"] + , ["||", "1.3.4"] + , ["2.x.x", "2.1.3"] + , ["1.2.x", "1.2.3"] + , ["1.2.x || 2.x", "2.1.3"] + , ["1.2.x || 2.x", "1.2.3"] + , ["x", "1.2.3"] + , ["2.*.*", "2.1.3"] + , ["1.2.*", "1.2.3"] + , ["1.2.* || 2.*", "2.1.3"] + , ["1.2.* || 2.*", "1.2.3"] + , ["*", "1.2.3"] + , ["2", "2.1.2"] + , ["2.3", "2.3.1"] + , ["~2.4", "2.4.0"] // >=2.4.0 <2.5.0 + , ["~2.4", "2.4.5"] + , ["~>3.2.1", "3.2.2"] // >=3.2.1 <3.3.0 + , ["~1", "1.2.3"] // >=1.0.0 <2.0.0 + , ["~>1", "1.2.3"] + , ["~> 1", "1.2.3"] + , ["~1.0", "1.0.2"] // >=1.0.0 <1.1.0 + , ["~ 1.0", "1.0.2"] + , [">=1", "1.0.0"] + , [">= 1", "1.0.0"] + , ["<1.2", "1.1.1"] + , ["< 1.2", "1.1.1"] + , ["1", "1.0.0beta"] + , ["~v0.5.4-pre", "0.5.5"] + , ["~v0.5.4-pre", "0.5.4"] + ].forEach(function (v) { + t.ok(satisfies(v[1], v[0]), v[0]+" satisfied by "+v[1]) + }) + t.end() +}) + +test("\nnegative range tests", function (t) { +; [ ["1.0.0 - 2.0.0", "2.2.3"] + , ["1.0.0", "1.0.1"] + , [">=1.0.0", "0.0.0"] + , [">=1.0.0", "0.0.1"] + , [">=1.0.0", "0.1.0"] + , [">1.0.0", "0.0.1"] + , [">1.0.0", "0.1.0"] + , ["<=2.0.0", "3.0.0"] + , ["<=2.0.0", "2.9999.9999"] + , ["<=2.0.0", "2.2.9"] + , ["<2.0.0", "2.9999.9999"] + , ["<2.0.0", "2.2.9"] + , [">=0.1.97", "v0.1.93"] + , [">=0.1.97", "0.1.93"] + , ["0.1.20 || 1.2.4", "1.2.3"] + , [">=0.2.3 || <0.0.1", "0.0.3"] + , [">=0.2.3 || <0.0.1", "0.2.2"] + , ["2.x.x", "1.1.3"] + , ["2.x.x", "3.1.3"] + , ["1.2.x", "1.3.3"] + , ["1.2.x || 2.x", "3.1.3"] + , ["1.2.x || 2.x", "1.1.3"] + , ["2.*.*", "1.1.3"] + , ["2.*.*", "3.1.3"] + , ["1.2.*", "1.3.3"] + , ["1.2.* || 2.*", "3.1.3"] + , ["1.2.* || 2.*", "1.1.3"] + , ["2", "1.1.2"] + , ["2.3", "2.4.1"] + , ["~2.4", "2.5.0"] // >=2.4.0 <2.5.0 + , ["~2.4", "2.3.9"] + , ["~>3.2.1", "3.3.2"] // >=3.2.1 <3.3.0 + , ["~>3.2.1", "3.2.0"] // >=3.2.1 <3.3.0 + , ["~1", "0.2.3"] // >=1.0.0 <2.0.0 + , ["~>1", "2.2.3"] + , ["~1.0", "1.1.0"] // >=1.0.0 <1.1.0 + , ["<1", "1.0.0"] + , [">=1.2", "1.1.1"] + , ["1", "2.0.0beta"] + , ["~v0.5.4-beta", "0.5.4-alpha"] + , ["<1", "1.0.0beta"] + , ["< 1", "1.0.0beta"] + ].forEach(function (v) { + t.ok(!satisfies(v[1], v[0]), v[0]+" not satisfied by "+v[1]) + }) + t.end() +}) + +test("\nincrement versions test", function (t) { +; [ [ "1.2.3", "major", "2.0.0" ] + , [ "1.2.3", "minor", "1.3.0" ] + , [ "1.2.3", "patch", "1.2.4" ] + , [ "1.2.3", "build", "1.2.3-1" ] + , [ "1.2.3-4", "build", "1.2.3-5" ] + , [ "1.2.3tag", "major", "2.0.0" ] + , [ "1.2.3-tag", "major", "2.0.0" ] + , [ "1.2.3tag", "build", "1.2.3-1" ] + , [ "1.2.3-tag", "build", "1.2.3-1" ] + , [ "1.2.3-4-tag", "build", "1.2.3-5" ] + , [ "1.2.3-4tag", "build", "1.2.3-5" ] + , [ "1.2.3", "fake", null ] + , [ "fake", "major", null ] + ].forEach(function (v) { + t.equal(inc(v[0], v[1]), v[2], "inc("+v[0]+", "+v[1]+") === "+v[2]) + }) + + t.end() +}) + +test("\nreplace stars test", function (t) { +; [ [ "", "" ] + , [ "*", "" ] + , [ "> *", "" ] + , [ "<*", "" ] + , [ " >= *", "" ] + , [ "* || 1.2.3", " || 1.2.3" ] + ].forEach(function (v) { + t.equal(replaceStars(v[0]), v[1], "replaceStars("+v[0]+") === "+v[1]) + }) + + t.end() +}) + +test("\nvalid range test", function (t) { +; [ ["1.0.0 - 2.0.0", ">=1.0.0 <=2.0.0"] + , ["1.0.0", "1.0.0"] + , [">=*", ""] + , ["", ""] + , ["*", ""] + , ["*", ""] + , [">=1.0.0", ">=1.0.0"] + , [">1.0.0", ">1.0.0"] + , ["<=2.0.0", "<=2.0.0"] + , ["1", ">=1.0.0- <2.0.0-"] + , ["<=2.0.0", "<=2.0.0"] + , ["<=2.0.0", "<=2.0.0"] + , ["<2.0.0", "<2.0.0"] + , ["<2.0.0", "<2.0.0"] + , [">= 1.0.0", ">=1.0.0"] + , [">= 1.0.0", ">=1.0.0"] + , [">= 1.0.0", ">=1.0.0"] + , ["> 1.0.0", ">1.0.0"] + , ["> 1.0.0", ">1.0.0"] + , ["<= 2.0.0", "<=2.0.0"] + , ["<= 2.0.0", "<=2.0.0"] + , ["<= 2.0.0", "<=2.0.0"] + , ["< 2.0.0", "<2.0.0"] + , ["< 2.0.0", "<2.0.0"] + , [">=0.1.97", ">=0.1.97"] + , [">=0.1.97", ">=0.1.97"] + , ["0.1.20 || 1.2.4", "0.1.20||1.2.4"] + , [">=0.2.3 || <0.0.1", ">=0.2.3||<0.0.1"] + , [">=0.2.3 || <0.0.1", ">=0.2.3||<0.0.1"] + , [">=0.2.3 || <0.0.1", ">=0.2.3||<0.0.1"] + , ["||", "||"] + , ["2.x.x", ">=2.0.0- <3.0.0-"] + , ["1.2.x", ">=1.2.0- <1.3.0-"] + , ["1.2.x || 2.x", ">=1.2.0- <1.3.0-||>=2.0.0- <3.0.0-"] + , ["1.2.x || 2.x", ">=1.2.0- <1.3.0-||>=2.0.0- <3.0.0-"] + , ["x", ""] + , ["2.*.*", null] + , ["1.2.*", null] + , ["1.2.* || 2.*", null] + , ["1.2.* || 2.*", null] + , ["*", ""] + , ["2", ">=2.0.0- <3.0.0-"] + , ["2.3", ">=2.3.0- <2.4.0-"] + , ["~2.4", ">=2.4.0- <2.5.0-"] + , ["~2.4", ">=2.4.0- <2.5.0-"] + , ["~>3.2.1", ">=3.2.1- <3.3.0-"] + , ["~1", ">=1.0.0- <2.0.0-"] + , ["~>1", ">=1.0.0- <2.0.0-"] + , ["~> 1", ">=1.0.0- <2.0.0-"] + , ["~1.0", ">=1.0.0- <1.1.0-"] + , ["~ 1.0", ">=1.0.0- <1.1.0-"] + , ["<1", "<1.0.0-"] + , ["< 1", "<1.0.0-"] + , [">=1", ">=1.0.0-"] + , [">= 1", ">=1.0.0-"] + , ["<1.2", "<1.2.0-"] + , ["< 1.2", "<1.2.0-"] + , ["1", ">=1.0.0- <2.0.0-"] + ].forEach(function (v) { + t.equal(validRange(v[0]), v[1], "validRange("+v[0]+") === "+v[1]) + }) + + t.end() +}) + +test("\ncomparators test", function (t) { +; [ ["1.0.0 - 2.0.0", [[">=1.0.0", "<=2.0.0"]] ] + , ["1.0.0", [["1.0.0"]] ] + , [">=*", [[">=0.0.0-"]] ] + , ["", [[""]]] + , ["*", [[""]] ] + , ["*", [[""]] ] + , [">=1.0.0", [[">=1.0.0"]] ] + , [">=1.0.0", [[">=1.0.0"]] ] + , [">=1.0.0", [[">=1.0.0"]] ] + , [">1.0.0", [[">1.0.0"]] ] + , [">1.0.0", [[">1.0.0"]] ] + , ["<=2.0.0", [["<=2.0.0"]] ] + , ["1", [[">=1.0.0-", "<2.0.0-"]] ] + , ["<=2.0.0", [["<=2.0.0"]] ] + , ["<=2.0.0", [["<=2.0.0"]] ] + , ["<2.0.0", [["<2.0.0"]] ] + , ["<2.0.0", [["<2.0.0"]] ] + , [">= 1.0.0", [[">=1.0.0"]] ] + , [">= 1.0.0", [[">=1.0.0"]] ] + , [">= 1.0.0", [[">=1.0.0"]] ] + , ["> 1.0.0", [[">1.0.0"]] ] + , ["> 1.0.0", [[">1.0.0"]] ] + , ["<= 2.0.0", [["<=2.0.0"]] ] + , ["<= 2.0.0", [["<=2.0.0"]] ] + , ["<= 2.0.0", [["<=2.0.0"]] ] + , ["< 2.0.0", [["<2.0.0"]] ] + , ["<\t2.0.0", [["<2.0.0"]] ] + , [">=0.1.97", [[">=0.1.97"]] ] + , [">=0.1.97", [[">=0.1.97"]] ] + , ["0.1.20 || 1.2.4", [["0.1.20"], ["1.2.4"]] ] + , [">=0.2.3 || <0.0.1", [[">=0.2.3"], ["<0.0.1"]] ] + , [">=0.2.3 || <0.0.1", [[">=0.2.3"], ["<0.0.1"]] ] + , [">=0.2.3 || <0.0.1", [[">=0.2.3"], ["<0.0.1"]] ] + , ["||", [[""], [""]] ] + , ["2.x.x", [[">=2.0.0-", "<3.0.0-"]] ] + , ["1.2.x", [[">=1.2.0-", "<1.3.0-"]] ] + , ["1.2.x || 2.x", [[">=1.2.0-", "<1.3.0-"], [">=2.0.0-", "<3.0.0-"]] ] + , ["1.2.x || 2.x", [[">=1.2.0-", "<1.3.0-"], [">=2.0.0-", "<3.0.0-"]] ] + , ["x", [[""]] ] + , ["2.*.*", [[">=2.0.0-", "<3.0.0-"]] ] + , ["1.2.*", [[">=1.2.0-", "<1.3.0-"]] ] + , ["1.2.* || 2.*", [[">=1.2.0-", "<1.3.0-"], [">=2.0.0-", "<3.0.0-"]] ] + , ["1.2.* || 2.*", [[">=1.2.0-", "<1.3.0-"], [">=2.0.0-", "<3.0.0-"]] ] + , ["*", [[""]] ] + , ["2", [[">=2.0.0-", "<3.0.0-"]] ] + , ["2.3", [[">=2.3.0-", "<2.4.0-"]] ] + , ["~2.4", [[">=2.4.0-", "<2.5.0-"]] ] + , ["~2.4", [[">=2.4.0-", "<2.5.0-"]] ] + , ["~>3.2.1", [[">=3.2.1-", "<3.3.0-"]] ] + , ["~1", [[">=1.0.0-", "<2.0.0-"]] ] + , ["~>1", [[">=1.0.0-", "<2.0.0-"]] ] + , ["~> 1", [[">=1.0.0-", "<2.0.0-"]] ] + , ["~1.0", [[">=1.0.0-", "<1.1.0-"]] ] + , ["~ 1.0", [[">=1.0.0-", "<1.1.0-"]] ] + , ["<1", [["<1.0.0-"]] ] + , ["< 1", [["<1.0.0-"]] ] + , [">=1", [[">=1.0.0-"]] ] + , [">= 1", [[">=1.0.0-"]] ] + , ["<1.2", [["<1.2.0-"]] ] + , ["< 1.2", [["<1.2.0-"]] ] + , ["1", [[">=1.0.0-", "<2.0.0-"]] ] + ].forEach(function (v) { + t.equivalent(toComparators(v[0]), v[1], "toComparators("+v[0]+") === "+JSON.stringify(v[1])) + }) + + t.end() +}) diff --git a/deps/npm/node_modules/slide/.npmignore b/deps/npm/node_modules/slide/.npmignore new file mode 100644 index 0000000000..a136337994 --- /dev/null +++ b/deps/npm/node_modules/slide/.npmignore @@ -0,0 +1 @@ +*.pdf diff --git a/deps/npm/node_modules/slide/LICENSE b/deps/npm/node_modules/slide/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/slide/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/slide/README.md b/deps/npm/node_modules/slide/README.md new file mode 100644 index 0000000000..6e4be2f94e --- /dev/null +++ b/deps/npm/node_modules/slide/README.md @@ -0,0 +1,32 @@ +# Slide - a tiny flow control library + +Callbacks are simple and easy if you keep the pattern consistent. + +Check out the [slide +presentation](http://github.com/isaacs/slide-flow-control/raw/master/nodejs-controlling-flow.pdf), +or the [blog post](http://howto.no.de/flow-control-in-npm). + +You'll laugh when you see how little code is actually in this thing. +It's so not-enterprisey, you won't believe it. It does almost nothing, +but it's super handy. + +I use this util in [a real world program](http://npmjs.org/). + +You should use it as an example of how to write your own flow control +utilities. You'll never fully appreciate a flow control lib that you +didn't write yourself. + +## Installation + +Just copy the files into your project, and use them that way, or +you can do this: + + npm install slide + +and then: + + var asyncMap = require("slide").asyncMap + , chain = require("slide").chain + // use the power! + +Enjoy! diff --git a/deps/npm/node_modules/slide/index.js b/deps/npm/node_modules/slide/index.js new file mode 100644 index 0000000000..0a9277f6ee --- /dev/null +++ b/deps/npm/node_modules/slide/index.js @@ -0,0 +1 @@ +module.exports=require("./lib/slide") diff --git a/deps/npm/node_modules/slide/lib/async-map-ordered.js b/deps/npm/node_modules/slide/lib/async-map-ordered.js new file mode 100644 index 0000000000..5cca79a82c --- /dev/null +++ b/deps/npm/node_modules/slide/lib/async-map-ordered.js @@ -0,0 +1,65 @@ + +throw new Error("TODO: Not yet implemented.") + +/* +usage: + +Like asyncMap, but only can take a single cb, and guarantees +the order of the results. +*/ + +module.exports = asyncMapOrdered + +function asyncMapOrdered (list, fn, cb_) { + if (typeof cb_ !== "function") throw new Error( + "No callback provided to asyncMapOrdered") + + if (typeof fn !== "function") throw new Error( + "No map function provided to asyncMapOrdered") + + if (list === undefined || list === null) return cb_(null, []) + if (!Array.isArray(list)) list = [list] + if (!list.length) return cb_(null, []) + + var errState = null + , l = list.length + , a = l + , res = [] + , resCount = 0 + , maxArgLen = 0 + + function cb (index) { return function () { + if (errState) return + var er = arguments[0] + var argLen = arguments.length + maxArgLen = Math.max(maxArgLen, argLen) + res[index] = argLen === 1 ? [er] : Array.apply(null, arguments) + + // see if any new things have been added. + if (list.length > l) { + var newList = list.slice(l) + a += (list.length - l) + var oldLen = l + l = list.length + process.nextTick(function () { + newList.forEach(function (ar, i) { fn(ar, cb(i + oldLen)) }) + }) + } + + if (er || --a === 0) { + errState = er + cb_.apply(null, [errState].concat(flip(res, resCount, maxArgLen))) + } + }} + // expect the supplied cb function to be called + // "n" times for each thing in the array. + list.forEach(function (ar) { + steps.forEach(function (fn, i) { fn(ar, cb(i)) }) + }) +} + +function flip (res, resCount, argLen) { + var flat = [] + // res = [[er, x, y], [er, x1, y1], [er, x2, y2, z2]] + // return [[x, x1, x2], [y, y1, y2], [undefined, undefined, z2]] + diff --git a/deps/npm/node_modules/slide/lib/async-map.js b/deps/npm/node_modules/slide/lib/async-map.js new file mode 100644 index 0000000000..1ced158e0e --- /dev/null +++ b/deps/npm/node_modules/slide/lib/async-map.js @@ -0,0 +1,56 @@ + +/* +usage: + +// do something to a list of things +asyncMap(myListOfStuff, function (thing, cb) { doSomething(thing.foo, cb) }, cb) +// do more than one thing to each item +asyncMap(list, fooFn, barFn, cb) + +*/ + +module.exports = asyncMap + +function asyncMap () { + var steps = Array.prototype.slice.call(arguments) + , list = steps.shift() || [] + , cb_ = steps.pop() + if (typeof cb_ !== "function") throw new Error( + "No callback provided to asyncMap") + if (!list) return cb_(null, []) + if (!Array.isArray(list)) list = [list] + var n = steps.length + , data = [] // 2d array + , errState = null + , l = list.length + , a = l * n + if (!a) return cb_(null, []) + function cb (er) { + if (errState) return + var argLen = arguments.length + for (var i = 1; i < argLen; i ++) if (arguments[i] !== undefined) { + data[i - 1] = (data[i - 1] || []).concat(arguments[i]) + } + // see if any new things have been added. + if (list.length > l) { + var newList = list.slice(l) + a += (list.length - l) * n + l = list.length + process.nextTick(function () { + newList.forEach(function (ar) { + steps.forEach(function (fn) { fn(ar, cb) }) + }) + }) + } + + if (er || --a === 0) { + errState = er + cb_.apply(null, [errState].concat(data)) + } + } + // expect the supplied cb function to be called + // "n" times for each thing in the array. + list.forEach(function (ar) { + steps.forEach(function (fn) { fn(ar, cb) }) + }) +} diff --git a/deps/npm/node_modules/slide/lib/bind-actor.js b/deps/npm/node_modules/slide/lib/bind-actor.js new file mode 100644 index 0000000000..6a37072749 --- /dev/null +++ b/deps/npm/node_modules/slide/lib/bind-actor.js @@ -0,0 +1,16 @@ +module.exports = bindActor +function bindActor () { + var args = + Array.prototype.slice.call + (arguments) // jswtf. + , obj = null + , fn + if (typeof args[0] === "object") { + obj = args.shift() + fn = args.shift() + if (typeof fn === "string") + fn = obj[ fn ] + } else fn = args.shift() + return function (cb) { + fn.apply(obj, args.concat(cb)) } +} diff --git a/deps/npm/node_modules/slide/lib/chain.js b/deps/npm/node_modules/slide/lib/chain.js new file mode 100644 index 0000000000..17b3711494 --- /dev/null +++ b/deps/npm/node_modules/slide/lib/chain.js @@ -0,0 +1,20 @@ +module.exports = chain +var bindActor = require("./bind-actor.js") +chain.first = {} ; chain.last = {} +function chain (things, cb) { + var res = [] + ;(function LOOP (i, len) { + if (i >= len) return cb(null,res) + if (Array.isArray(things[i])) + things[i] = bindActor.apply(null, + things[i].map(function(i){ + return (i===chain.first) ? res[0] + : (i===chain.last) + ? res[res.length - 1] : i })) + if (!things[i]) return LOOP(i + 1, len) + things[i](function (er, data) { + if (er) return cb(er, res) + if (data !== undefined) res = res.concat(data) + LOOP(i + 1, len) + }) + })(0, things.length) } diff --git a/deps/npm/node_modules/slide/lib/slide.js b/deps/npm/node_modules/slide/lib/slide.js new file mode 100644 index 0000000000..6e9ec2327a --- /dev/null +++ b/deps/npm/node_modules/slide/lib/slide.js @@ -0,0 +1,3 @@ +exports.asyncMap = require("./async-map") +exports.bindActor = require("./bind-actor") +exports.chain = require("./chain") diff --git a/deps/npm/node_modules/slide/package.json b/deps/npm/node_modules/slide/package.json new file mode 100644 index 0000000000..5cc2689e63 --- /dev/null +++ b/deps/npm/node_modules/slide/package.json @@ -0,0 +1,19 @@ +{ + "name": "slide", + "version": "1.1.3", + "author": "Isaac Z. Schlueter (http://blog.izs.me/)", + "contributors": [ + "S. Sriram (http://www.565labs.com)" + ], + "description": "A flow control lib small enough to fit on in a slide presentation. Derived live at Oak.JS", + "main": "./lib/slide.js", + "dependencies": {}, + "devDependencies": {}, + "engines": { + "node": "*" + }, + "repository": { + "type": "git", + "url": "git://github.com/isaacs/slide-flow-control.git" + } +} diff --git a/deps/npm/node_modules/tar/.gitignore b/deps/npm/node_modules/tar/.gitignore new file mode 100644 index 0000000000..f96c7db109 --- /dev/null +++ b/deps/npm/node_modules/tar/.gitignore @@ -0,0 +1,4 @@ +.*.swp +node_modules +examples/extract/ +test/tmp/ diff --git a/deps/npm/node_modules/tar/README.md b/deps/npm/node_modules/tar/README.md new file mode 100644 index 0000000000..7cfe3bbca5 --- /dev/null +++ b/deps/npm/node_modules/tar/README.md @@ -0,0 +1,50 @@ +# node-tar + +Tar for Node.js. + +## Goals of this project + +1. Be able to parse and reasonably extract the contents of any tar file + created by any program that creates tar files, period. + + At least, this includes every version of: + + * bsdtar + * gnutar + * solaris posix tar + * Joerg Schilling's star ("Schilly tar") + +2. Create tar files that can be extracted by any of the following tar + programs: + + * bsdtar/libarchive version 2.6.2 + * gnutar 1.15 and above + * SunOS Posix tar + * Joerg Schilling's star ("Schilly tar") + +3. 100% test coverage. Speed is important. Correctness is slightly + more important. + +4. Create the kind of tar interface that Node users would want to use. + +5. Satisfy npm's needs for a portable tar implementation with a + JavaScript interface. + +6. No excuses. No complaining. No tolerance for failure. + +## But isn't there already a tar.js? + +Yes, there are a few. This one is going to be better, and it will be +fanatically maintained, because npm will depend on it. + +That's why I need to write it from scratch. Creating and extracting +tarballs is such a large part of what npm does, I simply can't have it +be a black box any longer. + +## Didn't you have something already? Where'd it go? + +It's in the "old" folder. It's not functional. Don't use it. + +It was a useful exploration to learn the issues involved, but like most +software of any reasonable complexity, node-tar won't be useful until +it's been written at least 3 times. diff --git a/deps/npm/node_modules/tar/examples/extracter.js b/deps/npm/node_modules/tar/examples/extracter.js new file mode 100644 index 0000000000..e150abf25d --- /dev/null +++ b/deps/npm/node_modules/tar/examples/extracter.js @@ -0,0 +1,11 @@ +var tar = require("../tar.js") + , fs = require("fs") + +fs.createReadStream(__dirname + "/../test/fixtures/c.tar") + .pipe(tar.Extract({ path: __dirname + "/extract" })) + .on("error", function (er) { + console.error("error here") + }) + .on("end", function () { + console.error("done") + }) diff --git a/deps/npm/node_modules/tar/examples/reader.js b/deps/npm/node_modules/tar/examples/reader.js new file mode 100644 index 0000000000..c2584d398f --- /dev/null +++ b/deps/npm/node_modules/tar/examples/reader.js @@ -0,0 +1,36 @@ +var tar = require("../tar.js") + , fs = require("fs") + +fs.createReadStream(__dirname + "/../test/fixtures/c.tar") + .pipe(tar.Reader()) + .on("extendedHeader", function (e) { + console.error("extended pax header", e.props) + e.on("end", function () { + console.error("extended pax fields:", e.fields) + }) + }) + .on("ignoredEntry", function (e) { + console.error("ignoredEntry?!?", e.props) + }) + .on("longLinkpath", function (e) { + console.error("longLinkpath entry", e.props) + e.on("end", function () { + console.error("value=%j", e.body.toString()) + }) + }) + .on("longPath", function (e) { + console.error("longPath entry", e.props) + e.on("end", function () { + console.error("value=%j", e.body.toString()) + }) + }) + .on("entry", function (e) { + console.error("entry", e.props) + e.on("data", function (c) { + console.error(" >>>" + c.toString().replace(/\n/g, "\\n")) + }) + e.on("end", function () { + console.error(" << 0 + return !this._needDrain +} + +EntryWriter.prototype.end = function (c) { + // console.error(".. ew end") + if (c) this._buffer.push(c) + this._buffer.push(EOF) + this._ended = true + this._process() + this._needDrain = this._buffer.length > 0 +} + +EntryWriter.prototype.pause = function () { + // console.error(".. ew pause") + this._paused = true + this.emit("pause") +} + +EntryWriter.prototype.resume = function () { + // console.error(".. ew resume") + this._paused = false + this.emit("resume") + this._process() +} + +EntryWriter.prototype.add = function (entry) { + // console.error(".. ew add") + if (!this.parent) return this.emit("error", new Error("no parent")) + + // make sure that the _header and such is emitted, and clear out + // the _currentEntry link on the parent. + if (!this._ended) this.end() + + return this.parent.add(entry) +} + +EntryWriter.prototype._header = function () { + // console.error(".. ew header") + if (this._didHeader) return + this._didHeader = true + + var headerBlock = TarHeader.encode(this.props) + + if (this.props.needExtended && !this._meta) { + var me = this + + ExtendedHeaderWriter = ExtendedHeaderWriter || + require("./extended-header-writer.js") + + ExtendedHeaderWriter(this.props) + .on("data", function (c) { + me.emit("data", c) + }) + .end() + } + + // console.error(".. .. ew headerBlock emitting") + this.emit("data", headerBlock) + this.emit("header") +} + +EntryWriter.prototype._process = function () { + // console.error(".. .. ew process") + if (!this._didHeader && !this._meta) { + this._header() + } + + if (this._paused || this._processing) { + // console.error(".. .. .. paused=%j, processing=%j", this._paused, this._processing) + return + } + + this._processing = true + + var buf = this._buffer + for (var i = 0; i < buf.length; i ++) { + // console.error(".. .. .. i=%d", i) + + var c = buf[i] + + if (c === EOF) this._stream.end() + else this._stream.write(c) + + if (this._paused) { + // console.error(".. .. .. paused mid-emission") + this._processing = false + if (i < buf.length) { + this._needDrain = true + this._buffer = buf.slice(i + 1) + } + return + } + } + + // console.error(".. .. .. emitted") + this._buffer.length = 0 + this._processing = false + + // console.error(".. .. .. emitting drain") + this.emit("drain") +} + +EntryWriter.prototype.destroy = function () {} diff --git a/deps/npm/node_modules/tar/lib/entry.js b/deps/npm/node_modules/tar/lib/entry.js new file mode 100644 index 0000000000..4fc331eb94 --- /dev/null +++ b/deps/npm/node_modules/tar/lib/entry.js @@ -0,0 +1,212 @@ +// A passthrough read/write stream that sets its properties +// based on a header, extendedHeader, and globalHeader +// +// Can be either a file system object of some sort, or +// a pax/ustar metadata entry. + +module.exports = Entry + +var TarHeader = require("./header.js") + , tar = require("../tar") + , assert = require("assert").ok + , Stream = require("stream").Stream + , inherits = require("inherits") + , fstream = require("fstream").Abstract + +function Entry (header, extended, global) { + Stream.call(this) + this.readable = true + this.writable = true + + this._needDrain = false + this._paused = false + this._reading = false + this._ending = false + this._ended = false + this._remaining = 0 + this._queue = [] + this._index = 0 + this._queueLen = 0 + + this._read = this._read.bind(this) + + this.props = {} + this._header = header + this._extended = extended || {} + + // globals can change throughout the course of + // a file parse operation. Freeze it at its current state. + this._global = {} + var me = this + Object.keys(global || {}).forEach(function (g) { + me._global[g] = global[g] + }) + + this._setProps() +} + +inherits(Entry, Stream, +{ write: function (c) { + if (this._ending) this.error("write() after end()", null, true) + if (this._remaining === 0) { + this.error("invalid bytes past eof") + } + + // often we'll get a bunch of \0 at the end of the last write, + // since chunks will always be 512 bytes when reading a tarball. + if (c.length > this._remaining) { + c = c.slice(0, this._remaining) + } + this._remaining -= c.length + + // put it on the stack. + var ql = this._queueLen + this._queue.push(c) + this._queueLen ++ + + this._read() + + // either paused, or buffered + if (this._paused || ql > 0) { + this._needDrain = true + return false + } + + return true + } + +, end: function (c) { + if (c) this.write(c) + this._ending = true + this._read() + } + +, pause: function () { + this._paused = true + this.emit("pause") + } + +, resume: function () { + // console.error(" Tar Entry resume", this.path) + this.emit("resume") + this._paused = false + this._read() + return this._queueLen - this._index > 1 + } + + // This is bound to the instance +, _read: function () { + // console.error(" Tar Entry _read", this.path) + + if (this._paused || this._reading || this._ended) return + + // set this flag so that event handlers don't inadvertently + // get multiple _read() calls running. + this._reading = true + + // have any data to emit? + if (this._index < this._queueLen) { + var chunk = this._queue[this._index ++] + this.emit("data", chunk) + } + + // check if we're drained + if (this._index >= this._queueLen) { + this._queue.length = this._queueLen = this._index = 0 + if (this._needDrain) { + this._needDrain = false + this.emit("drain") + } + if (this._ending) { + this._ended = true + this.emit("end") + } + } + + // if the queue gets too big, then pluck off whatever we can. + // this should be fairly rare. + var mql = this._maxQueueLen + if (this._queueLen > mql && this._index > 0) { + mql = Math.min(this._index, mql) + this._index -= mql + this._queueLen -= mql + this._queue = this._queue.slice(mql) + } + + this._reading = false + } + +, _setProps: function () { + // props = extended->global->header->{} + var header = this._header + , extended = this._extended + , global = this._global + , props = this.props + + // first get the values from the normal header. + var fields = tar.fields + for (var f = 0; fields[f] !== null; f ++) { + var field = fields[f] + , val = header[field] + if (typeof val !== "undefined") props[field] = val + } + + // next, the global header for this file. + // numeric values, etc, will have already been parsed. + ;[global, extended].forEach(function (p) { + Object.keys(p).forEach(function (f) { + if (typeof p[f] !== "undefined") props[f] = p[f] + }) + }) + + // no nulls allowed in path or linkpath + ;["path", "linkpath"].forEach(function (p) { + if (props.hasOwnProperty(p)) { + props[p] = props[p].split("\0")[0] + } + }) + + + // set date fields to be a proper date + ;["mtime", "ctime", "atime"].forEach(function (p) { + if (props.hasOwnProperty(p)) { + props[p] = new Date(props[p] * 1000) + } + }) + + // set the type so that we know what kind of file to create + var type + switch (tar.types[props.type]) { + case "OldFile": + case "ContiguousFile": + type = "File" + break + + case "GNUDumpDir": + type = "Directory" + break + + case undefined: + type = "Unknown" + break + + case "Link": + case "SymbolicLink": + case "CharacterDevice": + case "BlockDevice": + case "Directory": + case "FIFO": + default: + type = tar.types[props.type] + } + + this.type = type + this.path = props.path + this.size = props.size + + // size is special, since it signals when the file needs to end. + this._remaining = props.size + } +, warn: fstream.warn +, error: fstream.error +}) diff --git a/deps/npm/node_modules/tar/lib/extended-header-writer.js b/deps/npm/node_modules/tar/lib/extended-header-writer.js new file mode 100644 index 0000000000..92e09681a8 --- /dev/null +++ b/deps/npm/node_modules/tar/lib/extended-header-writer.js @@ -0,0 +1,165 @@ + +module.exports = ExtendedHeaderWriter + +var inherits = require("inherits") + , EntryWriter = require("./entry-writer.js") + +inherits(ExtendedHeaderWriter, EntryWriter) + +var tar = require("../tar.js") + , path = require("path") + , inherits = require("inherits") + , TarHeader = require("./header.js") + +// props is the props of the thing we need to write an +// extended header for. +// Don't be shy with it. Just encode everything. +function ExtendedHeaderWriter (props) { + // console.error(">> ehw ctor") + var me = this + + if (!(me instanceof ExtendedHeaderWriter)) { + return new ExtendedHeaderWriter(props) + } + + me.fields = props + + var p = + { path : ("PaxHeader" + path.join("/", props.path || "")) + .replace(/\\/g, "/").substr(0, 100) + , mode : props.mode || 0666 + , uid : props.uid || 0 + , gid : props.gid || 0 + , size : 0 // will be set later + , mtime : props.mtime || Date.now() / 1000 + , type : "x" + , linkpath : "" + , ustar : "ustar\0" + , ustarver : "00" + , uname : props.uname || "" + , gname : props.gname || "" + , devmaj : props.devmaj || 0 + , devmin : props.devmin || 0 + } + + + EntryWriter.call(me, p) + // console.error(">> ehw props", me.props) + me.props = p + + me._meta = true +} + +ExtendedHeaderWriter.prototype.end = function () { + // console.error(">> ehw end") + var me = this + + if (me._ended) return + me._ended = true + + me._encodeFields() + me._stream.write(TarHeader.encode(me.props)) + me.body.forEach(function (l) { + me._stream.write(l) + }) + me._ready = true + + // console.error(">> ehw _process calling end()", me.props) + this._stream.end() +} + +ExtendedHeaderWriter.prototype._encodeFields = function () { + // console.error(">> ehw _encodeFields") + this.body = [] + if (this.fields.prefix) { + this.fields.path = this.fields.prefix + "/" + this.fields.path + this.fields.prefix = "" + } + encodeFields(this.fields, "", this.body) + var me = this + this.body.forEach(function (l) { + me.props.size += l.length + }) +} + +function encodeFields (fields, prefix, body) { + // console.error(">> >> ehw encodeFields") + // "%d %s=%s\n", , , + // The length is a decimal number, and includes itself and the \n + // Numeric values are decimal strings. + + Object.keys(fields).forEach(function (k) { + var val = fields[k] + , numeric = tar.numeric[k] + + if (prefix) k = prefix + "." + k + + // already including NODETAR.type, don't need File=true also + if (k === fields.type && val === true) return + + switch (k) { + // don't include anything that's always handled just fine + // in the normal header, or only meaningful in the context + // of nodetar + case "mode": + case "cksum": + case "ustar": + case "ustarver": + case "prefix": + case "basename": + case "dirname": + case "needExtended": + case "block": + case "filter": + return + + case "rdev": + if (val === 0) return + break + + case "nlink": + case "dev": // Truly a hero among men, Creator of Star! + case "ino": // Speak his name with reverent awe! It is: + k = "SCHILY." + k + break + + default: break + } + + if (val && typeof val === "object" && + !Buffer.isBuffer(val)) encodeFields(val, k, body) + else if (val === null || val === undefined) return + else body.push.apply(body, encodeField(k, val)) + }) + + return body +} + +function encodeField (k, v) { + // lowercase keys must be valid, otherwise prefix with + // "NODETAR." + if (k.charAt(0) === k.charAt(0).toLowerCase()) { + var m = k.split(".")[0] + if (!tar.knownExtended[m]) k = "NODETAR." + k + } + + if (typeof val === "number") val = val.toString(10) + + var s = new Buffer(" " + k + "=" + v + "\n") + , digits = Math.floor(Math.log(s.length) / Math.log(10)) + 1 + + // if adding that many digits will make it go over that length, + // then add one to it. For example, if the string is: + // " foo=bar\n" + // then that's 9 characters. With the "9", that bumps the length + // up to 10. However, this is invalid: + // "10 foo=bar\n" + // but, since that's actually 11 characters, since 10 adds another + // character to the length, and the length includes the number + // itself. In that case, just bump it up by 1. + if (s.length > Math.pow(10, digits) - digits) digits ++ + + var len = digits + s.length + + return [new Buffer("" + len), s] +} diff --git a/deps/npm/node_modules/tar/lib/extended-header.js b/deps/npm/node_modules/tar/lib/extended-header.js new file mode 100644 index 0000000000..4346d6c59b --- /dev/null +++ b/deps/npm/node_modules/tar/lib/extended-header.js @@ -0,0 +1,139 @@ +// An Entry consisting of: +// +// "%d %s=%s\n", , , +// +// The length is a decimal number, and includes itself and the \n +// \0 does not terminate anything. Only the length terminates the string. +// Numeric values are decimal strings. + +module.exports = ExtendedHeader + +var Entry = require("./entry.js") + , inherits = require("inherits") + , tar = require("../tar.js") + , numeric = tar.numeric + , keyTrans = { "SCHILY.dev": "dev" + , "SCHILY.ino": "ino" + , "SCHILY.nlink": "nlink" } + +function ExtendedHeader () { + Entry.apply(this, arguments) + this.on("data", this._parse) + this.fields = {} + this._position = 0 + this._fieldPos = 0 + this._state = SIZE + this._sizeBuf = [] + this._keyBuf = [] + this._valBuf = [] + this._size = -1 + this._key = "" +} + +inherits(ExtendedHeader, Entry, { _parse: parse }) + +var s = 0 + , states = ExtendedHeader.states = {} + , SIZE = states.SIZE = s++ + , KEY = states.KEY = s++ + , VAL = states.VAL = s++ + , ERR = states.ERR = s++ + +Object.keys(states).forEach(function (s) { + states[states[s]] = states[s] +}) + +states[s] = null + +// char code values for comparison +var _0 = "0".charCodeAt(0) + , _9 = "9".charCodeAt(0) + , point = ".".charCodeAt(0) + , a = "a".charCodeAt(0) + , Z = "Z".charCodeAt(0) + , a = "a".charCodeAt(0) + , z = "z".charCodeAt(0) + , space = " ".charCodeAt(0) + , eq = "=".charCodeAt(0) + , cr = "\n".charCodeAt(0) + +function parse (c) { + if (this._state === ERR) return + + for ( var i = 0, l = c.length + ; i < l + ; this._position++, this._fieldPos++, i++) { + // console.error("top of loop, size="+this._size) + + var b = c[i] + + if (this._size >= 0 && this._fieldPos > this._size) { + error(this, "field exceeds length="+this._size) + return + } + + switch (this._state) { + case ERR: return + + case SIZE: + // console.error("parsing size, b=%d, rest=%j", b, c.slice(i).toString()) + if (b === space) { + this._state = KEY + // this._fieldPos = this._sizeBuf.length + this._size = parseInt(new Buffer(this._sizeBuf).toString(), 10) + this._sizeBuf.length = 0 + continue + } + if (b < _0 || b > _9) { + error(this, "expected [" + _0 + ".." + _9 + "], got " + b) + return + } + this._sizeBuf.push(b) + continue + + case KEY: + // can be any char except =, not > size. + if (b === eq) { + this._state = VAL + this._key = new Buffer(this._keyBuf).toString() + if (keyTrans[this._key]) this._key = keyTrans[this._key] + this._keyBuf.length = 0 + continue + } + this._keyBuf.push(b) + continue + + case VAL: + // field must end with cr + if (this._fieldPos === this._size - 1) { + // console.error("finished with "+this._key) + if (b !== cr) { + error(this, "expected \\n at end of field") + return + } + var val = new Buffer(this._valBuf).toString() + if (numeric[this._key]) { + val = parseFloat(val) + } + this.fields[this._key] = val + + this._valBuf.length = 0 + this._state = SIZE + this._size = -1 + this._fieldPos = -1 + continue + } + this._valBuf.push(b) + continue + } + } +} + +function error (me, msg) { + msg = "invalid header: " + msg + + "\nposition=" + me._position + + "\nfield position=" + me._fieldPos + + me.error(msg) + me.state = ERR +} diff --git a/deps/npm/node_modules/tar/lib/extract.js b/deps/npm/node_modules/tar/lib/extract.js new file mode 100644 index 0000000000..e45974c723 --- /dev/null +++ b/deps/npm/node_modules/tar/lib/extract.js @@ -0,0 +1,64 @@ +// give it a tarball and a path, and it'll dump the contents + +module.exports = Extract + +var tar = require("../tar.js") + , fstream = require("fstream") + , inherits = require("inherits") + , path = require("path") + +function Extract (opts) { + if (!(this instanceof Extract)) return new Extract(opts) + tar.Parse.apply(this) + + // have to dump into a directory + opts.type = "Directory" + opts.Directory = true + + if (typeof opts !== "object") { + opts = { path: opts } + } + + // better to drop in cwd? seems more standard. + opts.path = opts.path || path.resolve("node-tar-extract") + opts.type = "Directory" + opts.Directory = true + + this._fst = fstream.Writer(opts) + + this.pause() + var me = this + + // Hardlinks in tarballs are relative to the root + // of the tarball. So, they need to be resolved against + // the target directory in order to be created properly. + me.on("entry", function (entry) { + if (entry.type !== "Link") return + entry.linkpath = entry.props.linkpath = + path.join(opts.path, path.join("/", entry.props.linkpath)) + }) + + this._fst.on("ready", function () { + me.pipe(me._fst, { end: false }) + me.resume() + }) + + // this._fst.on("end", function () { + // console.error("\nEEEE Extract End", me._fst.path) + // }) + + this._fst.on("close", function () { + // console.error("\nEEEE Extract End", me._fst.path) + me.emit("end") + me.emit("close") + }) +} + +inherits(Extract, tar.Parse) + +Extract.prototype._streamEnd = function () { + var me = this + if (!me._ended) me.error("unexpected eof") + me._fst.end() + // my .end() is coming later. +} diff --git a/deps/npm/node_modules/tar/lib/global-header-writer.js b/deps/npm/node_modules/tar/lib/global-header-writer.js new file mode 100644 index 0000000000..0bfc7b80aa --- /dev/null +++ b/deps/npm/node_modules/tar/lib/global-header-writer.js @@ -0,0 +1,14 @@ +module.exports = GlobalHeaderWriter + +var ExtendedHeaderWriter = require("./extended-header-writer.js") + , inherits = require("inherits") + +inherits(GlobalHeaderWriter, ExtendedHeaderWriter) + +function GlobalHeaderWriter (props) { + if (!(this instanceof GlobalHeaderWriter)) { + return new GlobalHeaderWriter(props) + } + ExtendedHeaderWriter.call(this, props) + this.props.type = "g" +} diff --git a/deps/npm/node_modules/tar/lib/header.js b/deps/npm/node_modules/tar/lib/header.js new file mode 100644 index 0000000000..05b237c0c7 --- /dev/null +++ b/deps/npm/node_modules/tar/lib/header.js @@ -0,0 +1,385 @@ +// parse a 512-byte header block to a data object, or vice-versa +// If the data won't fit nicely in a simple header, then generate +// the appropriate extended header file, and return that. + +module.exports = TarHeader + +var tar = require("../tar.js") + , fields = tar.fields + , fieldOffs = tar.fieldOffs + , fieldEnds = tar.fieldEnds + , fieldSize = tar.fieldSize + , numeric = tar.numeric + , assert = require("assert").ok + , space = " ".charCodeAt(0) + , slash = "/".charCodeAt(0) + , bslash = process.platform === "win32" ? "\\".charCodeAt(0) : null + +function TarHeader (block) { + if (!(this instanceof TarHeader)) return new TarHeader(block) + if (block) this.decode(block) +} + +TarHeader.prototype = + { decode : decode + , encode: encode + , calcSum: calcSum + , checkSum: checkSum + } + +TarHeader.parseNumeric = parseNumeric +TarHeader.encode = encode +TarHeader.decode = decode + +// note that this will only do the normal ustar header, not any kind +// of extended posix header file. If something doesn't fit comfortably, +// then it will set obj.needExtended = true, and set the block to +// the closest approximation. +function encode (obj) { + if (!obj && !(this instanceof TarHeader)) throw new Error( + "encode must be called on a TarHeader, or supplied an object") + + obj = obj || this + var block = obj.block = new Buffer(512) + + // if the object has a "prefix", then that's actually an extension of + // the path field. + if (obj.prefix) { + // console.error("%% header encoding, got a prefix", obj.prefix) + obj.path = obj.prefix + "/" + obj.path + // console.error("%% header encoding, prefixed path", obj.path) + obj.prefix = "" + } + + obj.needExtended = false + + if (obj.mode) { + if (typeof obj.mode === "string") obj.mode = parseInt(obj.mode, 8) + obj.mode = obj.mode & 0777 + } + + for (var f = 0; fields[f] !== null; f ++) { + var field = fields[f] + , off = fieldOffs[f] + , end = fieldEnds[f] + , ret + + switch (field) { + case "cksum": + // special, done below, after all the others + break + + case "prefix": + // special, this is an extension of the "path" field. + // console.error("%% header encoding, skip prefix later") + break + + case "type": + // convert from long name to a single char. + var type = obj.type || "0" + if (type.length > 1) { + type = tar.types[obj.type] + if (!type) type = "0" + } + writeText(block, off, end, type) + break + + case "path": + // uses the "prefix" field if > 100 bytes, but <= 255 + var pathLen = Buffer.byteLength(obj.path) + , pathFSize = fieldSize[fields.path] + , prefFSize = fieldSize[fields.prefix] + + // paths between 100 and 255 should use the prefix field. + // longer than 255 + if (pathLen > pathFSize && + pathLen <= pathFSize + prefFSize) { + // need to find a slash somewhere in the middle so that + // path and prefix both fit in their respective fields + var searchStart = pathLen - 1 - pathFSize + , searchEnd = prefFSize + , found = false + , pathBuf = new Buffer(obj.path) + + for ( var s = searchStart + ; (s <= searchEnd) + ; s ++ ) { + if (pathBuf[s] === slash || pathBuf[s] === bslash) { + found = s + break + } + } + + if (found !== false) { + prefix = pathBuf.slice(0, found).toString("utf8") + path = pathBuf.slice(found + 1).toString("utf8") + + ret = writeText(block, off, end, path) + off = fieldOffs[fields.prefix] + end = fieldEnds[fields.prefix] + // console.error("%% header writing prefix", off, end, prefix) + ret = writeText(block, off, end, prefix) || ret + break + } + } + + // paths less than 100 chars don't need a prefix + // and paths longer than 255 need an extended header and will fail + // on old implementations no matter what we do here. + // Null out the prefix, and fallthrough to default. + // console.error("%% header writing no prefix") + var poff = fieldOffs[fields.prefix] + , pend = fieldEnds[fields.prefix] + writeText(block, poff, pend, "") + // fallthrough + + // all other fields are numeric or text + default: + ret = numeric[field] + ? writeNumeric(block, off, end, obj[field]) + : writeText(block, off, end, obj[field] || "") + break + } + obj.needExtended = obj.needExtended || ret + } + + var off = fieldOffs[fields.cksum] + , end = fieldEnds[fields.cksum] + + writeNumeric(block, off, end, calcSum.call(this, block)) + + return block +} + +// if it's a negative number, or greater than will fit, +// then use write256. +var MAXNUM = { 12: 077777777777 + , 11: 07777777777 + , 8 : 07777777 + , 7 : 0777777 } +function writeNumeric (block, off, end, num) { + var writeLen = end - off + , maxNum = MAXNUM[writeLen] || 0 + + num = num || 0 + // console.error(" numeric", num) + + if (num instanceof Date || + Object.prototype.toString.call(num) === "[object Date]") { + num = num.getTime() / 1000 + } + + if (num > maxNum || num < 0) { + write256(block, off, end, num) + // need an extended header if negative or too big. + return true + } + + // god, tar is so annoying + // if the string is small enough, you should put a space + // between the octal string and the \0, but if it doesn't + // fit, then don't. + var numStr = Math.floor(num).toString(8) + if (num < MAXNUM[writeLen - 1]) numStr += " " + + // pad with "0" chars + if (numStr.length < writeLen) { + numStr = (new Array(writeLen - numStr.length).join("0")) + numStr + } + + if (numStr.length !== writeLen - 1) { + throw new Error("invalid length: " + JSON.stringify(numStr) + "\n" + + "expected: "+writeLen) + } + block.write(numStr, off, writeLen, "utf8") + block[end - 1] = 0 +} + +function write256 (block, off, end, num) { + var buf = block.slice(off, end) + var positive = num >= 0 + buf[0] = positive ? 0x80 : 0xFF + + // get the number as a base-256 tuple + if (!positive) num *= -1 + var tuple = [] + do { + var n = num % 256 + tuple.push(n) + num = (num - n) / 256 + } while (num) + + var bytes = tuple.length + + var fill = buf.length - bytes + for (var i = 1; i < fill; i ++) { + buf[i] = positive ? 0 : 0xFF + } + + // tuple is a base256 number, with [0] as the *least* significant byte + // if it's negative, then we need to flip all the bits once we hit the + // first non-zero bit. The 2's-complement is (0x100 - n), and the 1's- + // complement is (0xFF - n). + var zero = true + for (i = bytes; i > 0; i --) { + var byte = tuple[bytes - i] + if (positive) buf[fill + i] = byte + else if (zero && byte === 0) buf[fill + i] = 0 + else if (zero) { + zero = false + buf[fill + i] = 0x100 - byte + } else buf[fill + i] = 0xFF - byte + } +} + +function writeText (block, off, end, str) { + // strings are written as utf8, then padded with \0 + var strLen = Buffer.byteLength(str) + , writeLen = Math.min(strLen, end - off) + // non-ascii fields need extended headers + // long fields get truncated + , needExtended = strLen !== str.length || strLen > writeLen + + // write the string, and null-pad + if (writeLen > 0) block.write(str, off, writeLen, "utf8") + for (var i = off + writeLen; i < end; i ++) block[i] = 0 + + return needExtended +} + +function calcSum (block) { + block = block || this.block + assert(Buffer.isBuffer(block) && block.length === 512) + + if (!block) throw new Error("Need block to checksum") + + // now figure out what it would be if the cksum was " " + var sum = 0 + , start = fieldOffs[fields.cksum] + , end = fieldEnds[fields.cksum] + + for (var i = 0; i < fieldOffs[fields.cksum]; i ++) { + sum += block[i] + } + + for (var i = start; i < end; i ++) { + sum += space + } + + for (var i = end; i < 512; i ++) { + sum += block[i] + } + + return sum +} + + +function checkSum (block) { + var sum = calcSum.call(this, block) + block = block || this.block + + var cksum = block.slice(fieldOffs[fields.cksum], fieldEnds[fields.cksum]) + cksum = parseNumeric(cksum) + + return cksum === sum +} + +function decode (block) { + block = block || this.block + assert(Buffer.isBuffer(block) && block.length === 512) + + this.block = block + this.cksumValid = this.checkSum() + + var prefix = null + + // slice off each field. + for (var f = 0; fields[f] !== null; f ++) { + var field = fields[f] + , val = block.slice(fieldOffs[f], fieldEnds[f]) + + switch (field) { + case "ustar": + // if not ustar, then everything after that is just padding. + if (val.toString() !== "ustar\0") { + this.ustar = false + return + } else { + // console.error("ustar:", val, val.toString()) + this.ustar = val.toString() + } + break + + // prefix is special, since it might signal the xstar header + case "prefix": + var atime = parseNumeric(val.slice(131, 131 + 12)) + , ctime = parseNumeric(val.slice(131 + 12, 131 + 12 + 12)) + if ((val[130] === 0 || val[130] === space) && + typeof atime === "number" && + typeof ctime === "number" && + val[131 + 12] === space && + val[131 + 12 + 12] === space) { + this.atime = atime + this.ctime = ctime + val = val.slice(0, 130) + } + prefix = val.toString("utf8").replace(/\0+$/, "") + // console.error("%% header reading prefix", prefix) + break + + // all other fields are null-padding text + // or a number. + default: + if (numeric[field]) { + this[field] = parseNumeric(val) + } else { + this[field] = val.toString("utf8").replace(/\0+$/, "") + } + break + } + } + + // if we got a prefix, then prepend it to the path. + if (prefix) { + this.path = prefix + "/" + this.path + // console.error("%% header got a prefix", this.path) + } +} + +function parse256 (buf) { + // first byte MUST be either 80 or FF + // 80 for positive, FF for 2's comp + var positive + if (buf[0] === 0x80) positive = true + else if (buf[0] === 0xFF) positive = false + else return null + + // build up a base-256 tuple from the least sig to the highest + var zero = false + , tuple = [] + for (var i = buf.length - 1; i > 0; i --) { + var byte = buf[i] + if (positive) tuple.push(byte) + else if (zero && byte === 0) tuple.push(0) + else if (zero) { + zero = false + tuple.push(0x100 - byte) + } else tuple.push(0xFF - byte) + } + + for (var sum = 0, i = 0, l = tuple.length; i < l; i ++) { + sum += tuple[i] * Math.pow(256, i) + } + + return positive ? sum : -1 * sum +} + +function parseNumeric (f) { + if (f[0] & 0x80) return parse256(f) + + var str = f.toString("utf8").split("\0")[0].trim() + , res = parseInt(str, 8) + + return isNaN(res) ? null : res +} + diff --git a/deps/npm/node_modules/tar/lib/pack.js b/deps/npm/node_modules/tar/lib/pack.js new file mode 100644 index 0000000000..411b8b6b0c --- /dev/null +++ b/deps/npm/node_modules/tar/lib/pack.js @@ -0,0 +1,212 @@ +// pipe in an fstream, and it'll make a tarball. +// key-value pair argument is global extended header props. + +module.exports = Pack + +var EntryWriter = require("./entry-writer.js") + , Stream = require("stream").Stream + , path = require("path") + , inherits = require("inherits") + , GlobalHeaderWriter = require("./global-header-writer.js") + , collect = require("fstream").collect + , eof = new Buffer(512) + +for (var i = 0; i < 512; i ++) eof[i] = 0 + +inherits(Pack, Stream) + +function Pack (props) { + // console.error("-- p ctor") + var me = this + if (!(me instanceof Pack)) return new Pack(props) + + me._global = props + + me.readable = true + me.writable = true + me._buffer = [] + // console.error("-- -- set current to null in ctor") + me._currentEntry = null + me._processing = false + + me._pipeRoot = null + me.on("pipe", function (src) { + if (src.root === me._pipeRoot) return + me._pipeRoot = src + src.on("end", function () { + me._pipeRoot = null + }) + me.add(src) + }) +} + +Pack.prototype.addGlobal = function (props) { + // console.error("-- p addGlobal") + if (this._didGlobal) return + this._didGlobal = true + + var me = this + GlobalHeaderWriter(props) + .on("data", function (c) { + me.emit("data", c) + }) + .end() +} + +Pack.prototype.add = function (stream) { + if (this._global && !this._didGlobal) this.addGlobal(this._global) + + if (this._ended) return this.emit("error", new Error("add after end")) + + collect(stream) + this._buffer.push(stream) + this._process() + this._needDrain = this._buffer.length > 0 + return !this._needDrain +} + +Pack.prototype.pause = function () { + this._paused = true + if (this._currentEntry) this._currentEntry.pause() + this.emit("pause") +} + +Pack.prototype.resume = function () { + this._paused = false + if (this._currentEntry) this._currentEntry.resume() + this.emit("resume") + this._process() +} + +Pack.prototype.end = function () { + this._ended = true + this._buffer.push(eof) + this._process() +} + +Pack.prototype._process = function () { + var me = this + if (me._paused || me._processing) { + return + } + + var entry = me._buffer.shift() + + if (!entry) { + if (me._needDrain) { + me.emit("drain") + } + return + } + + if (entry.ready === false) { + // console.error("-- entry is not ready", entry) + me._buffer.unshift(entry) + entry.on("ready", function () { + // console.error("-- -- ready!", entry) + me._process() + }) + return + } + + me._processing = true + + if (entry === eof) { + // need 2 ending null blocks. + me.emit("data", eof) + me.emit("data", eof) + me.emit("end") + me.emit("close") + return + } + + // Change the path to be relative to the root dir that was + // added to the tarball. + // + // XXX This should be more like how -C works, so you can + // explicitly set a root dir, and also explicitly set a pathname + // in the tarball to use. That way we can skip a lot of extra + // work when resolving symlinks for bundled dependencies in npm. + + var root = path.dirname((entry.root || entry).path) + var wprops = {} + + Object.keys(entry.props).forEach(function (k) { + wprops[k] = entry.props[k] + }) + + wprops.path = path.relative(root, entry.path) + + switch (wprops.type) { + case "Directory": + wprops.path += "/" + wprops.size = 0 + break + case "Link": + var lp = path.resolve(path.dirname(entry.path), entry.linkpath) + wprops.linkpath = path.relative(root, lp) + wprops.size = 0 + break + case "SymbolicLink": + var lp = path.resolve(path.dirname(entry.path), entry.linkpath) + wprops.linkpath = path.relative(path.dirname(entry.path), lp) + wprops.size = 0 + break + } + + // console.error("-- new writer", wprops) + // if (!wprops.type) { + // // console.error("-- no type?", entry.constructor.name, entry) + // } + + // console.error("-- -- set current to new writer", wprops.path) + var writer = me._currentEntry = EntryWriter(wprops) + + writer.parent = me + + // writer.on("end", function () { + // // console.error("-- -- writer end", writer.path) + // }) + + writer.on("data", function (c) { + me.emit("data", c) + }) + + writer.on("header", function () { + Buffer.prototype.toJSON = function () { + return this.toString().split(/\0/).join(".") + } + // console.error("-- -- writer header %j", writer.props) + if (writer.props.size === 0) nextEntry() + }) + writer.on("close", nextEntry) + + var ended = false + function nextEntry () { + if (ended) return + ended = true + + // console.error("-- -- writer close", writer.path) + // console.error("-- -- set current to null", wprops.path) + me._currentEntry = null + me._processing = false + me._process() + } + + writer.on("error", function (er) { + // console.error("-- -- writer error", writer.path) + me.emit("error", er) + }) + + // if it's the root, then there's no need to add its entries, + // or data, since they'll be added directly. + if (entry === me._pipeRoot) { + // console.error("-- is the root, don't auto-add") + writer.add = null + } + + entry.pipe(writer) +} + +Pack.prototype.destroy = function () {} +Pack.prototype.write = function () {} diff --git a/deps/npm/node_modules/tar/lib/parse.js b/deps/npm/node_modules/tar/lib/parse.js new file mode 100644 index 0000000000..d9784b5ff7 --- /dev/null +++ b/deps/npm/node_modules/tar/lib/parse.js @@ -0,0 +1,253 @@ + +// A writable stream. +// It emits "entry" events, which provide a readable stream that has +// header info attached. + +module.exports = Parse.create = Parse + +var stream = require("stream") + , Stream = stream.Stream + , BlockStream = require("block-stream") + , tar = require("../tar.js") + , TarHeader = require("./header.js") + , Entry = require("./entry.js") + , BufferEntry = require("./buffer-entry.js") + , ExtendedHeader = require("./extended-header.js") + , assert = require("assert").ok + , inherits = require("inherits") + , fstream = require("fstream") + +// reading a tar is a lot like reading a directory +// However, we're actually not going to run the ctor, +// since it does a stat and various other stuff. +// This inheritance gives us the pause/resume/pipe +// behavior that is desired. +inherits(Parse, fstream.Reader) + +function Parse () { + var me = this + if (!(me instanceof Parse)) return new Parse() + + // doesn't apply fstream.Reader ctor? + // no, becasue we don't want to stat/etc, we just + // want to get the entry/add logic from .pipe() + Stream.apply(me) + + me.writable = true + me.readable = true + me._stream = new BlockStream(512) + + me._stream.on("error", function (e) { + me.emit("error", e) + }) + + me._stream.on("data", function (c) { + me._process(c) + }) + + me._stream.on("end", function () { + me._streamEnd() + }) + + me._stream.on("drain", function () { + me.emit("drain") + }) +} + +// overridden in Extract class, since it needs to +// wait for its DirWriter part to finish before +// emitting "end" +Parse.prototype._streamEnd = function () { + var me = this + if (!me._ended) me.error("unexpected eof") + me.emit("end") +} + +// a tar reader is actually a filter, not just a readable stream. +// So, you should pipe a tarball stream into it, and it needs these +// write/end methods to do that. +Parse.prototype.write = function (c) { + if (this._ended) { + return this.error("write() after end()") + } + return this._stream.write(c) +} + +Parse.prototype.end = function (c) { + this._ended = true + return this._stream.end(c) +} + +// don't need to do anything, since we're just +// proxying the data up from the _stream. +// Just need to override the parent's "Not Implemented" +// error-thrower. +Parse.prototype._read = function () {} + +Parse.prototype._process = function (c) { + assert(c && c.length === 512, "block size should be 512") + + // one of three cases. + // 1. A new header + // 2. A part of a file/extended header + // 3. One of two or more EOF null blocks + + if (this._entry) { + var entry = this._entry + entry.write(c) + if (entry._remaining === 0) { + entry.end() + this._entry = null + } + } else { + // either zeroes or a header + var zero = true + for (var i = 0; i < 512 && zero; i ++) { + zero = c[i] === 0 + } + + // eof is *at least* 2 blocks of nulls, and then the end of the + // file. you can put blocks of nulls between entries anywhere, + // so appending one tarball to another is technically valid. + // ending without the eof null blocks is not allowed, however. + if (zero) { + this._ended = this._eofStarted + this._eofStarted = true + } else { + this._ended = this._eofStarted = false + this._startEntry(c) + } + + } +} + +// take a header chunk, start the right kind of entry. +Parse.prototype._startEntry = function (c) { + var header = new TarHeader(c) + , self = this + , entry + , ev + , EntryType + , onend + , meta = false + + switch (tar.types[header.type]) { + case "File": + case "OldFile": + case "Link": + case "SymbolicLink": + case "CharacterDevice": + case "BlockDevice": + case "Directory": + case "FIFO": + case "ContiguousFile": + case "GNUDumpDir": + // start a file. + // pass in any extended headers + // These ones consumers are typically most interested in. + EntryType = Entry + ev = "entry" + break + + case "GlobalExtendedHeader": + // extended headers that apply to the rest of the tarball + EntryType = ExtendedHeader + onend = function () { + self._global = self._global || {} + Object.keys(entry.fields).forEach(function (k) { + self._global[k] = entry.fields[k] + }) + } + ev = "globalExtendedHeader" + meta = true + break + + case "ExtendedHeader": + case "OldExtendedHeader": + // extended headers that apply to the next entry + EntryType = ExtendedHeader + onend = function () { + self._extended = entry.fields + } + ev = "extendedHeader" + meta = true + break + + case "NextFileHasLongLinkpath": + // set linkpath= in extended header + EntryType = BufferEntry + onend = function () { + self._extended = self._extended || {} + self._extended.linkpath = entry.body + } + ev = "longLinkpath" + meta = true + break + + case "NextFileHasLongPath": + case "OldGnuLongPath": + // set path= in file-extended header + EntryType = BufferEntry + onend = function () { + self._extended = self._extended || {} + self._extended.path = entry.body + } + ev = "longPath" + meta = true + break + + default: + // all the rest we skip, but still set the _entry + // member, so that we can skip over their data appropriately. + // emit an event to say that this is an ignored entry type? + EntryType = Entry + ev = "ignoredEntry" + break + } + + var global, extended + if (meta) { + global = extended = null + } else { + var global = this._global + var extended = this._extended + + // extendedHeader only applies to one entry, so once we start + // an entry, it's over. + this._extended = null + } + entry = new EntryType(header, extended, global) + entry.meta = meta + + // only proxy data events of normal files. + if (!meta) { + entry.on("data", function (c) { + me.emit("data", c) + }) + } + + if (onend) entry.on("end", onend) + + this._entry = entry + var me = this + + entry.on("pause", function () { + me.pause() + }) + + entry.on("resume", function () { + me.resume() + }) + + if (this.listeners("*").length) { + this.emit("*", ev, entry) + } + + this.emit(ev, entry) + + // Zero-byte entry. End immediately. + if (entry.props.size === 0) { + entry.end() + this._entry = null + } +} diff --git a/deps/npm/node_modules/tar/old/README.md b/deps/npm/node_modules/tar/old/README.md new file mode 100644 index 0000000000..aef9844978 --- /dev/null +++ b/deps/npm/node_modules/tar/old/README.md @@ -0,0 +1 @@ +tar for node diff --git a/deps/npm/node_modules/tar/old/doc/example.js b/deps/npm/node_modules/tar/old/doc/example.js new file mode 100644 index 0000000000..d29517e49f --- /dev/null +++ b/deps/npm/node_modules/tar/old/doc/example.js @@ -0,0 +1,24 @@ +// request a tar file, and then write it +require("http").request({...}, function (resp) { + resp.pipe(tar.createParser(function (file) { + if (file.isDirectory()) { + this.pause() + return fs.mkdir(file.name, function (er) { + if (er) return this.emit("error", er) + this.resume() + }) + } else if (file.isSymbolicLink()) { + this.pause() + return fs.symlink(file.link, file.name, function (er) { + if (er) return this.emit("error", er) + this.resume() + }) + } else if (file.isFile()) { + file.pipe(fs.createWriteStream(file.name)) + } + })) + // or maybe just have it do all that internally? + resp.pipe(tar.createParser(function (file) { + this.create("/extract/target/path", file) + })) +}) diff --git a/deps/npm/node_modules/tar/old/generator.js b/deps/npm/node_modules/tar/old/generator.js new file mode 100644 index 0000000000..c2506c4151 --- /dev/null +++ b/deps/npm/node_modules/tar/old/generator.js @@ -0,0 +1,387 @@ +module.exports = Generator +Generator.create = create + +var tar = require("./tar") + , Stream = require("stream").Stream + , Parser = require("./parser") + , fs = require("fs") + +function create (opts) { + return new Generator(opts) +} + +function Generator (opts) { + this.readable = true + this.currentFile = null + + this._paused = false + this._ended = false + this._queue = [] + + this.options = { cwd: process.cwd() } + Object.keys(opts).forEach(function (o) { + this.options[o] = opts[o] + }, this) + if (this.options.cwd.slice(-1) !== "/") { + this.options.cwd += "/" + } + + Stream.apply(this) +} + +Generator.prototype = Object.create(Stream.prototype) + +Generator.prototype.pause = function () { + if (this.currentFile) this.currentFile.pause() + this.paused = true + this.emit("pause") +} + +Generator.prototype.resume = function () { + this.paused = false + if (this.currentFile) this.currentFile.resume() + this.emit("resume") + this._processQueue() +} + +Generator.prototype.end = function () { + this._ended = true + this._processQueue() +} + +Generator.prototype.append = function (f, st) { + if (this._ended) return this.emit("error", new Error( + "Cannot append after ending")) + + // if it's a string, then treat it as a filename. + // if it's a number, then treat it as a fd + // if it's a Stats, then treat it as a stat object + // if it's a Stream, then stream it in. + var s = toFileStream(f, st) + if (!s) return this.emit("error", new TypeError( + "Invalid argument: "+f)) + + // make sure it's in the folder being added. + if (s.name.indexOf(this.options.cwd) !== 0) { + this.emit("error", new Error( + "Invalid argument: "+s.name+"\nOutside of "+this.options.cwd)) + } + + s.name = s.name.substr(this.options.cwd.length) + s.pause() + this._queue.push(s) + + if (!s._needStat) return this._processQueue() + + var self = this + fs.lstat(s.name, function (er, st) { + if (er) return self.emit("error", new Error( + "invalid file "+s.name+"\n"+er.message)) + s.mode = st.mode & 0777 + s.uid = st.uid + s.gid = st.gid + s.size = st.size + s.mtime = +st.mtime / 1000 + s.type = st.isFile() ? "0" + : st.isSymbolicLink() ? "2" + : st.isCharacterDevice() ? "3" + : st.isBlockDevice() ? "4" + : st.isDirectory() ? "5" + : st.isFIFO() ? "6" + : null + + // TODO: handle all the types in + // http://cdrecord.berlios.de/private/man/star/star.4.html + // for now, skip over unknown ones. + if (s.type === null) { + console.error("Unknown file type: " + s.name) + // kick out of the queue + var i = self._queue.indexOf(s) + if (i !== -1) self._queue.splice(i, 1) + self._processQueue() + return + } + + if (s.type === "2") return fs.readlink(s.name, function (er, n) { + if (er) return self.emit("error", new Error( + "error reading link value "+s.name+"\n"+er.message)) + s.linkname = n + s._needStat = false + self._processQueue() + }) + s._needStat = false + self._processQueue() + }) + return false +} + +function toFileStream (thing) { + if (typeof thing === "string") { + return toFileStream(fs.createReadStream(thing)) + } + + if (thing && typeof thing === "object") { + if (thing instanceof (Parser.File)) return thing + + if (thing instanceof Stream) { + if (thing.hasOwnProperty("name") && + thing.hasOwnProperty("mode") && + thing.hasOwnProperty("uid") && + thing.hasOwnProperty("gid") && + thing.hasOwnProperty("size") && + thing.hasOwnProperty("mtime") && + thing.hasOwnProperty("type")) return thing + + if (thing instanceof (fs.ReadStream)) { + thing.name = thing.path + } + + if (thing.name) { + thing._needStat = true + return thing + } + } + } + + return null +} + +Generator.prototype._processQueue = function processQueue () { + console.error("processQueue", this._queue[0]) + if (this._paused) return false + + if (this.currentFile || + this._queue.length && this._queue[0]._needStat) { + // either already processing one, or waiting on something. + return + } + + var f = this.currentFile = this._queue.shift() + if (!f) { + if (this._ended) { + // close it off with 2 blocks of nulls. + this.emit("data", new Buffer(new Array(512 * 2))) + this.emit("end") + this.emit("close") + } + return true + } + + if (f.type === Parser.File.types.Directory && + f.name.slice(-1) !== "/") { + f.name += "/" + } + + // write out a Pax header if the file isn't kosher. + if (this._needPax(f)) this._emitPax(f) + + // write out the header + f.ustar = true + this._emitHeader(f) + var fpos = 0 + , self = this + console.error("about to read body data", f) + f.on("data", function (c) { + self.emit("data", c) + self.fpos += c.length + }) + f.on("error", function (er) { self.emit("error", er) }) + f.on("end", function $END () { + // pad with \0 out to an even multiple of 512 bytes. + // this ensures that every file starts on a block. + var b = new Buffer(fpos % 512 || 512) + + for (var i = 0, l = b.length; i < l; i ++) b[i] = 0 + //console.log(b.length, b) + self.emit("data", b) + self.currentFile = null + self._processQueue() + }) + f.resume() +} + +Generator.prototype._needPax = function (f) { + // meh. why not? + return true + + return oddTextField(f.name, "NAME") || + oddTextField(f.link, "LINK") || + oddTextField(f.gname, "GNAME") || + oddTextField(f.uname, "UNAME") || + oddTextField(f.prefix, "PREFIX") +} + +// check if a text field is too long or non-ascii +function oddTextField (val, field) { + var nl = Buffer.byteLength(val) + , len = tar.fieldSize[field] + if (nl > len || nl !== val.length) return true +} + +// emit a Pax header of "key = val" for any file with +// odd or too-long field values. +Generator.prototype._emitPax = function (f) { + // since these tend to be relatively small, just go ahead + // and emit it all in-band. That saves having to keep + // track of the pax state in the generator, and we can + // go right back to emitting the file in the same tick. + var dir = f.name.replace(/[^\/]+\/?$/, "") + , base = f.name.substr(dir.length) + var pax = { name: dir + "PaxHeader/" +base + , mode: 0644 + , uid: f.uid + , gid: f.gid + , mtime: +f.mtime + // don't know size yet. + , size: -1 + , type: "x" // extended header + , ustar: true + , ustarVersion: "00" + , user: f.user || f.uname || "" + , group: f.group || f.gname || "" + , dev: { major: f.dev && f.dev.major || 0 + , minor: f.dev && f.dev.minor || 0 } + , prefix: f.prefix + , linkname: "" } + + // generate the Pax body + var kv = { path: (f.prefix ? f.prefix + "/" : "") + f.name + , atime: f.atime + , mtime: f.mtime + , ctime: f.ctime + , charset: "UTF-8" + , gid: f.gid + , uid: f.uid + , uname: f.user || f.uname || "" + , gname: f.group || f.gname || "" + , linkpath: f.linkpath || "" + , size: f.size + } + // "%d %s=%s\n", , , + // length includes the length of the length number, + // the key=val, and the \n. + var body = new Buffer(Object.keys(kv).map(function (key) { + if (!kv[key]) return ["", ""] + + var s = new Buffer(" " + key + "=" + kv[key]+"\n") + , digits = Math.floor(Math.log(s.length) / Math.log(10)) + 1 + + // if adding that many digits will make it go over that length, + // then add one to it + if (s.length > Math.pow(10, digits) - digits) digits ++ + + return [s.length + digits, s] + }).reduce(function (l, r) { + return l + r[0] + r[1] + }, "")) + + pax.size = body.length + this._emitHeader(pax) + this.emit("data", body) + // now the trailing buffer to make it an even number of 512 blocks + var b = new Buffer(512 + (body.length % 512 || 512)) + for (var i = 0, l = b.length; i < l; i ++) b[i] = 0 + this.emit("data", b) +} + +Generator.prototype._emitHeader = function (f) { + var header = new Buffer(new Array(512)) + , fields = tar.fields + , offs = tar.fieldOffs + , sz = tar.fieldSize + + addField(header, "NAME", f.name) + addField(header, "MODE", f.mode) + addField(header, "UID", f.uid) + addField(header, "GID", f.gid) + addField(header, "SIZE", f.size) + addField(header, "MTIME", +f.mtime) + // checksum is generated based on it being spaces + // then it's written as: "######\0 " + // where ### is a zero-lead 6-digit octal number + addField(header, "CKSUM", " ") + + addField(header, "TYPE", f.type) + addField(header, "LINKNAME", f.linkname || "") + if (f.ustar) { + console.error(">>> ustar!!") + addField(header, "USTAR", tar.ustar) + addField(header, "USTARVER", 0) + addField(header, "UNAME", f.user || "") + addField(header, "GNAME", f.group || "") + if (f.dev) { + addField(header, "DEVMAJ", f.dev.major || 0) + addField(header, "DEVMIN", f.dev.minor || 0) + } + addField(header, "PREFIX", f.prefix) + } else { + console.error(">>> no ustar!") + } + + // now the header is written except for checksum. + var ck = 0 + for (var i = 0; i < 512; i ++) ck += header[i] + addField(header, "CKSUM", nF(ck, 7)) + header[ offs[fields.CKSUM] + 7 ] = 0 + + this.emit("data", header) +} + +function addField (buf, field, val) { + var f = tar.fields[field] + console.error("Adding field", field, val) + val = typeof val === "number" + ? nF(val, tar.fieldSize[f]) + : new Buffer(val || "") + val.copy(buf, tar.fieldOffs[f]) +} + +function toBase256 (num, len) { + console.error("toBase256", num, len) + var positive = num > 0 + , buf = new Buffer(len) + if (!positive) { + // rare and slow + var b = num.toString(2).substr(1) + , padTo = (len - 1) * 8 + b = new Array(padTo - b.length + 1).join("0") + b + + // take the 2's complement + var ht = b.match(/^([01]*)(10*)?$/) + , head = ht[1] + , tail = ht[2] + head = head.split("1").join("2") + .split("0").join("1") + .split("2").join("0") + b = head + tail + + buf[0] = 0xFF + for (var i = 1; i < len; i ++) { + buf[i] = parseInt(buf.substr(i * 8, 8), 2) + } + return buf + } + + buf[0] = 0x80 + for (var i = 1, l = len, p = l - 1; i < l; i ++, p --) { + buf[p] = num % 256 + num = Math.floor(num / 256) + } + return buf +} + +function nF (num, size) { + var ns = num.toString(8) + + if (num < 0 || ns.length >= size) { + // make a base 256 buffer + // then return it + return toBase256(num, size) + } + + var buf = new Buffer(size) + ns = new Array(size - ns.length - 1).join("0") + ns + " " + buf[size - 1] = 0 + buf.asciiWrite(ns) + return buf +} diff --git a/deps/npm/node_modules/tar/old/parser.js b/deps/npm/node_modules/tar/old/parser.js new file mode 100644 index 0000000000..1582ee7577 --- /dev/null +++ b/deps/npm/node_modules/tar/old/parser.js @@ -0,0 +1,344 @@ +module.exports = Parser +Parser.create = create +Parser.File = File + +var tar = require("./tar") + , Stream = require("stream").Stream + , fs = require("fs") + +function create (cb) { + return new Parser(cb) +} + +var s = 0 + , HEADER = s ++ + , BODY = s ++ + , PAD = s ++ + +function Parser (cb) { + this.fields = tar.fields + this.fieldSize = tar.fieldSize + this.state = HEADER + this.position = 0 + this.currentFile = null + this._header = [] + this._headerPosition = 0 + this._bodyPosition = 0 + this.writable = true + Stream.apply(this) + if (cb) this.on("file", cb) +} + +Parser.prototype = Object.create(Stream.prototype) + +Parser.prototype.write = function (chunk) { + switch (this.state) { + case HEADER: + // buffer up to 512 bytes in memory, and then + // parse it, emit a "file" event, and stream the rest + this._header.push(chunk) + this._headerPosition += chunk.length + if (this._headerPosition >= tar.headerSize) { + return this._parseHeader() + } + return true + + case BODY: + // stream it through until the end of the file is reached, + // and then step over any \0 byte padding. + var cl = chunk.length + , bp = this._bodyPosition + , np = cl + bp + , s = this.currentFile.size + if (np < s) { + this._bodyPosition = np + return this.currentFile.write(chunk) + } + var c = chunk.slice(0, (s - bp)) + this.currentFile.write(c) + this._closeFile() + return this.write(chunk.slice(s - bp)) + + case PAD: + for (var i = 0, l = chunk.length; i < l; i ++) { + if (chunk[i] !== 0) { + this.state = HEADER + return this.write(chunk.slice(i)) + } + } + } + return true +} + +Parser.prototype.end = function (chunk) { + if (chunk) this.write(chunk) + if (this.currentFile) this._closeFile() + this.emit("end") + this.emit("close") +} + +// at this point, we have at least 512 bytes of header chunks +Parser.prototype._parseHeader = function () { + var hp = this._headerPosition + , last = this._header.pop() + , rem + + if (hp < 512) return this.emit("error", new Error( + "Trying to parse header before finished")) + + if (hp > 512) { + var ll = last.length + , llIntend = 512 - hp + ll + rem = last.slice(llIntend) + last = last.slice(0, llIntend) + } + this._header.push(last) + + var fields = tar.fields + , pos = 0 + , field = 0 + , fieldEnds = tar.fieldEnds + , fieldSize = tar.fieldSize + , set = {} + , fpos = 0 + + Object.keys(fieldSize).forEach(function (f) { + set[ fields[f] ] = new Buffer(fieldSize[f]) + }) + + this._header.forEach(function (chunk) { + for (var i = 0, l = chunk.length; i < l; i ++, pos ++, fpos ++) { + if (pos >= fieldEnds[field]) { + field ++ + fpos = 0 + } + // header is null-padded, so when the fields run out, + // just finish. + if (null === fields[field]) return + set[fields[field]][fpos] = chunk[i] + } + }) + + this._header.length = 0 + + // type definitions here: + // http://cdrecord.berlios.de/private/man/star/star.4.html + var type = set.TYPE.toString() + , file = this.currentFile = new File(set) + if (type === "\0" || + type >= "0" && type <= "7") { + this._addExtended(file) + this.emit("file", file) + } else if (type === "g") { + this._global = this._global || {} + readPax(this, file, this._global) + } else if (type === "h" || type === "x" || type === "X") { + this._extended = this._extended || {} + readPax(this, file, this._extended) + } else if (type === "K") { + this._readLongField(file, "linkname") + } else if (type === "L") { + this._readLongField(file, "name") + } + + this.state = BODY + if (rem) return this.write(rem) + return true +} + +function readPax (self, file, obj) { + var buf = "" + file.on("data", function (c) { + buf += c + var lines = buf.split(/\r?\n/) + buf = lines.pop() + lines.forEach(function (line) { + line = line.match(/^[0-9]+ ([^=]+)=(.*)/) + if (!line) return + obj[line[1]] = line[2] + }) + }) +} + +Parser.prototype._readLongField = function (f, field) { + var self = this + this._longFields[field] = "" + f.on("data", function (c) { + self._longFields[field] += c + }) +} + +Parser.prototype._addExtended = function (file) { + var g = this._global || {} + , e = this._extended || {} + file.extended = {} + ;[g, e].forEach(function (h) { + Object.keys(h).forEach(function (k) { + file.extended[k] = h[k] + // handle known fields + switch (k) { + case "path": file.name = h[k]; break + case "ctime": file.ctime = new Date(1000 * h[k]); break + case "mtime": file.mtime = new Date(1000 * h[k]); break + case "gid": file.gid = parseInt(h[k], 10); break + case "uid": file.uid = parseInt(h[k], 10); break + case "charset": file.charset = h[k]; break + case "gname": file.group = h[k]; break + case "uname": file.user = h[k]; break + case "linkpath": file.linkname = h[k]; break + case "size": file.size = parseInt(h[k], 10); break + case "SCHILY.devmajor": file.dev.major = parseInt(h[k], 10); break + case "SCHILY.devminor": file.dev.minor = parseInt(h[k], 10); break + } + }) + }) + var lf = this._longFields || {} + Object.keys(lf).forEach(function (f) { + file[f] = lf[f] + }) + this._extended = {} + this._longFields = {} +} + +Parser.prototype._closeFile = function () { + if (!this.currentFile) return this.emit("error", new Error( + "Trying to close without current file")) + + this._headerPosition = this._bodyPosition = 0 + this.currentFile.end() + this.currentFile = null + this.state = PAD +} + + +// file stuff + +function strF (f) { + return f.toString("ascii").split("\0").shift() || "" +} + +function parse256 (buf) { + // first byte MUST be either 80 or FF + // 80 for positive, FF for 2's comp + var positive + if (buf[0] === 0x80) positive = true + else if (buf[0] === 0xFF) positive = false + else return 0 + + if (!positive) { + // this is rare enough that the string slowness + // is not a big deal. You need *very* old files + // to ever hit this path. + var s = "" + for (var i = 1, l = buf.length; i < l; i ++) { + var byte = buf[i].toString(2) + if (byte.length < 8) { + byte = new Array(byte.length - 8 + 1).join("1") + byte + } + s += byte + } + var ht = s.match(/^([01]*)(10*)$/) + , head = ht[1] + , tail = ht[2] + head = head.split("1").join("2") + .split("0").join("1") + .split("2").join("0") + return -1 * parseInt(head + tail, 2) + } + + var sum = 0 + for (var i = 1, l = buf.length, p = l - 1; i < l; i ++, p--) { + sum += buf[i] * Math.pow(256, p) + } + return sum +} + +function nF (f) { + if (f[0] & 128 === 128) { + return parse256(f) + } + return parseInt(f.toString("ascii").replace(/\0+/g, "").trim(), 8) || 0 +} + +function bufferMatch (a, b) { + if (a.length != b.length) return false + for (var i = 0, l = a.length; i < l; i ++) { + if (a[i] !== b[i]) return false + } + return true +} + +function File (fields) { + this._raw = fields + this.name = strF(fields.NAME) + this.mode = nF(fields.MODE) + this.uid = nF(fields.UID) + this.gid = nF(fields.GID) + this.size = nF(fields.SIZE) + this.mtime = new Date(nF(fields.MTIME) * 1000) + this.cksum = nF(fields.CKSUM) + this.type = strF(fields.TYPE) + this.linkname = strF(fields.LINKNAME) + + this.ustar = bufferMatch(fields.USTAR, tar.ustar) + + if (this.ustar) { + this.ustarVersion = nF(fields.USTARVER) + this.user = strF(fields.UNAME) + this.group = strF(fields.GNAME) + this.dev = { major: nF(fields.DEVMAJ) + , minor: nF(fields.DEVMIN) } + this.prefix = strF(fields.PREFIX) + if (this.prefix) { + this.name = this.prefix + "/" + this.name + } + } + + this.writable = true + this.readable = true + Stream.apply(this) +} + +File.prototype = Object.create(Stream.prototype) + +File.types = { File: "0" + , HardLink: "1" + , SymbolicLink: "2" + , CharacterDevice: "3" + , BlockDevice: "4" + , Directory: "5" + , FIFO: "6" + , ContiguousFile: "7" } + +Object.keys(File.types).forEach(function (t) { + File.prototype["is"+t] = function () { + return File.types[t] === this.type + } + File.types[ File.types[t] ] = File.types[t] +}) + +// contiguous files are treated as regular files for most purposes. +File.prototype.isFile = function () { + return this.type === "0" && this.name.slice(-1) !== "/" + || this.type === "7" +} + +File.prototype.isDirectory = function () { + return this.type === "5" + || this.type === "0" && this.name.slice(-1) === "/" +} + +File.prototype.write = function (c) { + this.emit("data", c) + return true +} + +File.prototype.end = function (c) { + if (c) this.write(c) + this.emit("end") + this.emit("close") +} + +File.prototype.pause = function () { this.emit("pause") } + +File.prototype.resume = function () { this.emit("resume") } diff --git a/deps/npm/node_modules/tar/old/tar.js b/deps/npm/node_modules/tar/old/tar.js new file mode 100644 index 0000000000..f70c081d27 --- /dev/null +++ b/deps/npm/node_modules/tar/old/tar.js @@ -0,0 +1,74 @@ +// field names that every tar file must have. +// header is padded to 512 bytes. +var f = 0 + , fields = {} + , NAME = fields.NAME = f++ + , MODE = fields.MODE = f++ + , UID = fields.UID = f++ + , GID = fields.GID = f++ + , SIZE = fields.SIZE = f++ + , MTIME = fields.MTIME = f++ + , CKSUM = fields.CKSUM = f++ + , TYPE = fields.TYPE = f++ + , LINKNAME = fields.LINKNAME = f++ + , headerSize = 512 + , fieldSize = [] + +fieldSize[NAME] = 100 +fieldSize[MODE] = 8 +fieldSize[UID] = 8 +fieldSize[GID] = 8 +fieldSize[SIZE] = 12 +fieldSize[MTIME] = 12 +fieldSize[CKSUM] = 8 +fieldSize[TYPE] = 1 +fieldSize[LINKNAME] = 100 + +// "ustar\0" may introduce another bunch of headers. +// these are optional, and will be nulled out if not present. +var ustar = new Buffer(6) +ustar.asciiWrite("ustar\0") + +var USTAR = fields.USTAR = f++ + , USTARVER = fields.USTARVER = f++ + , UNAME = fields.UNAME = f++ + , GNAME = fields.GNAME = f++ + , DEVMAJ = fields.DEVMAJ = f++ + , DEVMIN = fields.DEVMIN = f++ + , PREFIX = fields.PREFIX = f++ +// terminate fields. +fields[f] = null + +fieldSize[USTAR] = 6 +fieldSize[USTARVER] = 2 +fieldSize[UNAME] = 32 +fieldSize[GNAME] = 32 +fieldSize[DEVMAJ] = 8 +fieldSize[DEVMIN] = 8 +fieldSize[PREFIX] = 155 + +var fieldEnds = {} + , fieldOffs = {} + , fe = 0 +for (var i = 0; i < f; i ++) { + fieldOffs[i] = fe + fieldEnds[i] = (fe += fieldSize[i]) +} + +// build a translation table of field names. +Object.keys(fields).forEach(function (f) { + fields[fields[f]] = f +}) + +exports.ustar = ustar +exports.fields = fields +exports.fieldSize = fieldSize +exports.fieldOffs = fieldOffs +exports.fieldEnds = fieldEnds +exports.headerSize = headerSize + +var Parser = exports.Parser = require("./parser") +exports.createParser = Parser.create + +var Generator = exports.Generator = require("./generator") +exports.createGenerator = Generator.create diff --git a/deps/npm/node_modules/tar/old/test/test-generator.js b/deps/npm/node_modules/tar/old/test/test-generator.js new file mode 100644 index 0000000000..dea2732161 --- /dev/null +++ b/deps/npm/node_modules/tar/old/test/test-generator.js @@ -0,0 +1,13 @@ +// pipe this file to tar vt + +var Generator = require("../generator") + , fs = require("fs") + , path = require("path") + , ohm = fs.createReadStream(path.resolve(__dirname, "tar-files/Ω.txt")) + , foo = path.resolve(__dirname, "tar-files/foo.js") + , gen = Generator.create({cwd: __dirname}) + +gen.pipe(process.stdout) +gen.append(ohm) +//gen.append(foo) +gen.end() diff --git a/deps/npm/node_modules/tar/old/test/test-generator.tar b/deps/npm/node_modules/tar/old/test/test-generator.tar new file mode 100644 index 0000000000000000000000000000000000000000..6752aa51de2f8362d8db9c622e8c44e5ae7c8463 GIT binary patch literal 3260 zcmeHHOA5j;5Y0NLm?o4@kU<8GnrD|xR6Q;-Q;cZVZJYu z7hE+tlxJVh)~;E3xA)cT`Iv?#u)%=-RpCKa0%tbTarjZD=iHt1G|%} zY9H)&vz*NN0*>Y03gjf=*6x%pDBjlP-CM!?mysJ=8T(SM l_)}Ub!Cui5jfPUL#4^Tme?G>_obzS|G6R``%)qz|yZ~{8NM-;4 literal 0 HcmV?d00001 diff --git a/deps/npm/node_modules/tar/old/test/test-generator.txt b/deps/npm/node_modules/tar/old/test/test-generator.txt new file mode 100644 index 0000000000000000000000000000000000000000..349757ba692925ea24ac27eb77f256c6c0052fb3 GIT binary patch literal 3261 zcmeHHOA5j;5Y0NLm?o4@kU;zXEMbK;-VB(5Y5}!K5mV_1>9us@awi47_IyE*&~S8NXQK%1Ke)X<7YK)tAG{@rB(&1fa5Avk zxuW#JE*I0mm``AvudP5%5>8D|>5SrRnV-EC{0sP{6n<5K=O6nCp0TJqoeKr9N9Uqg hDF}?c>?VE4w9XZ;`%Bc9zoeZ=Ck2uMNr9vQ<5vbbNZP$|5kL z1Z1mJA^WRUFmusQ7k!&bsXOm?-Q~oTR#lh7Lo;JjawhYxLS*E(kMCuIFua5R=?en6 zi}hzHQ2}bRA?wbXA2oK&#-K=pe0|(LJ-@B?;kP8EokJiNq(;4NzLIV&ZI}hRQEm=( z^HHL>D;$d}W;*v<07z>-yZ`AonC!o{426VS6f?bZ;ro9&PJv`U6UYQIflOct1pWYk CGhK-Q literal 0 HcmV?d00001 diff --git a/deps/npm/node_modules/tar/old/test/test-tar.txt b/deps/npm/node_modules/tar/old/test/test-tar.txt new file mode 100644 index 0000000000000000000000000000000000000000..d9ac62cc5823cde33a0c0052136308972fa421b6 GIT binary patch literal 3073 zcmeH{&2GXl499!UQ{)8z+xcn_J#k^u4oIAKj3Cvbv2JR!sjtM#@Js~JUMkTONJxz@ zk^i (http://blog.izs.me/)", + "name": "tar", + "description": "tar for node", + "version": "0.1.0", + "repository": { + "type": "git", + "url": "git://github.com/isaacs/node-tar.git" + }, + "main": "tar.js", + "scripts": { + "test": "rm -rf test/tmp; tap test/*.js" + }, + "engines": { + "node": "~0.5.9 || 0.6 || 0.7 || 0.8" + }, + "dependencies": { + "inherits": "1.x", + "block-stream": "*", + "fstream": "~0.1" + }, + "devDependencies": { + "tap": "0.x", + "rimraf": "1.x" + } +} diff --git a/deps/npm/node_modules/tar/tar.js b/deps/npm/node_modules/tar/tar.js new file mode 100644 index 0000000000..b9dbca489c --- /dev/null +++ b/deps/npm/node_modules/tar/tar.js @@ -0,0 +1,172 @@ +// field paths that every tar file must have. +// header is padded to 512 bytes. +var f = 0 + , fields = {} + , path = fields.path = f++ + , mode = fields.mode = f++ + , uid = fields.uid = f++ + , gid = fields.gid = f++ + , size = fields.size = f++ + , mtime = fields.mtime = f++ + , cksum = fields.cksum = f++ + , type = fields.type = f++ + , linkpath = fields.linkpath = f++ + , headerSize = 512 + , blockSize = 512 + , fieldSize = [] + +fieldSize[path] = 100 +fieldSize[mode] = 8 +fieldSize[uid] = 8 +fieldSize[gid] = 8 +fieldSize[size] = 12 +fieldSize[mtime] = 12 +fieldSize[cksum] = 8 +fieldSize[type] = 1 +fieldSize[linkpath] = 100 + +// "ustar\0" may introduce another bunch of headers. +// these are optional, and will be nulled out if not present. + +var ustar = fields.ustar = f++ + , ustarver = fields.ustarver = f++ + , uname = fields.uname = f++ + , gname = fields.gname = f++ + , devmaj = fields.devmaj = f++ + , devmin = fields.devmin = f++ + , prefix = fields.prefix = f++ + , fill = fields.fill = f++ + +// terminate fields. +fields[f] = null + +fieldSize[ustar] = 6 +fieldSize[ustarver] = 2 +fieldSize[uname] = 32 +fieldSize[gname] = 32 +fieldSize[devmaj] = 8 +fieldSize[devmin] = 8 +fieldSize[prefix] = 155 +fieldSize[fill] = 12 + +// nb: prefix field may in fact be 130 bytes of prefix, +// a null char, 12 bytes for atime, 12 bytes for ctime. +// +// To recognize this format: +// 1. prefix[130] === ' ' or '\0' +// 2. atime and ctime are octal numeric values +// 3. atime and ctime have ' ' in their last byte + +var fieldEnds = {} + , fieldOffs = {} + , fe = 0 +for (var i = 0; i < f; i ++) { + fieldOffs[i] = fe + fieldEnds[i] = (fe += fieldSize[i]) +} + +// build a translation table of field paths. +Object.keys(fields).forEach(function (f) { + if (fields[f] !== null) fields[fields[f]] = f +}) + +// different values of the 'type' field +// paths match the values of Stats.isX() functions, where appropriate +var types = + { 0: "File" + , "\0": "OldFile" // like 0 + , 1: "Link" + , 2: "SymbolicLink" + , 3: "CharacterDevice" + , 4: "BlockDevice" + , 5: "Directory" + , 6: "FIFO" + , 7: "ContiguousFile" // like 0 + // posix headers + , g: "GlobalExtendedHeader" // k=v for the rest of the archive + , x: "ExtendedHeader" // k=v for the next file + // vendor-specific stuff + , A: "SolarisACL" // skip + , D: "GNUDumpDir" // like 5, but with data, which should be skipped + , I: "Inode" // metadata only, skip + , K: "NextFileHasLongLinkpath" // data = link path of next file + , L: "NextFileHasLongPath" // data = path of next file + , M: "ContinuationFile" // skip + , N: "OldGnuLongPath" // like L + , S: "SparseFile" // skip + , V: "TapeVolumeHeader" // skip + , X: "OldExtendedHeader" // like x + } + +Object.keys(types).forEach(function (t) { + types[types[t]] = types[types[t]] || t +}) + +// values for the mode field +var modes = + { suid: 04000 // set uid on extraction + , sgid: 02000 // set gid on extraction + , svtx: 01000 // set restricted deletion flag on dirs on extraction + , uread: 0400 + , uwrite: 0200 + , uexec: 0100 + , gread: 040 + , gwrite: 020 + , gexec: 010 + , oread: 4 + , owrite: 2 + , oexec: 1 + , all: 07777 + } + +var numeric = + { mode: true + , uid: true + , gid: true + , size: true + , mtime: true + , devmaj: true + , devmin: true + , cksum: true + , atime: true + , ctime: true + , dev: true + , ino: true + , nlink: true + } + +Object.keys(modes).forEach(function (t) { + modes[modes[t]] = modes[modes[t]] || t +}) + +var knownExtended = + { atime: true + , charset: true + , comment: true + , ctime: true + , gid: true + , gname: true + , linkpath: true + , mtime: true + , path: true + , realtime: true + , security: true + , size: true + , uid: true + , uname: true } + + +exports.fields = fields +exports.fieldSize = fieldSize +exports.fieldOffs = fieldOffs +exports.fieldEnds = fieldEnds +exports.types = types +exports.modes = modes +exports.numeric = numeric +exports.headerSize = headerSize +exports.blockSize = blockSize +exports.knownExtended = knownExtended + +exports.Pack = require("./lib/pack.js") +exports.Parse = require("./lib/parse.js") +exports.Extract = require("./lib/extract.js") diff --git a/deps/npm/node_modules/tar/test/extract.js b/deps/npm/node_modules/tar/test/extract.js new file mode 100644 index 0000000000..e2dea5c053 --- /dev/null +++ b/deps/npm/node_modules/tar/test/extract.js @@ -0,0 +1,406 @@ +var tap = require("tap") + , tar = require("../tar.js") + , fs = require("fs") + , path = require("path") + , file = path.resolve(__dirname, "fixtures/c.tar") + , target = path.resolve(__dirname, "tmp/extract-test") + , index = 0 + , fstream = require("fstream") + + , ee = 0 + , expectEntries = +[ { path: 'c.txt', + mode: '644', + type: '0', + depth: undefined, + size: 513, + linkpath: '', + nlink: undefined, + dev: undefined, + ino: undefined }, + { path: 'cc.txt', + mode: '644', + type: '0', + depth: undefined, + size: 513, + linkpath: '', + nlink: undefined, + dev: undefined, + ino: undefined }, + { path: 'r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: '644', + type: '0', + depth: undefined, + size: 100, + linkpath: '', + nlink: undefined, + dev: undefined, + ino: undefined }, + { path: 'Ω.txt', + mode: '644', + type: '0', + depth: undefined, + size: 2, + linkpath: '', + nlink: undefined, + dev: undefined, + ino: undefined }, + { path: 'Ω.txt', + mode: '644', + type: '0', + depth: undefined, + size: 2, + linkpath: '', + nlink: 1, + dev: 234881026, + ino: 51693379 }, + { path: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: '644', + type: '0', + depth: undefined, + size: 200, + linkpath: '', + nlink: 1, + dev: 234881026, + ino: 51681874 }, + { path: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: '644', + type: '0', + depth: undefined, + size: 201, + linkpath: '', + nlink: undefined, + dev: undefined, + ino: undefined }, + { path: '200LLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLL', + mode: '777', + type: '2', + depth: undefined, + size: 0, + linkpath: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + nlink: undefined, + dev: undefined, + ino: undefined }, + { path: '200-hard', + mode: '644', + type: '0', + depth: undefined, + size: 200, + linkpath: '', + nlink: 2, + dev: 234881026, + ino: 51681874 }, + { path: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: '644', + type: '1', + depth: undefined, + size: 0, + linkpath: path.resolve(target, '200-hard'), + nlink: 2, + dev: 234881026, + ino: 51681874 } ] + + , ef = 0 + , expectFiles = +[ { path: '', + mode: '40755', + type: 'Directory', + depth: 0, + size: 306, + linkpath: undefined, + nlink: 9 }, + { path: '/200-hard', + mode: '100644', + type: 'File', + depth: 1, + size: 200, + linkpath: undefined, + nlink: 2 }, + { path: '/200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: '100644', + type: 'Link', + depth: 1, + size: 200, + linkpath: '/Users/isaacs/dev-src/js/node-tar/test/tmp/extract-test/200-hard', + nlink: 2 }, + { path: '/200LLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLL', + mode: '120777', + type: 'SymbolicLink', + depth: 1, + size: 200, + linkpath: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + nlink: 1 }, + { path: '/c.txt', + mode: '100644', + type: 'File', + depth: 1, + size: 513, + linkpath: undefined, + nlink: 1 }, + { path: '/cc.txt', + mode: '100644', + type: 'File', + depth: 1, + size: 513, + linkpath: undefined, + nlink: 1 }, + { path: '/r', + mode: '40755', + type: 'Directory', + depth: 1, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e', + mode: '40755', + type: 'Directory', + depth: 2, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a', + mode: '40755', + type: 'Directory', + depth: 3, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l', + mode: '40755', + type: 'Directory', + depth: 4, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l', + mode: '40755', + type: 'Directory', + depth: 5, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y', + mode: '40755', + type: 'Directory', + depth: 6, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-', + mode: '40755', + type: 'Directory', + depth: 7, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d', + mode: '40755', + type: 'Directory', + depth: 8, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e', + mode: '40755', + type: 'Directory', + depth: 9, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e', + mode: '40755', + type: 'Directory', + depth: 10, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p', + mode: '40755', + type: 'Directory', + depth: 11, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-', + mode: '40755', + type: 'Directory', + depth: 12, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f', + mode: '40755', + type: 'Directory', + depth: 13, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o', + mode: '40755', + type: 'Directory', + depth: 14, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l', + mode: '40755', + type: 'Directory', + depth: 15, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d', + mode: '40755', + type: 'Directory', + depth: 16, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e', + mode: '40755', + type: 'Directory', + depth: 17, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r', + mode: '40755', + type: 'Directory', + depth: 18, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-', + mode: '40755', + type: 'Directory', + depth: 19, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p', + mode: '40755', + type: 'Directory', + depth: 20, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a', + mode: '40755', + type: 'Directory', + depth: 21, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t', + mode: '40755', + type: 'Directory', + depth: 22, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h', + mode: '40755', + type: 'Directory', + depth: 23, + size: 102, + linkpath: undefined, + nlink: 3 }, + { path: '/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: '100644', + type: 'File', + depth: 24, + size: 100, + linkpath: undefined, + nlink: 1 }, + { path: '/Ω.txt', + mode: '100644', + type: 'File', + depth: 1, + size: 2, + linkpath: undefined, + nlink: 1 } ] + + + +// The extract class basically just pipes the input +// to a Reader, and then to a fstream.DirWriter + +// So, this is as much a test of fstream.Reader and fstream.Writer +// as it is of tar.Extract, but it sort of makes sense. + +tap.test("extract test", function (t) { + var extract = tar.Extract(target) + var inp = fs.createReadStream(file) + + // give it a weird buffer size to try to break in odd places + inp.bufferSize = 1234 + + inp.pipe(extract) + + extract.on("end", function () { + t.equal(ee, expectEntries.length, "should see "+ee+" entries") + + // should get no more entries after end + extract.removeAllListeners("entry") + extract.on("entry", function (e) { + t.fail("Should not get entries after end!") + }) + + next() + }) + + extract.on("entry", function (entry) { + var found = + { path: entry.path + , mode: entry.props.mode.toString(8) + , type: entry.props.type + , depth: entry.props.depth + , size: entry.props.size + , linkpath: entry.props.linkpath + , nlink: entry.props.nlink + , dev: entry.props.dev + , ino: entry.props.ino + } + + var wanted = expectEntries[ee ++] + + t.equivalent(found, wanted, "tar entry " + ee + " " + wanted.path) + }) + + function next () { + var r = fstream.Reader({ path: target + , type: "Directory" + // this is just to encourage consistency + , sort: "alpha" }) + + r.on("ready", function () { + foundEntry(r) + }) + + r.on("end", finish) + + function foundEntry (entry) { + var p = entry.path.substr(target.length) + var found = + { path: p + , mode: entry.props.mode.toString(8) + , type: entry.props.type + , depth: entry.props.depth + , size: entry.props.size + , linkpath: entry.props.linkpath + , nlink: entry.props.nlink + } + + var wanted = expectFiles[ef ++] + + t.equivalent(found, wanted, "unpacked file " + ef + " " + wanted.path) + + entry.on("entry", foundEntry) + } + + function finish () { + t.equal(ef, expectFiles.length, "should have "+ef+" items") + t.end() + } + } +}) diff --git a/deps/npm/node_modules/tar/test/fixtures/200.tar b/deps/npm/node_modules/tar/test/fixtures/200.tar new file mode 100644 index 0000000000000000000000000000000000000000..7e3a8f3e77fa6081a8c32bfcfcdab57557bb60e2 GIT binary patch literal 3072 zcmeHHO;5ux49z*eBEMkR&PT%`2S7-hv;zVOiDT4dq5?Y97POy_E3G>)t*2$0pw1!6 z^Go8`etC%7d5WvlToFK-!-N0;XN(eHi^3t#Y}BuJW(_JtX9+E8c7QgdL8z@bgMGDK zY&evvjj?QxU&yre`nY@dnBh@sgn3aeLQtQvdkwkDpQEM5&XfeI?5d9xP~)}p{phi{ z#|nc;PHL}~i`$#WyZPezepWV#8Jl{~>i1ca+>TGoSia0w={t}%-XrjC(gR-An_$s- z^o5b=2e^K(>bHRMqN;ru%o)nwOJh|3ntSk<{f~ OrwmLPm@@FUGw=gL<}U)LWSD>|5lc=c#xhHE6ydzL=6+;bbZ%9j9p)5 zS>KQb_ya^D4n!yCQvivWHHc@La>7$TZl5+GV6%1)nQ(Q*huODTNyNDZ+(z)i3hfG= ze}y1S4z1e!n;MYgsnmGK@&6}pzc+QQwfA_^eE9o6dddTc6iYS8kr|LBjROdo!<^3f YU)(g!F5coRKEAUtO@XFBQ=mu%z6f;$5C8xG literal 0 HcmV?d00001 diff --git a/deps/npm/node_modules/tar/test/fixtures/200longname.tar b/deps/npm/node_modules/tar/test/fixtures/200longname.tar new file mode 100644 index 0000000000000000000000000000000000000000..5556567e1511b94317eded255d1827ea936402a1 GIT binary patch literal 3072 zcmeH{K?=km3`MiUSkcc&cFxQL|-}>^KU4?+vxh+=W>P(NjZ&US%<3Gf0fEPAs zj{OHyYo&t6Y2c5)j~t}0|A{pBk`)gcP)daw1m literal 0 HcmV?d00001 diff --git a/deps/npm/node_modules/tar/test/fixtures/a.hex b/deps/npm/node_modules/tar/test/fixtures/a.hex new file mode 100644 index 0000000000..529e9cbb36 --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/a.hex @@ -0,0 +1,14 @@ +-- header -- +612e7478740000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000303030363434200030353737363120003030303032342000303030303030303034303120313136353133363033333320303132343531002030000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 a.txt...............................................................................................000644..057761..000024..00000000401.11651360333.012451..0................................................................................................... +00757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 .ustar.00isaacs..........................staff...........................000000..000000......................................................................................................................................................................... + +-- file contents -- +61616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161616161 aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa +61000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 a............................................................................................................................................................................................................................................................... + +-- tar eof -- +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ + diff --git a/deps/npm/node_modules/tar/test/fixtures/a.tar b/deps/npm/node_modules/tar/test/fixtures/a.tar new file mode 100644 index 0000000000000000000000000000000000000000..27604d7ec37fb71f6906a9ca689a05947faf3421 GIT binary patch literal 2048 zcmeHENeX~44D3_#0!f-K?+HS|tF7SkjaZNMRuSx6LYR=uOk^s1F@1rE!8tJKV+^EK zVb8w8ApueF!~vNZkgW4$0Q<_iwr{z}s#2CLGUxV19ZRWAS2MRMc$oYYC4a}KzA?m% M1V#cQf%g(P0GjxA*Z=?k literal 0 HcmV?d00001 diff --git a/deps/npm/node_modules/tar/test/fixtures/a.txt b/deps/npm/node_modules/tar/test/fixtures/a.txt new file mode 100644 index 0000000000..a6c406965f --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/a.txt @@ -0,0 +1 @@ +aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa \ No newline at end of file diff --git a/deps/npm/node_modules/tar/test/fixtures/b.hex b/deps/npm/node_modules/tar/test/fixtures/b.hex new file mode 100644 index 0000000000..cf36eb6bbb --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/b.hex @@ -0,0 +1,14 @@ +-- normal header -- +622e7478740000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000303030363434200030353737363120003030303032342000303030303030303130303020313136353133363036373720303132343631002030000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 b.txt...............................................................................................000644..057761..000024..00000001000.11651360677.012461..0................................................................................................... +00757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 .ustar.00isaacs..........................staff...........................000000..000000......................................................................................................................................................................... + +-- file contents - exactly 512 bytes, no null padding -- +62626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262 bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb +62626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262626262 bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb + +-- tar eof blocks -- +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ + diff --git a/deps/npm/node_modules/tar/test/fixtures/b.tar b/deps/npm/node_modules/tar/test/fixtures/b.tar new file mode 100644 index 0000000000000000000000000000000000000000..2d8e7b3ac7a3d7a20bc21576e0b65c745c8efdda GIT binary patch literal 2048 zcmeHF(F%Yd4D?g>1+kgV@6kZ$H4Xaw7S^M^8A9I67(2GRV`0iiR&9ZZYz!#0&M`?9 z=&LUvDL}-QW=3`nkhEb^fNf=!?R(086De!*eo6IE$NYRnVzY$QdCC{ksQd5 zWqa2FX?HgJnD^|C^6~m^9r-N)V3dj@(AFBHCl%r9bwL1(5sjul3rfU*LTMAl;CET| z=)Z2)SxHM-+-6yR`}~a5?eg+@@WW=BO~oBO2L;MQIcW@x2E~nGG29r)bN>c_)cgzd zzj6BC2p#l42QQqo@(Y{KgZ}4`vf<9bAp$-6ACQ6r=l=`TsHKcSNr~#f^?#WZNtRsG z?@w};%&Ar+H}rOytfd*}Nzdf104z{#)3Re+};Z?~Vf_Cw~5iI=;!) zSJRsMBLQ)~Ui>VkC~;z?oxsGkWslk>U~ojow_o0$U(CLZ=f&@-kSa|Pgc%9hv@e#c zDV^v}Bn>jcv}s;mFP1;17{U$?-rivje^S4hyifjv{omk<`z_diZj29WzK#7C0A7N@ z$X$(fo&SPX_5GiV|0P)#kV;4h_Md}O1VZfq_#(d-p6VF1?ZTMU{kB{fg+i;5Oyk+b zySJaojrsavoRg+c#&k@xLE4+rDJS}6zpolsK%;A9-FuH?UYI#4D5mHn4mX=@^o zQUPoJAIX1T{$C$_SY+LKH5~NcmQ^41aG4y|A+jK*Qdqyh>sFJ1SNJ@ z%Ef(rTT-EpXjD%O+KZ&uzX|4;6J-1k4M=Rahc`t$gP|&5&>iv!QGj>xP@+~1GgqIgnYM0_(4I+AGa5#u~ z(yHl>{s{xoz`Yj~CP3;ui35C<>N>uaHDpDZO9)d%gs#O{0Xl!VpMuKfuSR3Y&98tA O#tfJNGhhZjF|Y%u7Z)G^ literal 0 HcmV?d00001 diff --git a/deps/npm/node_modules/tar/test/fixtures/hardlink-1 b/deps/npm/node_modules/tar/test/fixtures/hardlink-1 new file mode 100644 index 0000000000..c2b6e5096d --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/hardlink-1 @@ -0,0 +1 @@ +200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc \ No newline at end of file diff --git a/deps/npm/node_modules/tar/test/fixtures/hardlink-2 b/deps/npm/node_modules/tar/test/fixtures/hardlink-2 new file mode 100644 index 0000000000..c2b6e5096d --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/hardlink-2 @@ -0,0 +1 @@ +200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc \ No newline at end of file diff --git a/deps/npm/node_modules/tar/test/fixtures/omega.hex b/deps/npm/node_modules/tar/test/fixtures/omega.hex new file mode 100644 index 0000000000..eef879682e --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/omega.hex @@ -0,0 +1,22 @@ +-- pax header -- +5061784865616465722fcea92e747874000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000303030363434200030353737363120003030303032342000303030303030303031373020313135343337313036313120303135303531002078000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 PaxHeader/Ω.txt....................................................................................000644..057761..000024..00000000170.11543710611.015051..x................................................................................................... +00757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 .ustar.00isaacs..........................staff...........................000000..000000......................................................................................................................................................................... + +-- pax header contents -- +313520706174683dcea92e7478740a3230206374696d653d313330313435393237380a3230206174696d653d313330313431353738330a323420534348494c592e6465763d3233343838313032360a323320534348494c592e696e6f3d32333737323936360a313820534348494c592e6e6c696e6b3d310a00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 15.path=Ω.txt.20.ctime=1301459278.20.atime=1301415783.24.SCHILY.dev=234881026.23.SCHILY.ino=23772966.18.SCHILY.nlink=1......................................................................................................................................... +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ + +-- normal header -- +cea92e74787400000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000303030363434200030353737363120003030303032342000303030303030303030303220313135343337313036313120303133303732002030000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 Ω.txt..............................................................................................000644..057761..000024..00000000002.11543710611.013072..0................................................................................................... +00757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 .ustar.00isaacs..........................staff...........................000000..000000......................................................................................................................................................................... + +-- file contents -- +cea90000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 Ω.............................................................................................................................................................................................................................................................. +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ + +-- tar eof marker -- +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ +00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ................................................................................................................................................................................................................................................................ + diff --git a/deps/npm/node_modules/tar/test/fixtures/omega.tar b/deps/npm/node_modules/tar/test/fixtures/omega.tar new file mode 100644 index 0000000000000000000000000000000000000000..6590e58ce41c56920dfe19fa8b9d18c6f5d8d839 GIT binary patch literal 3072 zcmeH{&x*n@5XO7XQ{)NUnf#l0=xHws9u_=#3|iT;S{ECoujI@0nKs3(1qL6ZasFs;GvD|uLI8kKif5pS42Cs?=<@$S0*4)Ou3)56k^Izw`^ZV;F0etJrW09i-RC+GR zcA6VB)MO^V z)agG^)c>n|JV@-#HqZIC`a0F@C!n>Y?!rI&xXBpmDeY|ioD f>wnLG0a{1>?-2~Qi~1i9+ZZ__5D|z73>kq>zL21R literal 0 HcmV?d00001 diff --git a/deps/npm/node_modules/tar/test/fixtures/packtest/omega.txt b/deps/npm/node_modules/tar/test/fixtures/packtest/omega.txt new file mode 100644 index 0000000000..1ca042fff2 --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/packtest/omega.txt @@ -0,0 +1 @@ +Ω \ No newline at end of file diff --git a/deps/npm/node_modules/tar/test/fixtures/packtest/star.4.html b/deps/npm/node_modules/tar/test/fixtures/packtest/star.4.html new file mode 100644 index 0000000000..b600d772f5 --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/packtest/star.4.html @@ -0,0 +1,1184 @@ + + +Manpage for star.4 + + + + + +

NAME

+     star - tape archive file format
+
+
+
+

DESCRIPTION

+     Tar Archives are layered archives.  The basic  structure  is
+     defined by the POSIX.1-1988 archive format and documented in
+     the BASIC TAR HEADER DESCRIPTION section below.  The  higher
+     level  structure  is  defined  by  the POSIX.1-2001 extended
+     headers and documented in  the  EXTENDED  TAR  (PAX)  HEADER
+     STRUCTURE  section below.  POSIX.1-2001 extended headers are
+     pseudo files that contain an unlimited  number  of  extended
+     header  keywords  and associated values. The header keywords
+     are documented in the EXTENDED  TAR  (PAX)  HEADER  KEYWORDS
+     section below.
+
+
+
+

BASIC TAR HEADER DESCRIPTION

+     Physically, a POSIX.1-1988 tar archive consists of a  series
+     of  fixed  sized blocks of TBLOCK (512) characters.  It con-
+     tains a series of  file  entries  terminated  by  a  logical
+     end-of-archive  marker,  which consists of two blocks of 512
+     bytes of binary zeroes.  Each file entry is represented by a
+     header block that describes the file followed by one or more
+     blocks with the content of the file. The length of each file
+     is rounded up to a multiple of 512 bytes.
+
+     A number of TBLOCK sizes blocks are grouped  together  to  a
+     tape  record  for  physical I/O operations. Each record of n
+     blocks is written with a single write(2) operation.  On mag-
+     netic tapes, this results in a single tape record.
+
+     The header block is defined in star.h as follows:
+     /*
+      * POSIX.1-1988 field size values and magic.
+      */
+     #define   TBLOCK         512
+     #define   NAMSIZ         100
+     #define   PFXSIZ         155
+
+     #define   TMODLEN        8
+     #define   TUIDLEN        8
+     #define   TGIDLEN        8
+     #define   TSIZLEN        12
+     #define   TMTMLEN        12
+     #define   TCKSLEN        8
+
+     #define   TMAGIC         "ustar"   /* ustar magic 6 chars + '\0' */
+     #define   TMAGLEN        6         /* "ustar" including '\0' */
+     #define   TVERSION       "00"
+     #define   TVERSLEN       2
+     #define   TUNMLEN        32
+     #define   TGNMLEN        32
+     #define   TDEVLEN        8
+
+Joerg Schilling       Last change: 05/10/19                     1
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+     /*
+      * POSIX.1-1988 typeflag values
+      */
+     #define   REGTYPE        '0'  /* Regular File          */
+     #define   AREGTYPE       '\0' /* Regular File (outdated) */
+     #define   LNKTYPE        '1'  /* Hard Link             */
+     #define   SYMTYPE        '2'  /* Symbolic Link         */
+     #define   CHRTYPE        '3'  /* Character Special     */
+     #define   BLKTYPE        '4'  /* Block Special         */
+     #define   DIRTYPE        '5'  /* Directory             */
+     #define   FIFOTYPE       '6'  /* FIFO (named pipe)     */
+     #define   CONTTYPE       '7'  /* Contiguous File       */
+
+     /*
+      * POSIX.1-2001 typeflag extensions.
+      * POSIX.1-2001 calls the extended USTAR format PAX although it is
+      * definitely derived from and based on USTAR. The reason may be that
+      * POSIX.1-2001 calls the tar program outdated and lists the
+      * pax program as the successor.
+      */
+     #define   LF_GHDR        'g'  /* POSIX.1-2001 global extended header */
+     #define   LF_XHDR        'x'  /* POSIX.1-2001 extended header */
+
+     See section EXTENDED TAR  (PAX)  HEADER  KEYWORDS  for  more
+     information about the structure of a POSIX.1-2001 header.
+
+     /*
+      * star/gnu/Sun tar extensions:
+      *
+      * Note that the standards committee allows only capital A through
+      * capital Z for user-defined expansion.  This means that defining
+      * something as, say '8' is a *bad* idea.
+      */
+
+     #define   LF_ACL         'A'  /* Solaris Access Control List     */
+     #define   LF_DUMPDIR     'D'  /* GNU dump dir                    */
+     #define   LF_EXTATTR     'E'  /* Solaris Extended Attribute File */
+     #define   LF_META        'I'  /* Inode (metadata only) no file content */
+     #define   LF_LONGLINK    'K'  /* NEXT file has a long linkname   */
+     #define   LF_LONGNAME    'L'  /* NEXT file has a long name       */
+     #define   LF_MULTIVOL    'M'  /* Continuation file rest to be skipped */
+     #define   LF_NAMES       'N'  /* OLD GNU for names > 100 characters*/
+     #define   LF_SPARSE      'S'  /* This is for sparse files        */
+     #define   LF_VOLHDR      'V'  /* tape/volume header Ignore on extraction */
+     #define   LF_VU_XHDR     'X'  /* POSIX.1-2001 xtended (Sun VU version) */
+
+     /*
+      * Definitions for the t_mode field
+      */
+     #define   TSUID     04000     /* Set UID on execution  */
+     #define   TSGID     02000     /* Set GID on execution  */
+     #define   TSVTX     01000     /* On directories, restricted deletion flag */
+
+Joerg Schilling       Last change: 05/10/19                     2
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+     #define   TUREAD    00400     /* Read by owner         */
+     #define   TUWRITE   00200     /* Write by owner special */
+     #define   TUEXEC    00100     /* Execute/search by owner */
+     #define   TGREAD    00040     /* Read by group         */
+     #define   TGWRITE   00020     /* Write by group        */
+     #define   TGEXEC    00010     /* Execute/search by group */
+     #define   TOREAD    00004     /* Read by other         */
+     #define   TOWRITE   00002     /* Write by other        */
+     #define   TOEXEC    00001     /* Execute/search by other */
+
+     #define   TALLMODES 07777     /* The low 12 bits       */
+
+     /*
+      * This is the ustar (Posix 1003.1) header.
+      */
+     struct header {
+          char t_name[NAMSIZ];     /*   0 Filename               */
+          char t_mode[8];          /* 100 Permissions            */
+          char t_uid[8];           /* 108 Numerical User ID      */
+          char t_gid[8];           /* 116 Numerical Group ID     */
+          char t_size[12];         /* 124 Filesize               */
+          char t_mtime[12];        /* 136 st_mtime               */
+          char t_chksum[8];        /* 148 Checksum               */
+          char t_typeflag;         /* 156 Typ of File            */
+          char t_linkname[NAMSIZ]; /* 157 Target of Links        */
+          char t_magic[TMAGLEN];   /* 257 "ustar"                */
+          char t_version[TVERSLEN]; /* 263 Version fixed to 00   */
+          char t_uname[TUNMLEN];   /* 265 User Name              */
+          char t_gname[TGNMLEN];   /* 297 Group Name             */
+          char t_devmajor[8];      /* 329 Major for devices      */
+          char t_devminor[8];      /* 337 Minor for devices      */
+          char t_prefix[PFXSIZ];   /* 345 Prefix for t_name      */
+                                   /* 500 End                    */
+          char t_mfill[12];        /* 500 Filler up to 512       */
+     };
+
+     /*
+      * star header specific definitions
+      */
+     #define   STMAGIC        "tar"     /* star magic */
+     #define   STMAGLEN       4         /* "tar" including '\0' */
+
+     /*
+      * This is the new (post Posix 1003.1-1988) xstar header
+      * defined in 1994.
+      *
+      * t_prefix[130]    is guaranteed to be ' ' to prevent ustar
+      *                  compliant implementations from failing.
+      * t_mfill & t_xmagic need to be zero for a 100% ustar compliant
+      *                  implementation, so setting t_xmagic to
+      *                  "tar" should be avoided in the future.
+      *
+
+Joerg Schilling       Last change: 05/10/19                     3
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+      * A different method to recognize this format is to verify that
+      * t_prefix[130]              is equal to ' ' and
+      * t_atime[0]/t_ctime[0]      is an octal number and
+      * t_atime[11]                is equal to ' ' and
+      * t_ctime[11]                is equal to ' '.
+      *
+      * Note that t_atime[11]/t_ctime[11] may be changed in future.
+      */
+     struct xstar_header {
+          char t_name[NAMSIZ];     /*   0 Filename               */
+          char t_mode[8];          /* 100 Permissions            */
+          char t_uid[8];           /* 108 Numerical User ID      */
+          char t_gid[8];           /* 116 Numerical Group ID     */
+          char t_size[12];         /* 124 Filesize               */
+          char t_mtime[12];        /* 136 st_mtime               */
+          char t_chksum[8];        /* 148 Checksum               */
+          char t_typeflag;         /* 156 Typ of File            */
+          char t_linkname[NAMSIZ]; /* 157 Target of Links        */
+          char t_magic[TMAGLEN];   /* 257 "ustar"                */
+          char t_version[TVERSLEN]; /* 263 Version fixed to 00   */
+          char t_uname[TUNMLEN];   /* 265 User Name              */
+          char t_gname[TGNMLEN];   /* 297 Group Name             */
+          char t_devmajor[8];      /* 329 Major for devices      */
+          char t_devminor[8];      /* 337 Minor for devices      */
+          char t_prefix[131];      /* 345 Prefix for t_name      */
+          char t_atime[12];        /* 476 st_atime               */
+          char t_ctime[12];        /* 488 st_ctime               */
+          char t_mfill[8];         /* 500 Filler up to star magic     */
+          char t_xmagic[4];        /* 508 "tar"                  */
+     };
+
+     struct sparse {
+          char t_offset[12];
+          char t_numbytes[12];
+     };
+
+     #define   SPARSE_EXT_HDR  21
+
+     struct xstar_ext_header {
+          struct sparse t_sp[21];
+          char t_isextended;
+     };
+
+     typedef union hblock {
+          char dummy[TBLOCK];
+          long ldummy[TBLOCK/sizeof (long)]; /* force long alignment */
+          struct header            dbuf;
+          struct xstar_header      xstar_dbuf;
+          struct xstar_ext_header  xstar_ext_dbuf;
+     } TCB;
+
+Joerg Schilling       Last change: 05/10/19                     4
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+     For maximum portability, all fields that  contain  character
+     strings should be limited to use the low 7 bits of a charac-
+     ter.
+
+     The  name,  linkname  and  prefix  field  contain  character
+     strings.  The  strings  are null terminated except when they
+     use the full space of 100 characters for the name  or  link-
+     name field or 155 characters for the prefix field.
+
+     If the prefix does not start with  a  null  character,  then
+     prefix and name need to be concatenated by using the prefix,
+     followed a slash character followed by the name field.  If a
+     null  character appears in name or prefix before the maximum
+     size is reached, the field in question is terminated.   This
+     way  file  names  up to 256 characters may be archived.  The
+     prefix is not used together with the linkname field, so  the
+     maximum length of a link name is 100 characters.
+
+     The fields magic, uname and gname  contain  null  terminated
+     character strings.
+
+     The version field contains the string "00" without a  trail-
+     ing  zero.  It cannot be set to different values as POSIX.1-
+     1988 did not specify  a  way  to  handle  different  version
+     strings.  The typeflag field contains a single character.
+
+     All  numeric  fields  contain  size-1  leading   zero-filled
+     numbers  using  octal  digits.   They are followed by one or
+     more space or null characters.  All  recent  implementations
+     only use one space or null character at the end of a numeri-
+     cal field to get maximum space for the octal  number.   Star
+     always uses a space character as terminator.  Numeric fields
+     with 8 characters may hold up to 7  octal  digits  (7777777)
+     which results is a maximum value of 2097151.  Numeric fields
+     with  12  characters  may  hold  up  to  11   octal   digits
+     (77777777777)   which   results   is   a  maximum  value  of
+     8589934591.
+
+     Star implements  a  vendor  specific  (and  thus  non-POSIX)
+     extension  to  put  bigger  numbers into the numeric fields.
+     This is done by using a base 256 coding.  The top bit of the
+     first character in the appropriate 8 character or 12 charac-
+     ter field is set to flag non octal coding.  If base 256 cod-
+     ing  is  in  use,  then all remaining characters are used to
+     code the number. This results in 7  base  256  digits  in  8
+     character  fields  and in 11 base 256 digits in 12 character
+     fields.  All base 256 numbers are two's complement  numbers.
+     A base 256 number in a 8 character field may hold 56 bits, a
+     base 256 number in a 12 character field may  hold  88  bits.
+     This  may  extended to 64 bits for 8 character fields and to
+     95 bits for 12 character fields. For a negative  number  the
+     first  character  currently  is set to a value of 255 (all 8
+
+Joerg Schilling       Last change: 05/10/19                     5
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+     bits are set).  The rightmost character in a 8 or 12 charac-
+     ter  field  contains  the least significant base 256 number.
+     Recent GNU tar versions implement the same extension.
+
+     While the POSIX standard makes obvious that the fields mode,
+     uid,  gid,    size,  chksum, devmajor and devminor should be
+     treated as unsigned numbers, there is no such definition for
+     the time field.
+
+     The mode field contains 12  bits  holding  permissions,  see
+     above for the definitions for each of the permission bits.
+
+     The uid and gid fields contain the numerical user id of  the
+     file.
+
+     The size field contains the size of the file in  characters.
+     If  the tar header is followed by file data, then the amount
+     of data that follows is computed by (size + 511) / 512.
+
+     The mtime filed contains the number of seconds since Jan 1st
+     1970 00:00 UTC as retrived via stat(2) in st_mtime.
+
+     The chksum field contains a simple checksum over  all  bytes
+     of  the header.  To compute the value, all characters in the
+     header are treated as unsigned integers and  the  characters
+     in  the chksum field are treated as if they were all spaces.
+     When the computation starts, the checksum value is  initial-
+     ized to 0.
+
+     The typeflag field specifies the type of the  file  that  is
+     archived.  If a specific tar implementation does not include
+     support for a specific typeflag value,  this  implementation
+     will  extract  the  unknown file types as if they were plain
+     files.
+
+     '0' REGTYPE
+          A regular file.  If the size field is  non  zero,  then
+          file data follows the header.
+
+     '\0' AREGTYPE
+          For backwards compatibility with pre  POSIX.1-1988  tar
+          implementations,  a nul character is also recognized as
+          marker for plain files.  It is not generated by  recent
+          tar  implementations.   If  the size field is non zero,
+          then file data follows the header.
+
+     '1' LNKTYPE
+          The file is a hard link to another file.  The  name  of
+          the  file that the file is linked to is in the linkname
+          part of the header.  For tar archives  written  by  pre
+          POSIX.1-1988  implementations,  the  size field usually
+          contains the size of the file and needs to  be  ignored
+
+Joerg Schilling       Last change: 05/10/19                     6
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          as  no  data may follow this header type.  For POSIX.1-
+          1988 compliant archives, the size field needs to be  0.
+          For POSIX.1-2001 compliant archives, the size field may
+          be non zero, indicating that file data is  included  in
+          the archive.
+
+     '2' SYMTYPE
+          The file is a symbolic link to another file.  The  name
+          of  the file that the file is linked to is in the link-
+          name part of the header.  The size field needs to be 0.
+          No file data may follow the header.
+
+     '3' CHRTYPE
+          A character special file.  The fields devmajor and dev-
+          minor  contain  information that defines the file.  The
+          meaning of the size field is unspecified by  the  POSIX
+          standard.  No file data may follow the header.
+
+     '4' BLKTYPE
+          A block special file.  The fields devmajor and devminor
+          contain information that defines the file.  The meaning
+          of the size field is unspecified by the POSIX standard.
+          No file data may follow the header.
+
+     '5' DIRTYPE
+          A directory or sub directory.  Old  (pre  POSIX.1-1988)
+          tar  implementations did use the same typeflag value as
+          for plain files and added a slash to the name.  If  the
+          size  field  is  non zero then it indicates the maximum
+          size in characters the system  may  allocate  for  this
+          directory.  If  the  size  field  is 0, then the system
+          shall not limit the size of the directory. On operating
+          systems  where  the  disk  allocation  is not done on a
+          directory base, the size field is  ignored  on  extrac-
+          tion.  No file data may follow the header.
+
+     '6' FIFOTYPE
+          A named  pipe.   The  meaning  of  the  size  field  is
+          unspecified by the POSIX standard.  The size field must
+          be ignored on extraction.  No file data may follow  the
+          header.
+
+     '7' CONTTYPE
+          A contiguous file.  This is a file that  gives  special
+          performance  attributes.   Operating systems that don't
+          support this file type extract this file type as  plain
+          files.   If  the size field is non zero, then file data
+          follows the header.
+
+     'g' GLOBAL POSIX.1-2001 HEADER
+          With POSIX.1-2001 pax archives,  this  type  defines  a
+          global  extended  header.   The size is always non zero
+
+Joerg Schilling       Last change: 05/10/19                     7
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          and denotes  the  sum  of  the  length  fields  in  the
+          extended header data.  The data that follows the header
+          is in the pax extended  header  format.   The  extended
+          header records in this header type affect all following
+          files in the archive unless they are overwritten by new
+          values.   See  EXTENDED TAR (PAX) HEADER FORMAT section
+          below.
+
+     'x' EXTENDED POSIX.1-2001 HEADER
+          With POSIX.1-2001 pax archives, this  type  defines  an
+          extended  header.   The  size  is  always  non zero and
+          denotes the sum of the length fields  in  the  extended
+          header  data.   The  data that follows the header is in
+          the pax extended header format.   The  extended  header
+          records  in  this header type only affect the following
+          file in the archive.  See  EXTENDED  TAR  (PAX)  HEADER
+          FORMAT section below.
+
+     'A' - 'Z'
+          Reserved for vendor specific implementations.
+
+     'A'  A Solaris ACL entry as used by the  tar  implementation
+          from  Sun.  The size is always non zero and denotes the
+          length of the  data  that  follows  the  header.   Star
+          currently is not able to handle this header type.
+
+     'D'  A GNU dump directory.  This header type is not  created
+          by  star and handled like a directory during an extract
+          operation, so the content is ignored by star.  The size
+          field  denotes  the length of the data that follows the
+          header.
+
+     'E'  A Solaris Extended  Attribute  File.   The  size  field
+          denotes the length of the data that follows the header.
+          Star currently is not able to handle this header type.
+
+     'I'  A inode metadata entry.  This header type  is  used  by
+          star  to archive inode meta data only.  To archive more
+          inode meta data than possible with a  POSIX-1.1988  tar
+          header, a header with type 'I' is usually preceded by a
+          'x' header.  It is used with incremental backups.   The
+          size  field holds the length of the file.  No file data
+          follows this header.
+
+     'K'  A long link name.  Star is able to read and write  this
+          type  of  header.  With the xustar and exustar formats,
+          star  prefers  to  store  long  link  names  using  the
+          POSIX.1-2001  method.   The size is always non zero and
+          denotes the length of the long link name including  the
+          trailing  null  byte. The link name is in the data that
+          follows the header.
+
+Joerg Schilling       Last change: 05/10/19                     8
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+     'L'  A long file name.  Star is able to read and write  this
+          type  of  header.  With the xustar and exustar formats,
+          star  prefers  to  store  long  file  names  using  the
+          POSIX.1-2001  method.   The size is always non zero and
+          denotes the length of the long file name including  the
+          trailing  null  byte. The file name is in the data that
+          follows the header.
+
+     'M'  A multi volume continuation entry.  It is used by  star
+          to  tell the extraction program via the size field when
+          the next regular  archive  header  will  follow.   This
+          allows to start extracting multi volume archives with a
+          volume number greater than one.  It is used by GNU  tar
+          to  verify  multi  volume  continuation volumes.  Other
+          fields in the GNU multi volume continuation header  are
+          a  result  of  a  GNU tar miss conception and cannot be
+          used.  If the size field is non zero the data following
+          the header is skipped by star if the volume that starts
+          with it is mounted as the first volume.  This header is
+          ignored if the volume that starts with it is mounted as
+          continuation volume.
+
+     'N'  An outdated linktype used by old GNU  tar  versions  to
+          store  long  file  names.   This type is unsupported by
+          star.
+
+     'S'  A sparse file.  This header type is used  by  star  and
+          GNU  tar.   A  sparse header is uses instead of a plain
+          file header to  denote  a  sparse  file  that  follows.
+          Directly  after  the  header,  a  list  of  sparse hole
+          descriptors follows  followed  by  the  compacted  file
+          data.   With  star formats, the size field holds a size
+          that represents the sum of the sparse hole  descriptors
+          plus  the  size of the compacted file data. This allows
+          other tar implementations to correctly skip to the next
+          tar header.  With GNU tar, up to 4 sparse hole descrip-
+          tors fit  into  the  sparse  header.   Additional  hole
+          descriptors  are not needed if the file has less than 4
+          holes.  With GNU tar, the size field breaks general tar
+          header rules and is meaningless because the size of the
+          sparse hole descriptors does not count.
+
+     'V'  A volume header.  The name field is is used to hold the
+          volume  name.   Star  uses  the atime field to hold the
+          volume number in case there is no POSIX.1-2001 extended
+          header.   This header type is used by star and GNU tar.
+          If the size field is non zero the  data  following  the
+          header is skipped by star.
+
+     'X'  A vendor unique variant of  the  POSIX.1-2001  extended
+          header type.  It has been implemented by Sun many years
+          before the POSIX.1-2001  standard  has  been  approved.
+
+Joerg Schilling       Last change: 05/10/19                     9
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          See also the typeflag 'x' header type.  Star is able to
+          read and write this type of header.
+
+
+
+

EXTENDED TAR (PAX) HEADER STRUCTURE

+     Block type                            Description
+
+     Ustar Header [typeflag='g']      Global Extended Header
+     Global Extended Data
+     Ustar Header [typeflag='h']         Extended Header
+     Extended Data
+     Ustar header [typeflag='0']    File with Extended Header
+     Data for File #1
+     Ustar header [typeflag='0']   File without Extended Header
+     Data for File #2
+     Block of binary zeroes              First EOF Block
+     Block of binary zeroes              Second EOF Block
+
+
+
+

EXTENDED TAR (PAX) HEADER FORMAT

+     The data block  that  follows  a  tar  archive  header  with
+     typeflag 'g' or 'x' contains one or more records in the fol-
+     lowing format:
+
+          "%d %s=%s\n", <length>, <keyword>, <value>
+
+     Each record starts with a a decimal length field. The length
+     includes  the  total  size  of a record including the length
+     field itself and the trailing new line.
+
+     The keyword may not include an  equal  sign.   All  keywords
+     beginning  with  lower  case letters and digits are reserved
+     for future use by the POSIX standard.
+
+     If the value field is of zero length, it deletes any  header
+     field  of  the  same  name  that  is in effect from the same
+     extended header or from a previous global header.
+
+     Null characters do not delimit any value. The value is  only
+     limited by its implicit length.
+
+
+
+

EXTENDED TAR (PAX) HEADER KEYWORDS

+     POSIX.1-2001 extended pax  header  keywords.  All  numerical
+     values  are  represented  as decimal strings.  All texts are
+     represented as 7-bit ascii or UTF-8:
+
+     atime
+          The time from st_atime in sub second granularity.  Star
+          currently supports a nanosecond granularity.
+
+     charset
+          The name of the character set used to encode  the  data
+          in  the  following  file(s).  This keyword is currently
+
+Joerg Schilling       Last change: 05/10/19                    10
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          ignored by star.
+
+     comment
+          Any number of characters that  should  be  treated  as
+          comment.  Star ignores the comment as documented by the
+          POSIX standard.
+
+     ctime
+          The time from st_ctime in sub second granularity.  Star
+          currently supports a nanosecond granularity.
+
+     gid  The group ID of the group  that  owns  the  file.   The
+          argument  is  a  decimal number.  This field is used if
+          the group ID of a file is greater than  2097151  (octal
+          7777777).
+
+     gname
+          The group name of the following file(s) coded in  UTF-8
+          if  the  group name does not fit into 323 characters or
+          cannot be expressed in 7-Bit ASCII.
+
+     linkpath
+          The name of the linkpath coded in UTF-8 if it is longer
+          than  100  characters  or  cannot be expressed in 7-Bit
+          ASCII.
+
+     mtime
+          The time from st_mtime in sub second granularity.  Star
+          currently supports a nanosecond granularity.
+
+     path The name of the linkpath coded in UTF-8 if it does  not
+          fit into 100 characters + 155 characters prefix or can-
+          not be expressed in 7-Bit ASCII.
+
+     realtime.any
+          The keywords prefixed by  realtime.  are  reserved  for
+          future standardization.
+
+     security.any
+          The keywords prefixed by  security.  are  reserved  for
+          future standardization.
+
+     size The size of the file as decimal number if the file size
+          is  greater  than  8589934591  (octal 77777777777). The
+          size keyword may not refer to the real file size but is
+          related  to  the  size if the file in the archive.  See
+          also SCHILY.realsize for more information.
+
+     uid  The uid ID of the group that owns the file.  The  argu-
+          ment  is  a  decimal number.  This field is used if the
+          uid ID  of  a  file  is  greater  than  2097151  (octal
+          7777777).
+
+Joerg Schilling       Last change: 05/10/19                    11
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+     uname
+          The user name of the following file(s) coded  in  UTF-8
+          if  the  user  name does not fit into 323 characters or
+          cannot be expressed in 7-Bit ASCII.
+
+     VENDOR.keyword
+          Any keyword that starts with a vendor name  in  capital
+          letters  is  reserved for vendor specific extensions by
+          the standard.  Star uses a lot of these vendor specific
+          extension. See below for more informations.
+
+
+
+

SCHILY PAX EXTENSION KEYWORDS

+     Star uses own vendor specific extensions. The SCHILY  vendor
+     specific extended pax header keywords are:
+
+     SCHILY.acl.access
+          The ACL for a file.
+
+          Since no official backup format for POSIX  access  con-
+          trol  lists  has  been  defined,  star  uses the vendor
+          defined      attributes      SCHILY.acl.access      and
+          SCHILY.acl.default  for storing the ACL and Default ACL
+          of a file, respectively.  The access control lists  are
+          stored  in  the  short  text  form  as defined in POSIX
+          1003.1e draft standard 17.
+
+          To each named user ACL entry a fourth  colon  separated
+          field field containing the user identifier (UID) of the
+          associated user is appended.  To each named group entry
+          a  fourth  colon  separated  field containing the group
+          identifier (GID) of the associated group  is  appended.
+          (POSIX  1003.1e  draft standard 17 allows to add fields
+          to ACL entries.)
+
+          This  is  an   example   of   the   format   used   for
+          SCHILY.acl.access  (a space has been inserted after the
+          equal sign and lines are broken [marked with '\' ]  for
+          readability, additional fields in bold):
+
+          SCHILY.acl.access= user::rwx,user:lisa:r-x:502, \
+                             group::r-x,group:toolies:rwx:102, \
+                             mask::rwx,other::r--x
+
+          The numerical user and group identifiers are  essential
+          when  restoring  a  system completely from a backup, as
+          initially the name-to-identifier mappings  may  not  be
+          available,  and  then  file ownership restoration would
+          not work.
+
+          As the archive format  that  is  used  for  backing  up
+          access control lists is compatible with the pax archive
+          format, archives created that way can  be  restored  by
+
+Joerg Schilling       Last change: 05/10/19                    12
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          star  or  a POSIX.1-2001 compliant pax.  Note that pro-
+          grams other than star will ignore the ACL information.
+
+     SCHILY.acl.default
+          The default ACL for a file. See  SCHILY.acl.access  for
+          more information.
+
+          This  is  an   example   of   the   format   used   for
+          SCHILY.acl.default (a space has been inserted after the
+          equal sign and lines are broken [marked with '\' ]  for
+          readability, additional fields in bold):
+
+          SCHILY.acl.default= user::rwx,user:lisa:r-x:502, \
+                              group::r-x,mask::r-x,other::r-x
+
+     SCHILY.ddev
+          The device ids for names used is  the  SCHILY.dir  dump
+          directory  list  from  st_dev  of  the  file as decimal
+          number.  The SCHILY.ddev keyword is followed by a space
+          separated  list  of device id numbers. Each corresponds
+          exactly to a name in the list found in SCHILY.dir.   If
+          a  specific  device  id number is repeated, a comma (,)
+          without a following space may be use to denote that the
+          current  device  id number is identical to the previous
+          number.  This keyword is used in dump mode.  This  key-
+          word is not yet implemented.
+
+          The value is a signed int.  An implementation should be
+          able  to  handle at least 64 bit values.  Note that the
+          value is signed because POSIX  does  not  specify  more
+          than the type should be an int.
+
+     SCHILY.dev
+          The device id  from  st_dev  of  the  file  as  decimal
+          number.  This keyword is used in dump mode.
+
+          The value is a signed int.  An implementation should be
+          able  to  handle at least 64 bit values.  Note that the
+          value is signed because POSIX  does  not  specify  more
+          than the type should be an int.
+
+     SCHILY.devmajor
+          The device major number of the file if it is a  charac-
+          ter  or  block special file.  The argument is a decimal
+          number.  This field is used if the device major of  the
+          file is greater than 2097151 (octal 7777777).
+
+          The value is a signed int.  An implementation should be
+          able  to  handle at least 64 bit values.  Note that the
+          value is signed because POSIX  does  not  specify  more
+          than the type should be an int.
+
+Joerg Schilling       Last change: 05/10/19                    13
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+     SCHILY.devminor
+          The device minor number of the file if it is a  charac-
+          ter  or  block special file.  The argument is a decimal
+          number.  This field is used if the device minor of  the
+          file is greater than 2097151 (octal 7777777).
+
+          The value is a signed int.  An implementation should be
+          able  to  handle at least 64 bit values.  Note that the
+          value is signed because POSIX  does  not  specify  more
+          than the type should be an int.
+
+     SCHILY.dino
+          The inode numbers for names used is the SCHILY.dir dump
+          directory  list  from  st_ino  of  the  file as decimal
+          number.  The SCHILY.dino keyword is followed by a space
+          separated  list  of  inode  numbers.  Each  corresponds
+          exactly to a name in  the  list  found  in  SCHILY.dir.
+          This keyword is used in dump mode.
+
+          The values are unsigned int.  An implementation  should
+          be able to handle at least 64 bit unsigned values.
+
+     SCHILY.dir
+          A list of  filenames  (the  content)  for  the  current
+          directory.   The  names  are coded in UTF-8.  Each file
+          name is prefixed by a single character that is used  as
+          a flag.  Each file name is limited by a null character.
+          The null character is  directly  followed  by  he  flag
+          character  for  the  next file name in case the list is
+          not terminated by the  current  file  name.   The  flag
+          character  must not be a null character.  By default, a
+          ^A (octal  001)  is  used.   The  following  flags  are
+          defined:
+
+          \000 This is the list terminator character - the second
+               null byte, see below.
+
+          ^A   The default flag that is used in case the dump dir
+               features have not been active.
+
+          Y    A non  directory  file  that  is  in  the  current
+               (incremental) dump.
+
+          N    A non directory file that is not  in  the  current
+               (incremental) dump.
+
+          D    A directory that is in the  current  (incremental)
+               dump.
+
+          d    A directory that is not in the current  (incremen-
+               tal) dump.
+
+Joerg Schilling       Last change: 05/10/19                    14
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          The list is terminated by two  successive  null  bytes.
+          The first is the null byte for the last file name.  The
+          second null byte is at the position where a flag  char-
+          acter  would be expected, it acts ad a list terminator.
+          The length tag for the SCHILY.dir  data  includes  both
+          null bytes.
+
+          If a dump mode has been selected  that  writes  compact
+          complete  directory information to the beginning of the
+          archive, the flag character  may  contain  values  dif-
+          ferent from ^A.  Star implementations up to star-1.5 do
+          not include this  feature.   Tar  implementations  that
+          like  to read archives that use the SCHILY.dir keyword,
+          shall not rely on values other than \000 (^@)  or  \001
+          (^A).
+
+          This keyword is used in dump mode.
+
+     SCHILY.fflags
+          A textual version of the BSD  or  Linux  extended  file
+          flags.  As this tag has not yet been documented, please
+          look into the  star  source,  file  fflags.c  for  more
+          information.
+
+     SCHILY.filetype
+          A textual version of the real file type  of  the  file.
+          The following names are used:
+
+          unallocated             An unknown file type  that  may
+                                  be  a  result  of  a  unlink(2)
+                                  operation.  This  should  never
+                                  happen.
+
+          regular                 A regular file.
+
+          contiguous              A contiguous file. On operating
+                                  systems  or  file  systems that
+                                  don't support this  file  type,
+                                  it  is  handled  like a regular
+                                  file.
+
+          symlink                 A symbolic  link  to  any  file
+                                  type.
+
+          directory               A directory.
+
+          character special       A character special file.
+
+          block special           A block special file.
+
+          fifo                    A named pipe.
+
+Joerg Schilling       Last change: 05/10/19                    15
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          socket                  A UNIX domain socket.
+
+          mpx character special   A multiplexed character special
+                                  file.
+
+          mpx block special       A  multiplexed  block   special
+                                  file.
+
+          XENIX nsem              A XENIX named semaphore.
+
+          XENIX nshd              XENIX shared data.
+
+          door                    A Solaris door.
+
+          eventcount              A UNOS event count.
+
+          whiteout                A BSD whiteout directory entry.
+
+          sparse                  A sparse regular file.
+
+          volheader               A volume header.
+
+          unknown/bad             Any other  unknown  file  type.
+                                  This should never happen.
+
+     SCHILY.ino
+          The inode number from st_ino of  the  file  as  decimal
+          number.  This keyword is used in dump mode.
+
+          The value is an unsigned int.  An implementation should
+          be able to handle at least 64 bit unsigned values.
+
+     SCHILY.nlink
+          The link count of the file  as  decimal  number.   This
+          keyword is used in dump mode.
+
+          The value is an unsigned int.  An implementation should
+          be able to handle at least 32 bit unsigned values.
+
+     SCHILY.offset
+          The  offset  value  for  a  multi  volume  continuation
+          header.   This  keyword  is used with multi volume con-
+          tinuation headers.  Multi volume  continuation  headers
+          are  used  to  allow  to  start  reading a multi volume
+          archive past the first volume.
+
+          The value is an unsigned int.  An implementation should
+          be able to handle at least 64 bit unsigned values.
+
+     SCHILY.realsize
+          The real size of the  file  as  decimal  number.   This
+
+Joerg Schilling       Last change: 05/10/19                    16
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          keyword  is  used  if the real size of the file differs
+          from the visible size of the file in the archive.   The
+          real  file size differs from the size in the archive if
+          the file type is sparse or if the file is  a  continua-
+          tion  file  on  a  multi  volume  archive.  In case the
+          SCHILY.realsize keyword is needed, it must be past  any
+          size keyword in case a size keyword is also present.
+
+          The value is an unsigned int.  An implementation should
+          be able to handle at least 64 bit unsigned values.
+
+     SCHILY.tarfiletype
+          The  following  additional  file  types  are  used   in
+          SCHILY.tarfiletype:
+
+          hardlink
+               A hard link to any file type.
+
+          dumpdir
+               A directory with dump entries
+
+          multivol continuation
+               A multi volume continuation for any file type.
+
+          meta A meta entry (inode meta data only) for  any  file
+               type.
+
+     SCHILY.xattr.attr
+          A POSIX.1-2001 coded version of the Linux extended file
+          attributes.    Linux   extended   file  attributes  are
+          name/value pairs. Every attribute  name  results  in  a
+          SCHILY.xattr.name  tag  and  the  value of the extended
+          attribute is used as  the  value  of  the  POSIX.1-2001
+          header  tag.  Note that this way of coding is not port-
+          able across  platforms.   A  version  for  BSD  may  be
+          created  but  Solaris  includes  far more features with
+          extended attribute files than Linux does.
+
+          A future version  of  star  will  implement  a  similar
+          method  as  the  tar program on Solaris currently uses.
+          When    this    implementation    is     ready,     the
+          SCHILY.xattr.name  feature may be removed in favor of a
+          truly portable  implementation  that  supports  Solaris
+          also.
+
+
+
+

SCHILY 'G'LOBAL PAX EXTENSION KEYWORDS

+     The following star vendor unique extensions may only  appear
+     in 'g'lobal extended pax headers:
+
+     SCHILY.archtype
+          The textual version of  the  archive  type  used.   The
+
+Joerg Schilling       Last change: 05/10/19                    17
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          textual  values  used  for SCHILY.archtype are the same
+          names that are used in the star command line options to
+          set up a specific archive type.
+
+          In order to allow archive type  recognition  from  this
+          keyword,  the  minimum  tape  block  size must be 2x512
+          bytes (1024  bytes)  and  the  SCHILY.archtype  keyword
+          needs  to  be  in the first 512 bytes of the content of
+          the first 'g'lobal pax  header.  Then  the  first  tape
+          block may be scanned to recognize the archive type.
+
+     SCHILY.release
+          The textual version of the star version string and  the
+          platform  name  where this star has been compiled.  The
+          same text appears when calling star -version.
+
+     SCHILY.volhdr.blockoff
+          This keyword is used for  multi  volume  archives.   It
+          represents   the   offset   within  the  whole  archive
+          expressed in 512 byte units.
+
+          The value is an unsigned int with a valid range between
+          1  and  infinity.  An  implementation should be able to
+          handle at least 64 bit unsigned values.
+
+     SCHILY.volhdr.blocksize
+          The tape blocksize expressed in 512 byte units that was
+          used when writing the archive.
+
+          The value is an unsigned int with a valid range between
+          1  and  infinity.  An  implementation should be able to
+          handle at least 31 bit unsigned values.
+
+     SCHILY.volhdr.cwd
+          This keyword is used in dump mode.  It is only used  to
+          contain  the  real  backup  working  directory  if  the
+          fs-name= option  of  star  is  used  to  overwrite  the
+          SCHILY.volhdr.filesys         value.        Overwriting
+          SCHILY.volhdr.filesys is needed when backups are run on
+          file system snapshots rather than on the real file sys-
+          tem.
+
+     SCHILY.volhdr.device
+          This keyword is used in dump mode.  It  represents  the
+          name of the device that holds the file system data. For
+          disk based file systems, this is the device name of the
+          mounted device.
+
+          This keyword is optional. It helps to  correctly  iden-
+          tify  the  file  system  from  which this dump has been
+          made.
+
+Joerg Schilling       Last change: 05/10/19                    18
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+     SCHILY.volhdr.dumpdate
+          This keyword is used in dump mode.  It  represents  the
+          time the current dump did start.
+
+     SCHILY.volhdr.dumplevel
+          This keyword is used in dump mode.  It  represents  the
+          level  of  the  current  dump.   Dump  levels are small
+          numbers, the lowest possible number is 0.  Dump level 0
+          represents  a  full  backup.  Dump level 1 represents a
+          backup that contains all changes that did  occur  since
+          the  last  level  0  dump.   Dump  level 2 represents a
+          backup that contains all changes that did  occur  since
+          the last level 1 dump.  Star does not specify a maximum
+          allowed dump level but  you  should  try  to  keep  the
+          numbers less than 100.
+
+          The value is an unsigned int with a valid range between
+          0 and at least 100.
+
+     SCHILY.volhdr.dumptype
+          This keyword is used in dump mode.  If the  dump  is  a
+          complete  dump  of  a file system, then the argument is
+          the text full, else the argument is the text partial.
+
+     SCHILY.volhdr.filesys
+          This keyword is used in dump mode.  It  represents  the
+          top level directory for the file system from which this
+          dump has been made.  If the dump represents a dump that
+          has  an associated level, then the this directory needs
+          to be identical to the root directory of this file sys-
+          tem which is the mount point.
+
+     SCHILY.volhdr.hostname
+          This keyword is  used  in  dump  mode.   The  value  is
+          retrieved from gethostname(3) or uname(2).
+
+     SCHILY.volhdr.label
+          The textual volume label.  The  volume  label  must  be
+          identical within a set of multi volume archives.
+
+     SCHILY.volhdr.refdate
+          This keyword is used in dump mode if the  current  dump
+          is an incremental dump with a level > 0.  It represents
+          the time the related dump did start.
+
+     SCHILY.volhdr.reflevel
+          This keyword is used in dump mode if the  current  dump
+          is an incremental dump with a level > 0.  It represents
+          the level of the related dump.  The related dump is the
+          last  dump with a level that is lower that the level of
+          this dump.  If a dump with the  level  of  the  current
+          dump  -1  exists,  then this is the related dump level.
+
+Joerg Schilling       Last change: 05/10/19                    19
+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+          Otherwise, the dump level is decremented until a  valid
+          dump level could be found in the dump database.
+
+          The value is an unsigned int with a valid range between
+          0 and at least 100.
+
+     SCHILY.volhdr.tapesize
+          This keyword is used for multi volume archives and  may
+          be  used  to  verify  the volume size on read back.  It
+          represents the tape size expressed in 512  byte  units.
+          If  this  keyword is set in multi volume mode, the size
+          of the tape is not autodetected but set from a  command
+          line option.
+
+          The value is an unsigned int with a valid range between
+          1  and  infinity.  An  implementation should be able to
+          handle at least 64 bit unsigned values.
+
+     SCHILY.volhdr.volume
+          This keyword is used for  multi  volume  archives.   It
+          represents  the volume number within a volume set.  The
+          number used for the first volume is 1.
+
+          The value is an unsigned int with a valid range between
+          1  and  infinity.  An  implementation should be able to
+          handle at least 31 bit unsigned values.
+
+
+
+

MULTI VOLUME ARCHIVE HANDLING

+     To be documented in the future.
+
+
+
+

SEE ALSO

+
+
+

NOTES

+
+
+

BUGS

+
+
+

AUTHOR

+
+Joerg Schilling       Last change: 05/10/19                    20
+
+
+
+
+Man(1) output converted with +man2html +
+


+FhG +FhG FOKUS +BerliOS + +Schily +Schily's Home +VED powered + + + diff --git a/deps/npm/node_modules/tar/test/fixtures/packtest/Ω.txt b/deps/npm/node_modules/tar/test/fixtures/packtest/Ω.txt new file mode 100644 index 0000000000..1ca042fff2 --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/packtest/Ω.txt @@ -0,0 +1 @@ +Ω \ No newline at end of file diff --git a/deps/npm/node_modules/tar/test/fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc b/deps/npm/node_modules/tar/test/fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc new file mode 100644 index 0000000000..5a5d18e294 --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc @@ -0,0 +1 @@ +cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc \ No newline at end of file diff --git a/deps/npm/node_modules/tar/test/fixtures/symlink b/deps/npm/node_modules/tar/test/fixtures/symlink new file mode 120000 index 0000000000..218c28e640 --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/symlink @@ -0,0 +1 @@ +hardlink-1 \ No newline at end of file diff --git a/deps/npm/node_modules/tar/test/fixtures/Ω.txt b/deps/npm/node_modules/tar/test/fixtures/Ω.txt new file mode 100644 index 0000000000..1ca042fff2 --- /dev/null +++ b/deps/npm/node_modules/tar/test/fixtures/Ω.txt @@ -0,0 +1 @@ +Ω \ No newline at end of file diff --git a/deps/npm/node_modules/tar/test/header.js b/deps/npm/node_modules/tar/test/header.js new file mode 100644 index 0000000000..8ea6f79500 --- /dev/null +++ b/deps/npm/node_modules/tar/test/header.js @@ -0,0 +1,183 @@ +var tap = require("tap") +var TarHeader = require("../lib/header.js") +var tar = require("../tar.js") +var fs = require("fs") + + +var headers = + { "a.txt file header": + [ "612e747874000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000030303036343420003035373736312000303030303234200030303030303030303430312031313635313336303333332030313234353100203000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" + , { cksumValid: true + , path: 'a.txt' + , mode: 420 + , uid: 24561 + , gid: 20 + , size: 257 + , mtime: 1319493851 + , cksum: 5417 + , type: '0' + , linkpath: '' + , ustar: 'ustar\0' + , ustarver: '00' + , uname: 'isaacs' + , gname: 'staff' + , devmaj: 0 + , devmin: 0 + , fill: '' } + ] + + , "omega pax": // the extended header from omega tar. + [ "5061784865616465722fcea92e74787400000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000030303036343420003035373736312000303030303234200030303030303030303137302031313534333731303631312030313530353100207800000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" + , { cksumValid: true + , path: 'PaxHeader/Ω.txt' + , mode: 420 + , uid: 24561 + , gid: 20 + , size: 120 + , mtime: 1301254537 + , cksum: 6697 + , type: 'x' + , linkpath: '' + , ustar: 'ustar\0' + , ustarver: '00' + , uname: 'isaacs' + , gname: 'staff' + , devmaj: 0 + , devmin: 0 + , fill: '' } ] + + , "omega file header": + [ "cea92e7478740000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000030303036343420003035373736312000303030303234200030303030303030303030322031313534333731303631312030313330373200203000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" + , { cksumValid: true + , path: 'Ω.txt' + , mode: 420 + , uid: 24561 + , gid: 20 + , size: 2 + , mtime: 1301254537 + , cksum: 5690 + , type: '0' + , linkpath: '' + , ustar: 'ustar\0' + , ustarver: '00' + , uname: 'isaacs' + , gname: 'staff' + , devmaj: 0 + , devmin: 0 + , fill: '' } ] + + , "foo.js file header": + [ "666f6f2e6a730000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000030303036343420003035373736312000303030303234200030303030303030303030342031313534333637303734312030313236313700203000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" + , { cksumValid: true + , path: 'foo.js' + , mode: 420 + , uid: 24561 + , gid: 20 + , size: 4 + , mtime: 1301246433 + , cksum: 5519 + , type: '0' + , linkpath: '' + , ustar: 'ustar\0' + , ustarver: '00' + , uname: 'isaacs' + , gname: 'staff' + , devmaj: 0 + , devmin: 0 + , fill: '' } + ] + + , "b.txt file header": + [ "622e747874000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000030303036343420003035373736312000303030303234200030303030303030313030302031313635313336303637372030313234363100203000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000757374617200303069736161637300000000000000000000000000000000000000000000000000007374616666000000000000000000000000000000000000000000000000000000303030303030200030303030303020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" + , { cksumValid: true + , path: 'b.txt' + , mode: 420 + , uid: 24561 + , gid: 20 + , size: 512 + , mtime: 1319494079 + , cksum: 5425 + , type: '0' + , linkpath: '' + , ustar: 'ustar\0' + , ustarver: '00' + , uname: 'isaacs' + , gname: 'staff' + , devmaj: 0 + , devmin: 0 + , fill: '' } + ] + + , "deep nested file": + [ "636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363636363633030303634342000303537373631200030303030323420003030303030303030313434203131363532313531353333203034333331340020300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000075737461720030306973616163730000000000000000000000000000000000000000000000000000737461666600000000000000000000000000000000000000000000000000000030303030303020003030303030302000722f652f612f6c2f6c2f792f2d2f642f652f652f702f2d2f662f6f2f6c2f642f652f722f2d2f702f612f742f680000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" + , { cksumValid: true, + path: 'r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc' + , mode: 420 + , uid: 24561 + , gid: 20 + , size: 100 + , mtime: 1319687003 + , cksum: 18124 + , type: '0' + , linkpath: '' + , ustar: 'ustar\0' + , ustarver: '00' + , uname: 'isaacs' + , gname: 'staff' + , devmaj: 0 + , devmin: 0 + , fill: '' } + ] + } + +tap.test("parsing", function (t) { + Object.keys(headers).forEach(function (name) { + var h = headers[name] + , header = new Buffer(h[0], "hex") + , expect = h[1] + , parsed = new TarHeader(header) + + // console.error(parsed) + t.has(parsed, expect, "parse " + name) + }) + t.end() +}) + +tap.test("encoding", function (t) { + Object.keys(headers).forEach(function (name) { + var h = headers[name] + , expect = new Buffer(h[0], "hex") + , encoded = TarHeader.encode(h[1]) + + // might have slightly different bytes, since the standard + // isn't very strict, but should have the same semantics + // checkSum will be different, but cksumValid will be true + + var th = new TarHeader(encoded) + delete h[1].block + delete h[1].needExtended + delete h[1].cksum + t.has(th, h[1], "fields "+name) + }) + t.end() +}) + +// test these manually. they're a bit rare to find in the wild +tap.test("parseNumeric tests", function (t) { + var parseNumeric = TarHeader.parseNumeric + , numbers = + { "303737373737373700": 2097151 + , "30373737373737373737373700": 8589934591 + , "303030303036343400": 420 + , "800000ffffffffffff": 281474976710655 + , "ffffff000000000001": -281474976710654 + , "ffffff000000000000": -281474976710655 + , "800000000000200000": 2097152 + , "8000000000001544c5": 1393861 + , "ffffffffffff1544c5": -15383354 } + Object.keys(numbers).forEach(function (n) { + var b = new Buffer(n, "hex") + t.equal(parseNumeric(b), numbers[n], n + " === " + numbers[n]) + }) + t.end() +}) diff --git a/deps/npm/node_modules/tar/test/pack.js b/deps/npm/node_modules/tar/test/pack.js new file mode 100644 index 0000000000..3fc808d04f --- /dev/null +++ b/deps/npm/node_modules/tar/test/pack.js @@ -0,0 +1,953 @@ +var tap = require("tap") + , tar = require("../tar.js") + , pkg = require("../package.json") + , Pack = tar.Pack + , fstream = require("fstream") + , Reader = fstream.Reader + , Writer = fstream.Writer + , path = require("path") + , input = path.resolve(__dirname, "fixtures/") + , target = path.resolve(__dirname, "tmp/pack.tar") + , uid = process.getuid ? process.getuid() : 0 + , gid = process.getgid ? process.getgid() : 0 + + , entries = + + // the global header and root fixtures/ dir are going to get + // a different date each time, so omit that bit. + // Also, dev/ino values differ across machines, so that's not + // included. Rather than use + [ [ 'globalExtendedHeader', + { path: 'PaxHeader/', + mode: 438, + uid: 0, + gid: 0, + type: 'g', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' }, + { "NODETAR.author": pkg.author, + "NODETAR.name": pkg.name, + "NODETAR.description": pkg.description, + "NODETAR.version": pkg.version, + "NODETAR.repository.type": pkg.repository.type, + "NODETAR.repository.url": pkg.repository.url, + "NODETAR.main": pkg.main, + "NODETAR.scripts.test": pkg.scripts.test, + "NODETAR.engines.node": pkg.engines.node } ] + + , [ 'entry', + { path: 'fixtures/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'extendedHeader', + { path: 'PaxHeader/fixtures/200cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: uid, + gid: gid, + size: 402, + mtime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + cksum: 13492, + type: 'x', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' }, + { path: 'fixtures/200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + 'NODETAR.depth': '1', + 'NODETAR.type': 'File', + nlink: 1, + uid: uid, + gid: gid, + size: 200, + 'NODETAR.blksize': '4096', + 'NODETAR.blocks': '8' } ] + + , [ 'entry', + { path: 'fixtures/200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: uid, + gid: gid, + size: 200, + mtime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + cksum: 13475, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '', + 'NODETAR.depth': '1', + 'NODETAR.type': 'File', + nlink: 1, + 'NODETAR.blksize': '4096', + 'NODETAR.blocks': '8' } ] + + , [ 'entry', + { path: 'fixtures/a.txt', + mode: 420, + uid: uid, + gid: gid, + size: 257, + mtime: new Date('Mon, 24 Oct 2011 22:04:11 GMT'), + cksum: 5114, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/b.txt', + mode: 420, + uid: uid, + gid: gid, + size: 512, + mtime: new Date('Mon, 24 Oct 2011 22:07:59 GMT'), + cksum: 5122, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/c.txt', + mode: 420, + uid: uid, + gid: gid, + size: 513, + mtime: new Date('Wed, 26 Oct 2011 01:10:58 GMT'), + cksum: 5119, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/cc.txt', + mode: 420, + uid: uid, + gid: gid, + size: 513, + mtime: new Date('Wed, 26 Oct 2011 01:11:02 GMT'), + cksum: 5222, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/foo.js', + mode: 420, + uid: uid, + gid: gid, + size: 4, + mtime: new Date('Fri, 21 Oct 2011 21:19:29 GMT'), + cksum: 5211, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/hardlink-1', + mode: 420, + uid: uid, + gid: gid, + size: 200, + mtime: new Date('Tue, 15 Nov 2011 03:10:09 GMT'), + cksum: 5554, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/hardlink-2', + mode: 420, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Tue, 15 Nov 2011 03:10:09 GMT'), + cksum: 7428, + type: '1', + linkpath: 'fixtures/hardlink-1', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/omega.txt', + mode: 420, + uid: uid, + gid: gid, + size: 2, + mtime: new Date('Fri, 21 Oct 2011 21:19:29 GMT'), + cksum: 5537, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/packtest/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/packtest/omega.txt', + mode: 420, + uid: uid, + gid: gid, + size: 2, + mtime: new Date('Mon, 14 Nov 2011 21:42:24 GMT'), + cksum: 6440, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/packtest/star.4.html', + mode: 420, + uid: uid, + gid: gid, + size: 54081, + mtime: new Date("Sun, 06 May 2007 13:25:06 GMT"), + cksum: 6566, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'extendedHeader', + { path: 'PaxHeader/fixtures/packtest/Ω.txt', + mode: 420, + uid: uid, + gid: gid, + size: 213, + mtime: new Date('Mon, 14 Nov 2011 21:39:39 GMT'), + cksum: 7306, + type: 'x', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' }, + { path: 'fixtures/packtest/Ω.txt', + 'NODETAR.depth': '2', + 'NODETAR.type': 'File', + nlink: 1, + uid: uid, + gid: gid, + size: 2, + 'NODETAR.blksize': '4096', + 'NODETAR.blocks': '8' } ] + + , [ 'entry', + { path: 'fixtures/packtest/Ω.txt', + mode: 420, + uid: uid, + gid: gid, + size: 2, + mtime: new Date('Mon, 14 Nov 2011 21:39:39 GMT'), + cksum: 6297, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '', + 'NODETAR.depth': '2', + 'NODETAR.type': 'File', + nlink: 1, + 'NODETAR.blksize': '4096', + 'NODETAR.blocks': '8' } ] + + , [ 'entry', + { path: 'fixtures/r/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 4789, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 4937, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 5081, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 5236, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 5391, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 5559, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 5651, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 5798, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 5946, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 6094, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 6253, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 6345, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 6494, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 6652, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 6807, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 6954, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 7102, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 7263, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 7355, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 7514, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 7658, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:42:46 GMT'), + cksum: 7821, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:43:23 GMT'), + cksum: 7967, + type: '5', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: uid, + gid: gid, + size: 100, + mtime: new Date('Thu, 27 Oct 2011 03:43:23 GMT'), + cksum: 17821, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'entry', + { path: 'fixtures/symlink', + mode: 493, + uid: uid, + gid: gid, + size: 0, + mtime: new Date('Tue, 15 Nov 2011 19:57:48 GMT'), + cksum: 6337, + type: '2', + linkpath: 'hardlink-1', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' } ] + + , [ 'extendedHeader', + { path: 'PaxHeader/fixtures/Ω.txt', + mode: 420, + uid: uid, + gid: gid, + size: 204, + mtime: new Date('Thu, 27 Oct 2011 17:51:49 GMT'), + cksum: 6399, + type: 'x', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '' }, + { path: "fixtures/Ω.txt" + , "NODETAR.depth": "1" + , "NODETAR.type": "File" + , nlink: 1 + , uid: uid + , gid: gid + , size: 2 + , "NODETAR.blksize": "4096" + , "NODETAR.blocks": "8" } ] + + , [ 'entry', + { path: 'fixtures/Ω.txt', + mode: 420, + uid: uid, + gid: gid, + size: 2, + mtime: new Date('Thu, 27 Oct 2011 17:51:49 GMT'), + cksum: 5392, + type: '0', + linkpath: '', + ustar: 'ustar\u0000', + ustarver: '00', + uname: '', + gname: '', + devmaj: 0, + devmin: 0, + fill: '', + 'NODETAR.depth': '1', + 'NODETAR.type': 'File', + nlink: 1, + 'NODETAR.blksize': '4096', + 'NODETAR.blocks': '8' } ] + ] + + +// first, make sure that the hardlinks are actually hardlinks, or this +// won't work. Git has a way of replacing them with a copy. +var hard1 = path.resolve(__dirname, "fixtures/hardlink-1") + , hard2 = path.resolve(__dirname, "fixtures/hardlink-2") + , fs = require("fs") + +try { fs.unlinkSync(hard2) } catch (e) {} +fs.linkSync(hard1, hard2) + +tap.test("with global header", { timeout: 10000 }, function (t) { + runTest(t, true) +}) + +tap.test("without global header", { timeout: 10000 }, function (t) { + runTest(t, false) +}) + +function runTest (t, doGH) { + var reader = Reader({ path: input + , filter: function () { + return !this.path.match(/\.(tar|hex)$/) + } + }) + + var pack = Pack(doGH ? pkg : null) + var writer = Writer(target) + + // skip the global header if we're not doing that. + var entry = doGH ? 0 : 1 + + t.ok(reader, "reader ok") + t.ok(pack, "pack ok") + t.ok(writer, "writer ok") + + pack.pipe(writer) + + var parse = tar.Parse() + t.ok(parse, "parser should be ok") + + pack.on("data", function (c) { + // console.error("PACK DATA") + t.equal(c.length, 512, "parser should emit data in 512byte blocks") + parse.write(c) + }) + + pack.on("end", function () { + // console.error("PACK END") + t.pass("parser ends") + parse.end() + }) + + pack.on("error", function (er) { + t.fail("pack error", er) + }) + + parse.on("error", function (er) { + t.fail("parse error", er) + }) + + writer.on("error", function (er) { + t.fail("writer error", er) + }) + + reader.on("error", function (er) { + t.fail("reader error", er) + }) + + parse.on("*", function (ev, e) { + var wanted = entries[entry++] + if (!wanted) { + t.fail("unexpected event: "+ev) + return + } + t.equal(ev, wanted[0], "event type should be "+wanted[0]) + // if (ev !== wanted[0] || e.path !== wanted[1].path) { + // console.error(wanted) + // console.error([ev, e.props]) + // throw "break" + // } + t.has(e.props, wanted[1], "properties "+wanted[1].path) + if (wanted[2]) { + e.on("end", function () { + t.has(e.fields, wanted[2], "should get expected fields") + }) + } + }) + + reader.pipe(pack) + + writer.on("close", function () { + t.equal(entry, entries.length, "should get all expected entries") + t.pass("it finished") + t.end() + }) + +} diff --git a/deps/npm/node_modules/tar/test/parse.js b/deps/npm/node_modules/tar/test/parse.js new file mode 100644 index 0000000000..f765a50129 --- /dev/null +++ b/deps/npm/node_modules/tar/test/parse.js @@ -0,0 +1,359 @@ +var tap = require("tap") + , tar = require("../tar.js") + , fs = require("fs") + , path = require("path") + , file = path.resolve(__dirname, "fixtures/c.tar") + , index = 0 + + , expect = +[ [ 'entry', + { path: 'c.txt', + mode: 420, + uid: 24561, + gid: 20, + size: 513, + mtime: new Date('Wed, 26 Oct 2011 01:10:58 GMT'), + cksum: 5422, + type: '0', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '' }, + undefined ], + [ 'entry', + { path: 'cc.txt', + mode: 420, + uid: 24561, + gid: 20, + size: 513, + mtime: new Date('Wed, 26 Oct 2011 01:11:02 GMT'), + cksum: 5525, + type: '0', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '' }, + undefined ], + [ 'entry', + { path: 'r/e/a/l/l/y/-/d/e/e/p/-/f/o/l/d/e/r/-/p/a/t/h/cccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: 24561, + gid: 20, + size: 100, + mtime: new Date('Thu, 27 Oct 2011 03:43:23 GMT'), + cksum: 18124, + type: '0', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '' }, + undefined ], + [ 'entry', + { path: 'Ω.txt', + mode: 420, + uid: 24561, + gid: 20, + size: 2, + mtime: new Date('Thu, 27 Oct 2011 17:51:49 GMT'), + cksum: 5695, + type: '0', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '' }, + undefined ], + [ 'extendedHeader', + { path: 'PaxHeader/Ω.txt', + mode: 420, + uid: 24561, + gid: 20, + size: 120, + mtime: new Date('Thu, 27 Oct 2011 17:51:49 GMT'), + cksum: 6702, + type: 'x', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '' }, + { path: 'Ω.txt', + ctime: 1319737909, + atime: 1319739061, + dev: 234881026, + ino: 51693379, + nlink: 1 } ], + [ 'entry', + { path: 'Ω.txt', + mode: 420, + uid: 24561, + gid: 20, + size: 2, + mtime: new Date('Thu, 27 Oct 2011 17:51:49 GMT'), + cksum: 5695, + type: '0', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '', + ctime: new Date('Thu, 27 Oct 2011 17:51:49 GMT'), + atime: new Date('Thu, 27 Oct 2011 18:11:01 GMT'), + dev: 234881026, + ino: 51693379, + nlink: 1 }, + undefined ], + [ 'extendedHeader', + { path: 'PaxHeader/200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: 24561, + gid: 20, + size: 353, + mtime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + cksum: 14488, + type: 'x', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '' }, + { path: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + ctime: 1319686868, + atime: 1319741254, + 'LIBARCHIVE.creationtime': '1319686852', + dev: 234881026, + ino: 51681874, + nlink: 1 } ], + [ 'entry', + { path: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: 24561, + gid: 20, + size: 200, + mtime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + cksum: 14570, + type: '0', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '', + ctime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + atime: new Date('Thu, 27 Oct 2011 18:47:34 GMT'), + 'LIBARCHIVE.creationtime': '1319686852', + dev: 234881026, + ino: 51681874, + nlink: 1 }, + undefined ], + [ 'longPath', + { path: '././@LongLink', + mode: 0, + uid: 0, + gid: 0, + size: 201, + mtime: new Date('Thu, 01 Jan 1970 00:00:00 GMT'), + cksum: 4976, + type: 'L', + linkpath: '', + ustar: false }, + '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc' ], + [ 'entry', + { path: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: 1000, + gid: 1000, + size: 201, + mtime: new Date('Thu, 27 Oct 2011 22:21:50 GMT'), + cksum: 14086, + type: '0', + linkpath: '', + ustar: false }, + undefined ], + [ 'longLinkpath', + { path: '././@LongLink', + mode: 0, + uid: 0, + gid: 0, + size: 201, + mtime: new Date('Thu, 01 Jan 1970 00:00:00 GMT'), + cksum: 4975, + type: 'K', + linkpath: '', + ustar: false }, + '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc' ], + [ 'longPath', + { path: '././@LongLink', + mode: 0, + uid: 0, + gid: 0, + size: 201, + mtime: new Date('Thu, 01 Jan 1970 00:00:00 GMT'), + cksum: 4976, + type: 'L', + linkpath: '', + ustar: false }, + '200LLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLL' ], + [ 'entry', + { path: '200LLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLLL', + mode: 511, + uid: 1000, + gid: 1000, + size: 0, + mtime: new Date('Fri, 28 Oct 2011 23:05:17 GMT'), + cksum: 21603, + type: '2', + linkpath: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + ustar: false }, + undefined ], + [ 'extendedHeader', + { path: 'PaxHeader/200-hard', + mode: 420, + uid: 24561, + gid: 20, + size: 143, + mtime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + cksum: 6533, + type: 'x', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '' }, + { ctime: 1320617144, + atime: 1320617232, + 'LIBARCHIVE.creationtime': '1319686852', + dev: 234881026, + ino: 51681874, + nlink: 2 } ], + [ 'entry', + { path: '200-hard', + mode: 420, + uid: 24561, + gid: 20, + size: 200, + mtime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + cksum: 5526, + type: '0', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '', + ctime: new Date('Sun, 06 Nov 2011 22:05:44 GMT'), + atime: new Date('Sun, 06 Nov 2011 22:07:12 GMT'), + 'LIBARCHIVE.creationtime': '1319686852', + dev: 234881026, + ino: 51681874, + nlink: 2 }, + undefined ], + [ 'extendedHeader', + { path: 'PaxHeader/200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: 24561, + gid: 20, + size: 353, + mtime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + cksum: 14488, + type: 'x', + linkpath: '', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '' }, + { path: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + ctime: 1320617144, + atime: 1320617406, + 'LIBARCHIVE.creationtime': '1319686852', + dev: 234881026, + ino: 51681874, + nlink: 2 } ], + [ 'entry', + { path: '200ccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccccc', + mode: 420, + uid: 24561, + gid: 20, + size: 0, + mtime: new Date('Thu, 27 Oct 2011 03:41:08 GMT'), + cksum: 15173, + type: '1', + linkpath: '200-hard', + ustar: 'ustar\0', + ustarver: '00', + uname: 'isaacs', + gname: 'staff', + devmaj: 0, + devmin: 0, + fill: '', + ctime: new Date('Sun, 06 Nov 2011 22:05:44 GMT'), + atime: new Date('Sun, 06 Nov 2011 22:10:06 GMT'), + 'LIBARCHIVE.creationtime': '1319686852', + dev: 234881026, + ino: 51681874, + nlink: 2 }, + undefined ] ] + + +tap.test("parser test", function (t) { + var parser = tar.Parse() + + parser.on("end", function () { + t.equal(index, expect.length, "saw all expected events") + t.end() + }) + + fs.createReadStream(file) + .pipe(parser) + .on("*", function (ev, entry) { + var wanted = expect[index] + if (!wanted) { + return t.fail("Unexpected event: " + ev) + } + var result = [ev, entry.props] + entry.on("end", function () { + result.push(entry.fields || entry.body) + + t.equal(ev, wanted[0], index + " event type") + t.equivalent(entry.props, wanted[1], wanted[1].path + " entry properties") + if (wanted[2]) { + t.equivalent(result[2], wanted[2], "metadata values") + } + index ++ + }) + }) +}) diff --git a/deps/npm/node_modules/which/LICENSE b/deps/npm/node_modules/which/LICENSE new file mode 100644 index 0000000000..05a4010949 --- /dev/null +++ b/deps/npm/node_modules/which/LICENSE @@ -0,0 +1,23 @@ +Copyright 2009, 2010, 2011 Isaac Z. Schlueter. +All rights reserved. + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/deps/npm/node_modules/which/README.md b/deps/npm/node_modules/which/README.md new file mode 100644 index 0000000000..ff1eb531a7 --- /dev/null +++ b/deps/npm/node_modules/which/README.md @@ -0,0 +1,5 @@ +The "which" util from npm's guts. + +Finds the first instance of a specified executable in the PATH +environment variable. Does not cache the results, so `hash -r` is not +needed when the PATH changes. diff --git a/deps/npm/node_modules/which/bin/which b/deps/npm/node_modules/which/bin/which new file mode 100755 index 0000000000..8432ce2f6d --- /dev/null +++ b/deps/npm/node_modules/which/bin/which @@ -0,0 +1,14 @@ +#!/usr/bin/env node +var which = require("../") +if (process.argv.length < 3) { + console.error("Usage: which ") + process.exit(1) +} + +which(process.argv[2], function (er, thing) { + if (er) { + console.error(er.message) + process.exit(er.errno || 127) + } + console.log(thing) +}) diff --git a/deps/npm/node_modules/which/package.json b/deps/npm/node_modules/which/package.json new file mode 100644 index 0000000000..02990697f7 --- /dev/null +++ b/deps/npm/node_modules/which/package.json @@ -0,0 +1,17 @@ +{ + "author": "Isaac Z. Schlueter (http://blog.izs.me)", + "name": "which", + "description": "Like which(1) unix command. Find the first instance of an executable in the PATH.", + "version": "1.0.2", + "repository": { + "type": "git", + "url": "git://github.com/isaacs/node-which.git" + }, + "main": "which.js", + "bin": "./bin/which", + "engines": { + "node": "*" + }, + "dependencies": {}, + "devDependencies": {} +} diff --git a/deps/npm/node_modules/which/which.js b/deps/npm/node_modules/which/which.js new file mode 100644 index 0000000000..b124ead672 --- /dev/null +++ b/deps/npm/node_modules/which/which.js @@ -0,0 +1,67 @@ +module.exports = which +which.sync = whichSync + +var path = require("path") + , fs + , COLON = process.platform === "win32" ? ";" : ":" + +try { + fs = require("graceful-fs") +} catch (ex) { + fs = require("fs") +} + +// console.log(process.execPath) +// console.log(process.argv) + +function isExe (mod, uid, gid) { + //console.error("isExe?", (mod & 0111).toString(8)) + var ret = (mod & 0001) + || (mod & 0010) && process.getgid && gid === process.getgid() + || (mod & 0100) && process.getuid && uid === process.getuid() + //console.error("isExe?", ret) + return ret +} +function which (cmd, cb) { + if (cmd.charAt(0) === "/") return cb(null, cmd) + var pathEnv = (process.env.PATH || "").split(COLON) + , pathExt = [""] + if (process.platform === "win32") { + pathEnv.push(process.cwd()) + pathExt = (process.env.PATHEXT || ".EXE").split(COLON) + } + //console.error("pathEnv", pathEnv) + ;(function F (i, l) { + if (i === l) return cb(new Error("not found: "+cmd)) + var p = path.resolve(pathEnv[i], cmd) + ;(function E (ii, ll) { + if (ii === ll) return F(i + 1, l) + var ext = pathExt[ii] + //console.error(p + ext) + fs.stat(p + ext, function (er, stat) { + if (!er && + stat && + stat.isFile() && + isExe(stat.mode, stat.uid, stat.gid)) { + //console.error("yes, exe!", p + ext) + return cb(null, p + ext) + } + return E(ii + 1, ll) + }) + })(0, pathExt.length) + })(0, pathEnv.length) +} + + +function whichSync (cmd) { + if (cmd.charAt(0) === "/") return cmd + var pathEnv = (process.env.PATH || "").split(COLON) + for (var i = 0, l = pathEnv.length; i < l; i ++) { + var p = path.join(pathEnv[i], cmd) + if (p === process.execPath) return p + var stat + try { stat = fs.statSync(p) } catch (ex) {} + if (stat && isExe(stat.mode, stat.uid, stat.gid)) return p + } + throw new Error("not found: "+cmd) +} diff --git a/deps/npm/package.json b/deps/npm/package.json new file mode 100644 index 0000000000..3060e4d077 --- /dev/null +++ b/deps/npm/package.json @@ -0,0 +1,77 @@ +{ "name": "npm" +, "publishConfig": { "tag": "alpha" } +, "description": "A package manager for node" +, "keywords": [ "package manager", "modules", "install", "package.json" ] +, "version": "1.1.0-alpha" +, "preferGlobal": true +, "config": { "publishtest": false } +, "homepage": "http://npmjs.org/" +, "author": "Isaac Z. Schlueter (http://blog.izs.me)" +, "repository": + { "type": "git" + , "url": "https://github.com/isaacs/npm" + } +, "bugs": + { "email": "npm-@googlegroups.com" + , "url": "http://github.com/isaacs/npm/issues" + } +, "directories": { "doc": "./doc" + , "man": "./man" + , "lib": "./lib" + , "bin": "./bin" + } +, "main": "./lib/npm.js" +, "bin": { "npm": "./bin/npm-cli.js" + , "npm_g": "./bin/npm-cli.js" + , "npm-g": "./bin/npm-cli.js" } +, "dependencies": + { "semver": "1" + , "ini": "1" + , "slide": "1" + , "abbrev": "1" + , "graceful-fs": "1" + , "minimatch": "0" + , "nopt": "1" + , "node-uuid": "1.2" + , "proto-list": "1" + , "rimraf": "1" + , "request": "~2.1.1" + , "which": "1" + , "tar": "0" + , "fstream": "0" + , "block-stream": "*" + , "inherits": "1" + , "mkdirp": "0.1" + } +, "bundleDependencies": + [ "slide" + , "ini" + , "semver" + , "abbrev" + , "graceful-fs" + , "minimatch" + , "nopt" + , "node-uuid" + , "rimraf" + , "request" + , "proto-list" + , "which" + , "tar" + , "fstream" + , "block-stream" + , "inherits" + , "mkdirp" + ] +, "devDependencies": + { "ronn": "https://github.com/isaacs/ronnjs/tarball/master" } +, "engines": { "node": "0.6 || 0.7 || 0.8", "npm": "1" } +, "scripts": { "test": "./test/run" + , "prepublish": "make -j4 doc" + , "dumpconf": "env | grep npm | sort | uniq" + } +, "licenses": + [ { "type": "MIT +no-false-attribs" + , "url": "http://github.com/isaacs/npm/raw/master/LICENSE" + } + ] +} diff --git a/deps/npm/scripts/clean-old.sh b/deps/npm/scripts/clean-old.sh new file mode 100644 index 0000000000..cda80f2f48 --- /dev/null +++ b/deps/npm/scripts/clean-old.sh @@ -0,0 +1,165 @@ +#!/bin/bash + +# look for old 0.x cruft, and get rid of it. +# Should already be sitting in the npm folder. + +# This doesn't have to be quite as cross-platform as install.sh. +# There are some bash-isms, because maintaining *two* +# fully-portable posix/bourne sh scripts is too much for +# one project with a sane maintainer. + +# If readlink isn't available, then this is just too tricky. +# However, greadlink is fine, so Solaris can join the party, too. +readlink="readlink" +which $readlink >/dev/null 2>/dev/null +if [ $? -ne 0 ]; then + readlink="greadlink" + which $readlink >/dev/null 2>/dev/null + if [ $? -ne 0 ]; then + echo "Can't find the readlink or greadlink command. Aborting." + exit 1 + fi +fi + +if [ "x$npm_config_prefix" != "x" ]; then + PREFIXES=$npm_config_prefix +else + node="$NODE" + if [ "x$node" = "x" ]; then + node=`which node` + fi + if [ "x$node" = "x" ]; then + echo "Can't find node to determine prefix. Aborting." + exit 1 + fi + + + PREFIX=`dirname $node` + PREFIX=`dirname $PREFIX` + echo "cleanup prefix=$PREFIX" + PREFIXES=$PREFIX + + altprefix=`"$node" -e process.installPrefix` + if [ "x$altprefix" != "x" ] && [ "x$altprefix" != "x$PREFIX" ]; then + echo "altprefix=$altprefix" + PREFIXES="$PREFIX $altprefix" + fi +fi + +# now prefix is where npm would be rooted by default +# go hunting. + +packages= +for prefix in $PREFIXES; do + packages="$packages + "`ls "$prefix"/lib/node/.npm 2>/dev/null | grep -v .cache` +done + +packages=`echo $packages` + +filelist=() +fid=0 + +for prefix in $PREFIXES; do + # remove any links into the .npm dir, or links to + # version-named shims/symlinks. + for folder in share/man bin lib/node; do + find $prefix/$folder -type l | while read file; do + target=`$readlink $file | grep '/\.npm/'` + if [ "x$target" != "x" ]; then + # found one! + filelist[$fid]="$file" + let 'fid++' + # also remove any symlinks to this file. + base=`basename "$file"` + base=`echo "$base" | awk -F@ '{print $1}'` + if [ "x$base" != "x" ]; then + find "`dirname $file`" -type l -name "$base"'*' \ + | while read l; do + target=`$readlink "$l" | grep "$base"` + if [ "x$target" != "x" ]; then + filelist[$fid]="$1" + let 'fid++' + fi + done + fi + fi + done + + # Scour for shim files. These are relics of 0.2 npm installs. + # note: grep -r is not portable. + find $prefix/$folder -type f \ + | xargs grep -sl '// generated by npm' \ + | while read file; do + filelist[$fid]="$file" + let 'fid++' + done + done + + # now remove the package modules, and the .npm folder itself. + if [ "x$packages" != "x" ]; then + for pkg in $packages; do + filelist[$fid]="$prefix/lib/node/$pkg" + let 'fid++' + for i in $prefix/lib/node/$pkg\@*; do + filelist[$fid]="$i" + let 'fid++' + done + done + fi + + for folder in lib/node/.npm lib/npm share/npm; do + if [ -d $prefix/$folder ]; then + filelist[$fid]="$prefix/$folder" + let 'fid++' + fi + done +done + +# now actually clean, but only if there's anything TO clean +if [ "${#filelist[@]}" -gt 0 ]; then + echo "" + echo "This script will find and eliminate any shims, symbolic" + echo "links, and other cruft that was installed by npm 0.x." + echo "" + + if [ "x$packages" != "x" ]; then + echo "The following packages appear to have been installed with" + echo "an old version of npm, and will be removed forcibly:" + for pkg in $packages; do + echo " $pkg" + done + echo "Make a note of these. You may want to install them" + echo "with npm 1.0 when this process is completed." + echo "" + fi + + OK= + if [ "x$1" = "x-y" ]; then + OK="yes" + fi + + while [ "$OK" != "y" ] && [ "$OK" != "yes" ] && [ "$OK" != "no" ]; do + echo "Is this OK?" + echo " enter 'yes' or 'no'" + echo " or 'show' to see a list of files " + read OK + if [ "x$OK" = "xshow" ] || [ "x$OK" = "xs" ]; then + for i in "${filelist[@]}"; do + echo "$i" + done + fi + done + if [ "$OK" = "no" ]; then + echo "Aborting" + exit 1 + fi + for i in "${filelist[@]}"; do + rm -rf "$i" + done +fi + +echo "" +echo 'All clean!' + +exit 0 diff --git a/deps/npm/scripts/doc-build.sh b/deps/npm/scripts/doc-build.sh new file mode 100755 index 0000000000..6c32ea1838 --- /dev/null +++ b/deps/npm/scripts/doc-build.sh @@ -0,0 +1,71 @@ +#!/bin/bash + +if [[ $DEBUG != "" ]]; then + set -x +fi +set -o errexit +set -o pipefail + +if ! [ -x node_modules/.bin/ronn ]; then + if [ -f .building_ronn ]; then + while [ -f .building_ronn ]; do + sleep 1 + done + else + # a race to see which make process will be the one to install ronn + echo $$ > .building_ronn + sleep 1 + if [ $(cat .building_ronn) == $$ ]; then + make node_modules/ronn + rm .building_ronn + else + while [ -f .building_ronn ]; do + sleep 1 + done + fi + fi +fi + +src=$1 +dest=$2 +name=$(basename ${src%.*}) +date=$(date -u +'%Y-%M-%d %H:%m:%S') +version=$(node cli.js -v) + +mkdir -p $(dirname $dest) + +case $dest in + *.[13]) + ./node_modules/.bin/ronn --roff $src \ + | sed "s|@VERSION@|$version|g" \ + | perl -pi -e 's/npm\\-([^\(]*)\(1\)/npm help \1/g' \ + | perl -pi -e 's/npm\\-([^\(]*)\(3\)/npm apihelp \1/g' \ + | perl -pi -e 's/npm\(1\)/npm help npm/g' \ + | perl -pi -e 's/npm\(3\)/npm apihelp npm/g' \ + > $dest + exit $? + ;; + *.html) + (cat html/dochead.html && \ + ./node_modules/.bin/ronn -f $src && \ + cat html/docfoot.html )\ + | sed "s|@NAME@|$name|g" \ + | sed "s|@DATE@|$date|g" \ + | sed "s|@VERSION@|$version|g" \ + | perl -pi -e 's/

npm(-?[^\(]*\([0-9]\)) -- (.*?)<\/h1>/

npm\1<\/h1>

\2<\/p>/g' \ + | perl -pi -e 's/npm-npm/npm/g' \ + | perl -pi -e 's/([^"-])(npm-)?README(\(1\))?/\1README<\/a>/g' \ + | perl -pi -e 's/<a href="..\/doc\/README.html">README<\/a><\/title>/<title>README<\/title>/g' \ + | perl -pi -e 's/([^"-])npm-([^\(]+)(\(1\))/\1<a href="..\/doc\/\2.html">\2\3<\/a>/g' \ + | perl -pi -e 's/([^"-])npm-([^\(]+)(\(3\))/\1<a href="..\/api\/\2.html">\2\3<\/a>/g' \ + | perl -pi -e 's/([^"-])npm\(1\)/\1<a href="..\/doc\/npm.html">npm(1)<\/a>/g' \ + | perl -pi -e 's/([^"-])npm\(3\)/\1<a href="..\/api\/npm.html">npm(3)<\/a>/g' \ + | perl -pi -e 's/\([13]\)<\/a><\/h1>/<\/a><\/h1>/g' \ + > $dest + exit $? + ;; + *) + echo "Invalid destination type: $dest" >&2 + exit 1 + ;; +esac diff --git a/deps/npm/scripts/index-build.js b/deps/npm/scripts/index-build.js new file mode 100644 index 0000000000..b3c19a03a1 --- /dev/null +++ b/deps/npm/scripts/index-build.js @@ -0,0 +1,62 @@ +#!/usr/bin/env node +var fs = require("fs") + , path = require("path") + , cli = path.resolve(__dirname, "..", "doc", "cli") + , clidocs = null + , api = path.resolve(__dirname, "..", "doc", "api") + , apidocs = null + , readme = path.resolve(__dirname, "..", "README.md") + +fs.readdir(cli, done("cli")) +fs.readdir(api, done("api")) + +function done (which) { return function (er, docs) { + if (er) throw er + if (which === "api") apidocs = docs + else clidocs = docs + + if (apidocs && clidocs) next() +}} + +function filter (d) { + return d !== "index.md" + && d.charAt(0) !== "." + && d.match(/\.md$/) +} + +function next () { + console.log( + "npm-index(1) -- Index of all npm documentation\n" + + "==============================================\n") + + apidocs = apidocs.filter(filter).map(function (d) { + return [3, path.resolve(api, d)] + }) + + clidocs = clidocs.filter(filter).map(function (d) { + return [1, path.resolve(cli, d)] + }) + + writeLine([1, readme]) + + console.log("# Command Line Documentation") + + clidocs.forEach(writeLine) + + console.log("# API Documentation") + apidocs.forEach(writeLine) +} + +function writeLine (sd) { + var sxn = sd[0] + , doc = sd[1] + , d = path.basename(doc, ".md") + , s = fs.lstatSync(doc) + + if (s.isSymbolicLink()) return + + var content = fs.readFileSync(doc, "utf8").split("\n")[0].split("--")[1] + + console.log("## npm-%s(%d)\n", d, sxn) + console.log(content + "\n") +} diff --git a/deps/npm/scripts/install.sh b/deps/npm/scripts/install.sh new file mode 100644 index 0000000000..16fbfe618b --- /dev/null +++ b/deps/npm/scripts/install.sh @@ -0,0 +1,282 @@ +#!/bin/sh + +# A word about this shell script: +# +# It must work everywhere, including on systems that lack +# a /bin/bash, map 'sh' to ksh, ksh97, bash, ash, or zsh, +# and potentially have either a posix shell or bourne +# shell living at /bin/sh. +# +# See this helpful document on writing portable shell scripts: +# http://www.gnu.org/s/hello/manual/autoconf/Portable-Shell.html +# +# The only shell it won't ever work on is cmd.exe. + +if [ "x$0" = "xsh" ]; then + # run as curl | sh + # on some systems, you can just do cat>npm-install.sh + # which is a bit cuter. But on others, &1 is already closed, + # so catting to another script file won't do anything. + curl -s http://npmjs.org/install.sh > npm-install-$$.sh + sh npm-install-$$.sh + ret=$? + rm npm-install-$$.sh + exit $ret +fi + +# See what "npm_config_*" things there are in the env, +# and make them permanent. +# If this fails, it's not such a big deal. +configures="`env | grep 'npm_config_' | sed -e 's|^npm_config_||g'`" + +npm_config_loglevel="error" +if [ "x$npm_debug" = "x" ]; then + (exit 0) +else + echo "Running in debug mode." + echo "Note that this requires bash or zsh." + set -o xtrace + set -o pipefail + npm_config_loglevel="verbose" +fi +export npm_config_loglevel + +# make sure that node exists +node=`which node 2>&1` +ret=$? +if [ $ret -eq 0 ] && [ -x "$node" ]; then + (exit 0) +else + echo "npm cannot be installed without nodejs." >&2 + echo "Install node first, and then try again." >&2 + echo "" >&2 + echo "Maybe node is installed, but not in the PATH?" >&2 + echo "Note that running as sudo can change envs." >&2 + echo "" + echo "PATH=$PATH" >&2 + exit $ret +fi + +# set the temp dir +TMP="${TMPDIR}" +if [ "x$TMP" = "x" ]; then + TMP="/tmp" +fi +TMP="${TMP}/npm.$$" +rm -rf "$TMP" || true +mkdir "$TMP" +if [ $? -ne 0 ]; then + echo "failed to mkdir $TMP" >&2 + exit 1 +fi + +BACK="$PWD" + +ret=0 +tar="${TAR}" +if [ -z "$tar" ]; then + tar="${npm_config_tar}" +fi +if [ -z "$tar" ]; then + tar=`which tar 2>&1` + ret=$? +fi + +if [ $ret -eq 0 ] && [ -x "$tar" ]; then + echo "tar=$tar" + echo "version:" + $tar --version + ret=$? +fi + +if [ $ret -eq 0 ]; then + (exit 0) +else + echo "No suitable tar program found." + exit 1 +fi + + + +# Try to find a suitable make +# If the MAKE environment var is set, use that. +# otherwise, try to find gmake, and then make. +# If no make is found, then just execute the necessary commands. + +# XXX For some reason, make is building all the docs every time. This +# is an annoying source of bugs. Figure out why this happens. +MAKE=NOMAKE + +if [ "x$MAKE" = "x" ]; then + make=`which gmake 2>&1` + if [ $? -eq 0 ] && [ -x $make ]; then + (exit 0) + else + make=`which make 2>&1` + if [ $? -eq 0 ] && [ -x $make ]; then + (exit 0) + else + make=NOMAKE + fi + fi +else + make="$MAKE" +fi + +if [ -x "$make" ]; then + (exit 0) +else + # echo "Installing without make. This may fail." >&2 + make=NOMAKE +fi + +# If there's no bash, then don't even try to clean +if [ -x "/bin/bash" ]; then + (exit 0) +else + clean="no" +fi + +t="${npm_install}" +if [ -z "$t" ]; then + t="latest" +fi + +# the npmca cert +cacert=' +-----BEGIN CERTIFICATE----- +MIIChzCCAfACCQDauvz/KHp8ejANBgkqhkiG9w0BAQUFADCBhzELMAkGA1UEBhMC +VVMxCzAJBgNVBAgTAkNBMRAwDgYDVQQHEwdPYWtsYW5kMQwwCgYDVQQKEwNucG0x +IjAgBgNVBAsTGW5wbSBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkxDjAMBgNVBAMTBW5w +bUNBMRcwFQYJKoZIhvcNAQkBFghpQGl6cy5tZTAeFw0xMTA5MDUwMTQ3MTdaFw0y +MTA5MDIwMTQ3MTdaMIGHMQswCQYDVQQGEwJVUzELMAkGA1UECBMCQ0ExEDAOBgNV +BAcTB09ha2xhbmQxDDAKBgNVBAoTA25wbTEiMCAGA1UECxMZbnBtIENlcnRpZmlj +YXRlIEF1dGhvcml0eTEOMAwGA1UEAxMFbnBtQ0ExFzAVBgkqhkiG9w0BCQEWCGlA +aXpzLm1lMIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDLI4tIqPpRW+ACw9GE +OgBlJZwK5f8nnKCLK629Pv5yJpQKs3DENExAyOgDcyaF0HD0zk8zTp+ZsLaNdKOz +Gn2U181KGprGKAXP6DU6ByOJDWmTlY6+Ad1laYT0m64fERSpHw/hjD3D+iX4aMOl +y0HdbT5m1ZGh6SJz3ZqxavhHLQIDAQABMA0GCSqGSIb3DQEBBQUAA4GBAC4ySDbC +l7W1WpLmtLGEQ/yuMLUf6Jy/vr+CRp4h+UzL+IQpCv8FfxsYE7dhf/bmWTEupBkv +yNL18lipt2jSvR3v6oAHAReotvdjqhxddpe5Holns6EQd1/xEZ7sB1YhQKJtvUrl +ZNufy1Jf1r0ldEGeA+0ISck7s+xSh9rQD2Op +-----END CERTIFICATE----- +' + +echo "$cacert" > "$TMP/cafile.crt" +cacert="$TMP/cafile.crt" + +# need to echo "" after, because Posix sed doesn't treat EOF +# as an implied end of line. +url=`(curl -SsL --cacert "$cacert" https://registry.npmjs.org/npm/$t; echo "") \ + | sed -e 's/^.*tarball":"//' \ + | sed -e 's/".*$//'` + +ret=$? +if [ "x$url" = "x" ]; then + ret=125 + # try without the -e arg to sed. + url=`(curl -SsL --cacert "$cacert" https://registry.npmjs.org/npm/$t; echo "") \ + | sed 's/^.*tarball":"//' \ + | sed 's/".*$//'` + ret=$? + if [ "x$url" = "x" ]; then + ret=125 + fi +fi +if [ $ret -ne 0 ]; then + echo "Failed to get tarball url for npm/$t" >&2 + exit $ret +fi + + +echo "fetching: $url" >&2 + +cd "$TMP" \ + && curl -SsL --cacert "$cacert" "$url" \ + | $tar -xzf - \ + && rm "$cacert" \ + && cd "$TMP"/* \ + && (node_version=`"$node" --version 2>&1` + ret=$? + if [ $ret -eq 0 ]; then + req=`"$node" bin/read-package-json.js package.json engines.node` + if [ -d node_modules ]; then + "$node" node_modules/semver/bin/semver -v "$node_version" -r "$req" + ret=$? + else + "$node" bin/semver.js -v "$node_version" -r "$req" + ret=$? + fi + fi + if [ $ret -ne 0 ]; then + echo "You need node $req to run this program." >&2 + echo "node --version reports: $node_version" >&2 + echo "Please upgrade node before continuing." + exit $ret + fi) \ + && (ver=`"$node" bin/read-package-json.js package.json version` + isnpm10=0 + if [ $ret -eq 0 ]; then + req=`"$node" bin/read-package-json.js package.json engines.node` + if [ -d node_modules ]; then + if "$node" node_modules/semver/bin/semver -v "$ver" -r "1" + then + isnpm10=1 + fi + else + if "$node" bin/semver -v "$ver" -r ">=1.0"; then + isnpm10=1 + fi + fi + fi + + ret=0 + if [ $isnpm10 -eq 1 ] && [ -f "scripts/clean-old.sh" ]; then + if [ "x$skipclean" = "x" ]; then + (exit 0) + else + clean=no + fi + if [ "x$clean" = "xno" ] \ + || [ "x$clean" = "xn" ]; then + echo "Skipping 0.x cruft clean" >&2 + ret=0 + elif [ "x$clean" = "xy" ] || [ "x$clean" = "xyes" ]; then + NODE="$node" /bin/bash "scripts/clean-old.sh" "-y" + ret=$? + else + NODE="$node" /bin/bash "scripts/clean-old.sh" </dev/tty + ret=$? + fi + fi + + if [ $ret -ne 0 ]; then + echo "Aborted 0.x cleanup. Exiting." >&2 + exit $ret + fi) \ + && (if [ "x$configures" = "x" ]; then + (exit 0) + else + echo "./configure "$configures + echo "$configures" > npmrc + fi) \ + && (if [ "$make" = "NOMAKE" ]; then + (exit 0) + elif "$make" uninstall install; then + (exit 0) + else + make="NOMAKE" + fi + if [ "$make" = "NOMAKE" ]; then + "$node" cli.js rm npm -gf + "$node" cli.js install -gf + fi) \ + && cd "$BACK" \ + && rm -rf "$TMP" \ + && echo "It worked" + +ret=$? +if [ $ret -ne 0 ]; then + echo "It failed" >&2 +fi +exit $ret diff --git a/deps/npm/test/common.js b/deps/npm/test/common.js new file mode 100644 index 0000000000..2755056b1b --- /dev/null +++ b/deps/npm/test/common.js @@ -0,0 +1,7 @@ + +// whatever, it's just tests. +;["util","assert"].forEach(function (thing) { + thing = require("thing") + for (var i in thing) global[i] = thing[i] +} + diff --git a/deps/npm/test/disabled/bundlerecurs/package.json b/deps/npm/test/disabled/bundlerecurs/package.json new file mode 100644 index 0000000000..d870411706 --- /dev/null +++ b/deps/npm/test/disabled/bundlerecurs/package.json @@ -0,0 +1,4 @@ +{ "name" : "bundletest" +, "version" : "1.0.0" +, "dependencies" : { "bundletest" : "*" } +} diff --git a/deps/npm/test/disabled/failer/package.json b/deps/npm/test/disabled/failer/package.json new file mode 100644 index 0000000000..e1f8e946b7 --- /dev/null +++ b/deps/npm/test/disabled/failer/package.json @@ -0,0 +1,5 @@ +{ "name" : "failer" +, "version" : "9999.999.99" +, "dependencies" : { "base64" : "*" } +, "scripts" : { "preinstall" : "exit 1" } +} diff --git a/deps/npm/test/disabled/fast/package.json b/deps/npm/test/disabled/fast/package.json new file mode 100644 index 0000000000..fbf26e9b14 --- /dev/null +++ b/deps/npm/test/disabled/fast/package.json @@ -0,0 +1,9 @@ +{ "name" : "fast" +, "description" : "does nothing, and not very fast" +, "version" : "1.2.3" +, "scripts" : +{ "preinstall" : "sleep 1 && echo fast 1 $(date +%s) && echo fast 2" +, "install" : "sleep 1 && echo fast 2 $(date +%s) && echo fast 3" +, "postinstall" : "sleep 1 && echo fast 3 $(date +%s) && echo fast 4" +} +} diff --git a/deps/npm/test/disabled/package-config/package.json b/deps/npm/test/disabled/package-config/package.json new file mode 100644 index 0000000000..7ec97d3805 --- /dev/null +++ b/deps/npm/test/disabled/package-config/package.json @@ -0,0 +1,4 @@ +{"name":"package-config" +,"version":"1.2.3" +,"config":{"foo":"bar"} +,"scripts":{"test":"./test.js"}} diff --git a/deps/npm/test/disabled/package-config/test.js b/deps/npm/test/disabled/package-config/test.js new file mode 100755 index 0000000000..7337b237b5 --- /dev/null +++ b/deps/npm/test/disabled/package-config/test.js @@ -0,0 +1,17 @@ +#!/usr/bin/env node + +var env = process.env + , orig = require(process.env.npm_package_name+"/package.json").config + , assert = require("assert") + +console.log("Before running this test, do:\n" + +" npm config set package-config:foo boo\n" + +"or else it's about to fail.") +assert.equal(env.npm_package_config_foo, "boo", "foo != boo") +assert.equal(orig.foo, "bar", "original foo != bar") +assert.equal(env["npm_config_package-config:foo"], "boo", + "package-config:foo != boo") +console.log({ foo: env.npm_package_config_foo + , orig_foo: orig.foo + , "package-config:foo": env["npm_config_package-config:foo"] + }) diff --git a/deps/npm/test/disabled/slow/package.json b/deps/npm/test/disabled/slow/package.json new file mode 100644 index 0000000000..ba6be42fa9 --- /dev/null +++ b/deps/npm/test/disabled/slow/package.json @@ -0,0 +1,9 @@ +{ "name" : "slow" +, "description" : "just like fast, but even slower" +, "version" : "1.2.3" +, "scripts" : + { "preinstall" : "sleep 1 && echo slow 1 $(date +%s) && sleep 1 && echo slow 2 $(date +%s)" + , "install" : "sleep 1 && echo slow 2 $(date +%s) && sleep 1 && echo slow 3 $(date +%s)" + , "postinstall" : "sleep 1 && echo slow 3 $(date +%s) && sleep 1 && echo slow 4 $(date +%s)" + } +} diff --git a/deps/npm/test/disabled/startstop/package.json b/deps/npm/test/disabled/startstop/package.json new file mode 100644 index 0000000000..bee2a2fd3a --- /dev/null +++ b/deps/npm/test/disabled/startstop/package.json @@ -0,0 +1,3 @@ +{"name":"startstop" +,"version":"1.2.3" +,"scripts":{"start":"echo 'start'","stop":"echo 'stop'"}} diff --git a/deps/npm/test/packages/npm-test-blerg/package.json b/deps/npm/test/packages/npm-test-blerg/package.json new file mode 100644 index 0000000000..374b4432b2 --- /dev/null +++ b/deps/npm/test/packages/npm-test-blerg/package.json @@ -0,0 +1,4 @@ +{ "name":"npm-test-blerg" +, "version" : "0.0.0" +, "scripts" : { "test" : "node test.js" } +} diff --git a/deps/npm/test/packages/npm-test-blerg/test.js b/deps/npm/test/packages/npm-test-blerg/test.js new file mode 100644 index 0000000000..f548458ac0 --- /dev/null +++ b/deps/npm/test/packages/npm-test-blerg/test.js @@ -0,0 +1,5 @@ + +var assert = require("assert") +assert.equal(undefined, process.env.npm_config__password, "password exposed!") +assert.equal(undefined, process.env.npm_config__auth, "auth exposed!") +assert.equal(undefined, process.env.npm_config__authCrypt, "authCrypt exposed!") diff --git a/deps/npm/test/packages/npm-test-env-reader/package.json b/deps/npm/test/packages/npm-test-env-reader/package.json new file mode 100644 index 0000000000..ddd5c91abd --- /dev/null +++ b/deps/npm/test/packages/npm-test-env-reader/package.json @@ -0,0 +1,14 @@ +{ "name":"npm-test-env-reader" +, "version" : "1.2.3" +, "scripts" : + { "install" : "./test.sh" + , "preinstall" : "./test.sh" + , "preuninstall" : "./test.sh" + , "postuninstall" : "./test.sh" + , "test" : "./test.sh" + , "stop" : "./test.sh" + , "start" : "./test.sh" + , "restart" : "./test.sh" + , "foo" : "./test.sh" + } +} diff --git a/deps/npm/test/packages/npm-test-env-reader/test.sh b/deps/npm/test/packages/npm-test-env-reader/test.sh new file mode 100755 index 0000000000..b4ca4374ed --- /dev/null +++ b/deps/npm/test/packages/npm-test-env-reader/test.sh @@ -0,0 +1,3 @@ +#!/usr/bin/env sh +env | grep npm | sort | uniq +echo PATH=$PATH diff --git a/deps/npm/test/packages/npm-test-missing-bindir/package.json b/deps/npm/test/packages/npm-test-missing-bindir/package.json new file mode 100644 index 0000000000..49e26742df --- /dev/null +++ b/deps/npm/test/packages/npm-test-missing-bindir/package.json @@ -0,0 +1,4 @@ +{ "name":"npm-test-missing-bindir" +, "version" : "0.0.0" +, "scripts" : { "test" : "node test.js" } +, "directories": { "bin" : "./not-found" } } diff --git a/deps/npm/test/packages/npm-test-missing-bindir/test.js b/deps/npm/test/packages/npm-test-missing-bindir/test.js new file mode 100644 index 0000000000..f548458ac0 --- /dev/null +++ b/deps/npm/test/packages/npm-test-missing-bindir/test.js @@ -0,0 +1,5 @@ + +var assert = require("assert") +assert.equal(undefined, process.env.npm_config__password, "password exposed!") +assert.equal(undefined, process.env.npm_config__auth, "auth exposed!") +assert.equal(undefined, process.env.npm_config__authCrypt, "authCrypt exposed!") diff --git a/deps/npm/test/packages/npm-test-private/package.json b/deps/npm/test/packages/npm-test-private/package.json new file mode 100644 index 0000000000..3d95a37af1 --- /dev/null +++ b/deps/npm/test/packages/npm-test-private/package.json @@ -0,0 +1,4 @@ +{"name":"npm-test-private" +,"version":"9.9.9-9" +,"homepage":"http://www.youtube.com/watch?v=1MLry6Cn_D4" +,"private":"true"} diff --git a/deps/npm/test/packages/npm-test-test-package/package.json b/deps/npm/test/packages/npm-test-test-package/package.json new file mode 100644 index 0000000000..c5c5aeabc1 --- /dev/null +++ b/deps/npm/test/packages/npm-test-test-package/package.json @@ -0,0 +1,5 @@ +{ "name":"npm-test-test-package" +, "author" : "Testy McMock" +, "version" : "1.2.3-99-b" +, "description" : "This is a test package used for debugging. It has some random data and that's all." +} diff --git a/deps/npm/test/packages/npm-test-url-dep/package.json b/deps/npm/test/packages/npm-test-url-dep/package.json new file mode 100644 index 0000000000..72a139e547 --- /dev/null +++ b/deps/npm/test/packages/npm-test-url-dep/package.json @@ -0,0 +1,4 @@ +{ "name":"npm-test-url-dep" +, "version" : "1.2.3" +, "dependencies" : + { "dnode" : "https://github.com/substack/dnode/tarball/master" } } diff --git a/deps/npm/test/run b/deps/npm/test/run new file mode 100755 index 0000000000..2443726db8 --- /dev/null +++ b/deps/npm/test/run @@ -0,0 +1,138 @@ +#!/bin/bash + +if [ "$DEBUG" != "" ]; then + set -x +fi + + +# the "npm" command is set to a custom function here so that we can +# test the code in this repo, rather than whichever version of npm +# happens to be installed. + +main () { + # setup + FAILURES=0 + + cd "$TESTDIR" + + npm config ls + + # install + npm install "$NPMPKG" || exit 1 + + # used in test later + npm config set package-config:foo boo || exit 1 + + npm install $( ls packages | awk '{print "packages/" $1 }' ) || exit 1 + (ls packages | while read pkg; do + npm test "$pkg" + done) || exit 1 + if [ "$FAILURES" == "0" ]; then + npm rm $(ls packages) npm || exit 1 + fi + cleanup + + if ! [ "$npm_package_config_publishtest" == "true" ]; then + echo_err "To test publishing: npm config set npm:publishtest true" + else + # attempt to publish and unpublish each of them. + npm install "$NPMPKG" || exit 1 + + (ls packages | grep -v 'npm-test-private' | while read pkg; do + npm publish packages/$pkg || exit 1 + npm install $pkg || exit 1 + npm unpublish $pkg || exit 1 + done) || exit 1 + + # verify that the private package can't be published + # bypass the test-harness npm function. + "$NPMCLI" publish packages/npm-test-private && ( + npm unpublish npm-test-private + exit 1000 + ) + if [ $? -eq 1000 ]; then + fail "Private package shouldn't be publishable" >&2 + fi + + if [ "$FAILURES" == "0" ]; then + npm rm $(ls packages) npm || exit 1 + fi + cleanup + + fi + + if [ $FAILURES -eq 0 ]; then + echo_err "ok" + rm -rf $TMP + else + echo_err "FAILED: $FAILURES" + fi + exit $FAILURES +} + + + +#################### +# Test Harness below + +# fake functions +npm () { + echo -e "npm $@" + "$NPMCLI" "$@" \ + || fail npm "$@" +} + +# get the absolute path of the executable +SELF_PATH="$0" +if [ "${SELF_PATH:0:1}" != "." ] && [ "${SELF_PATH:0:1}" != "/" ]; then + SELF_PATH=./"$SELF_PATH" +fi +SELF_PATH=$( cd -P -- "$(dirname -- "$SELF_PATH")" \ + && pwd -P \ + ) && SELF_PATH=$SELF_PATH/$(basename -- "$0") +# resolve symlinks +while [ -h "$SELF_PATH" ]; do + DIR=$(dirname -- "$SELF_PATH") + SYM=$(readlink -- "$SELF_PATH") + SELF_PATH=$( cd -- "$DIR" \ + && cd -- $(dirname -- "$SYM") \ + && pwd \ + )/$(basename -- "$SYM") +done +NPMPKG="$(dirname -- "$(dirname -- "$SELF_PATH")")" +NPMCLI="$NPMPKG/cli.js" +TESTDIR="$NPMPKG/test/" +TMP=${TMPDIR:-/tmp} +rm -rf $TMP/npm* +TMP=$TMP/npm-test-$$ +echo "Testing in $TMP ..." +ROOTDIR="$TMP/root" + +cleanup () { + if [ "$FAILURES" != "0" ] && [ "$FAILURES" != "" ]; then + return + fi + [ -d "$ROOTDIR" ] && rm -rf -- "$ROOTDIR" + mkdir -p -- "$ROOTDIR" +} + +export npm_config_prefix="$ROOTDIR" +export npm_config_color="always" +export npm_config_global=true +# have to set this to false, or it'll try to test itself forever +export npm_config_npat=false +export PATH="$PATH":"$ROOTDIR/bin":"$ROOTDIR/node_modules/.bin" +export NODE_PATH="$ROOTDIR/node_modules" + +echo_err () { + echo "$@" >&2 +} +fail () { + let 'FAILURES += 1' + echo_err "" + echo_err -e "\033[33mFailure: $@\033[m" + exit 1 +} + +cleanup +main diff --git a/deps/npm/test/update-test.sh b/deps/npm/test/update-test.sh new file mode 100755 index 0000000000..f72c90dd98 --- /dev/null +++ b/deps/npm/test/update-test.sh @@ -0,0 +1,59 @@ +#!/bin/bash + +SELF_PATH="$0" +if [ "${SELF_PATH:0:1}" != "." ] && [ "${SELF_PATH:0:1}" != "/" ]; then + SELF_PATH=./"$SELF_PATH" +fi +SELF_PATH=$( cd -P -- "$(dirname -- "$SELF_PATH")" \ + && pwd -P \ + ) && SELF_PATH=$SELF_PATH/$(basename -- "$0") + +# resolve symlinks +while [ -h "$SELF_PATH" ]; do + DIR=$(dirname -- "$SELF_PATH") + SYM=$(readlink -- "$SELF_PATH") + SELF_PATH=$( cd -- "$DIR" \ + && cd -- $(dirname -- "$SYM") \ + && pwd \ + )/$(basename -- "$SYM") +done +DIR=$( dirname -- "$SELF_PATH" ) + +export npm_config_root=$DIR/root +export npm_config_binroot=$DIR/bin + +rm -rf $DIR/{root,bin} +mkdir -p $DIR/root +mkdir -p $DIR/bin +npm ls installed 2>/dev/null | grep -v npm | awk '{print $1}' | xargs npm rm &>/dev/null +npm install \ + base64@1.0.0 \ + eyes@0.1.1 \ + vows@0.2.5 \ + websocket-server@1.0.5 &>/dev/null +npm install ./test/packages/blerg &>/dev/null +npm install vows@0.3.0 &>/dev/null + +echo "" +echo "##" +echo "## starting update" +echo "##" +echo "" + +npm update + +echo "" +echo "##" +echo "## update done, all should be 'latest'" +echo "##" +echo "" + +list=$( npm ls installed remote 2>/dev/null ) +echo "$list" +notlatest=$( echo "$list" | grep -v latest ) +if [ "$notlatest" != "" ]; then + echo "Failed: not latest" + echo $notlatest +else + echo "ok" +fi diff --git a/tools/osx-pkg-postinstall.sh b/tools/osx-pkg-postinstall.sh new file mode 100644 index 0000000000..c4c872fc78 --- /dev/null +++ b/tools/osx-pkg-postinstall.sh @@ -0,0 +1,5 @@ +#!/bin/sh +# TODO Can this be done inside the .pmdoc? +# TODO Can we extract $PREFIX from the installer? +cd /usr/local/bin +ln -sf ../lib/node_modules/npm/bin/npm-cli.js npm diff --git a/tools/osx-pkg.pmdoc/01local-contents.xml b/tools/osx-pkg.pmdoc/01local-contents.xml new file mode 100644 index 0000000000..bc1e5a791f --- /dev/null +++ b/tools/osx-pkg.pmdoc/01local-contents.xml @@ -0,0 +1 @@ +<pkg-contents spec="1.12"/> \ No newline at end of file diff --git a/tools/osx-pkg.pmdoc/01local.xml b/tools/osx-pkg.pmdoc/01local.xml index a4b48a818b..18fd871248 100644 --- a/tools/osx-pkg.pmdoc/01local.xml +++ b/tools/osx-pkg.pmdoc/01local.xml @@ -1 +1 @@ -<pkgref spec="1.12" uuid="053587FE-BDF3-4EF5-815D-281427431048"><config><identifier>org.nodejs.pkg</identifier><version>1.0</version><description></description><post-install type="none"/><requireAuthorization/><installFrom relative="true" mod="true">../out/dist-osx/usr/local/</installFrom><installTo mod="true" relocatable="true">/usr/local</installTo><flags><followSymbolicLinks/></flags><packageStore type="internal"></packageStore><mod>installTo.isRelativeType</mod><mod>installTo</mod><mod>locationType</mod><mod>relocatable</mod><mod>installFrom.path</mod><mod>installTo.isAbsoluteType</mod><mod>identifier</mod><mod>parent</mod><mod>installTo.path</mod><mod>installFrom.isRelativeType</mod></config><contents><file-list>01local-contents.xml</file-list><filter>/CVS$</filter><filter>/\.svn$</filter><filter>/\.cvsignore$</filter><filter>/\.cvspass$</filter><filter>/\.DS_Store$</filter></contents></pkgref> \ No newline at end of file +<pkgref spec="1.12" uuid="053587FE-BDF3-4EF5-815D-281427431048"><config><identifier>org.nodejs.pkg</identifier><version>1.0</version><description></description><post-install type="none"/><requireAuthorization/><installFrom relative="true" mod="true">../out/dist-osx/usr/local/</installFrom><installTo mod="true" relocatable="true">/usr/local</installTo><flags><followSymbolicLinks/></flags><packageStore type="internal"></packageStore><mod>installTo.isRelativeType</mod><mod>installTo</mod><mod>locationType</mod><mod>relocatable</mod><mod>installFrom.path</mod><mod>installTo.isAbsoluteType</mod><mod>identifier</mod><mod>parent</mod><mod>installTo.path</mod><mod>installFrom.isRelativeType</mod></config></pkgref> \ No newline at end of file diff --git a/tools/osx-pkg.pmdoc/02npm-contents.xml b/tools/osx-pkg.pmdoc/02npm-contents.xml new file mode 100644 index 0000000000..bc1e5a791f --- /dev/null +++ b/tools/osx-pkg.pmdoc/02npm-contents.xml @@ -0,0 +1 @@ +<pkg-contents spec="1.12"/> \ No newline at end of file diff --git a/tools/osx-pkg.pmdoc/02npm.xml b/tools/osx-pkg.pmdoc/02npm.xml new file mode 100644 index 0000000000..f97de66e06 --- /dev/null +++ b/tools/osx-pkg.pmdoc/02npm.xml @@ -0,0 +1 @@ +<pkgref spec="1.12" uuid="DF0233A3-6B5D-4FBF-8048-8FC57F42278F"><config><identifier>org.nodejs.node.npm.pkg</identifier><version>1.0</version><description></description><post-install type="none"/><requireAuthorization/><installFrom relative="true">../deps/npm</installFrom><installTo mod="true">/usr/local/lib/node_modules/npm</installTo><flags><followSymbolicLinks/></flags><packageStore type="internal"></packageStore><mod>installTo.path</mod><mod>installFrom.isRelativeType</mod><mod>installTo.isAbsoluteType</mod><mod>scripts.postinstall.isRelativeType</mod><mod>parent</mod><mod>installTo</mod></config><scripts><postinstall relative="true" mod="true">osx-pkg-postinstall.sh</postinstall></scripts></pkgref> \ No newline at end of file diff --git a/tools/osx-pkg.pmdoc/index.xml b/tools/osx-pkg.pmdoc/index.xml index 56b96bd2e4..d805747fcf 100644 --- a/tools/osx-pkg.pmdoc/index.xml +++ b/tools/osx-pkg.pmdoc/index.xml @@ -1,9 +1,9 @@ -<pkmkdoc spec="1.12"><properties><title>Node/Users/ryan/Desktop/Node.pkgorg.nodejs../doc/mac_osx_nodejs_installer_logo.pngNode/Users/ryan/Desktop/Node.pkgorg.nodejs../doc/mac_osx_nodejs_installer_logo.png01local.xmlproperties.titleproperties.userDomainproperties.anywhereDomainproperties.systemDomain \ No newline at end of file +NPM was installed at\ +\ + /usr/local/bin/npm\ +\ +Make sure that /usr/local/bin is in your $PATH.}]]>01local.xml02npm.xmlproperties.titleproperties.userDomainproperties.anywhereDomainproperties.systemDomain \ No newline at end of file

+
+
+Schily's USER COMMANDS                                   STAR(4L)
+
+
+

UTw;G%FPskE>7l`A`x3p>v41ZK0+RtLuo-`& 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 W%*VYdx+ea&A19cPKV@d;`~Lv_DxZx2 literal 0 HcmV?d00001 diff --git a/deps/npm/html/npm.png b/deps/npm/html/npm.png new file mode 100644 index 0000000000000000000000000000000000000000..d78ff53f2affa1bb4a9943619ea263633f8dba5f GIT binary patch literal 1164 zcmeAS@N?(olHy`uVBq!ia0y~yU}6WdPjRpTNkfmv6+nu!z$3Dlfq`2Xgc%uT&5>YW zV6pRbaSW-r_4elJEa5_t0}uaiN!FYE$nA>L8@Geo7K%8DU)#H{MMF#DRd7cwtN7C2 z*SPmJ#Lf^85Mc3MTj9<#DNA&=fOFJ#y`^t^-ah#q_u#<8luzDjRh2(qr8UUDs{1n` z#WH_I#2vemnk~tn-yQ$!V8wU-%$?rb>%)XrKUnah+-&i^W9H`FzZ<8h1P5PL%w}Nt zr^d&?P{_d0!OY+w!@%%}fkA+sp}~TI;Rqvx0EI`z{e||-h4ydF^EdxBjn!%k+N#BOW2e6N_rurOy=R_VqkE70$SF?3|Q$K#quG(Vl9OU0*~XCXq~G{c>XmiVsD}L_wKiK|EByr zQ(j!0xFe+Os^x{69~wC(QeqpA|GL}vy#M+6#$W$ltzFOWCiBI5_QUJl&*ueY%DlU^ zO7+Q+_i2}(tMEoge|z=w@$c92v~}_(rn1K`-^|R7KGA+r`?_B5-97sE@8>ddD7M(&NSKz88J%z+SiCTJy85}Sb4q9e E0I8{5D*ylh literal 0 HcmV?d00001 diff --git a/deps/npm/lib/adduser.js b/deps/npm/lib/adduser.js new file mode 100644 index 0000000000..b6d08ed760 --- /dev/null +++ b/deps/npm/lib/adduser.js @@ -0,0 +1,50 @@ + +module.exports = adduser + +var registry = require("./utils/npm-registry-client/index.js") + , ini = require("./utils/ini.js") + , log = require("./utils/log.js") + , npm = require("./npm.js") + , prompt = require("./utils/prompt.js") + , promiseChain = require("./utils/promise-chain.js") + , crypto + +try { + crypto = process.binding("crypto") && require("crypto") +} catch (ex) {} + +adduser.usage = "npm adduser\nThen enter stuff at the prompts" + +function adduser (args, cb) { + if (!crypto) return cb(new Error( + "You must compile node with ssl support to use the adduser feature")) + + var u = { u : npm.config.get("username") + , p : npm.config.get("_password") + , e : npm.config.get("email") + } + , changed = false + + promiseChain(cb) + (prompt, ["Username: ", u.u], function (un) { + changed = u.u !== un + u.u = un + }) + (function (cb) { + if (u.p && !changed) return cb(null, u.p) + prompt("Password: ", u.p, true, cb) + }, [], function (pw) { u.p = pw }) + (prompt, ["Email: ", u.e], function (em) { u.e = em }) + (function (cb) { + if (changed) npm.config.del("_auth") + registry.adduser(u.u, u.p, u.e, function (er) { + if (er) return cb(er) + ini.set("username", u.u, "user") + ini.set("_password", u.p, "user") + ini.set("email", u.e, "user") + log("Authorized user " + u.u, "adduser") + ini.save("user", cb) + }) + }) + () +} diff --git a/deps/npm/lib/bin.js b/deps/npm/lib/bin.js new file mode 100644 index 0000000000..bd82bc9e31 --- /dev/null +++ b/deps/npm/lib/bin.js @@ -0,0 +1,19 @@ +module.exports = bin + +var npm = require("./npm.js") + , output = require("./utils/output.js") + +bin.usage = "npm bin\nnpm bin -g\n(just prints the bin folder)" + +function bin (args, cb) { + var path = require("path") + , b = npm.bin + , PATH = (process.env.PATH || "").split(":") + + output.write(b, function (er) { cb(er, b) }) + + if (npm.config.get("global") && PATH.indexOf(b) === -1) { + output.write("(not in PATH env variable)" + ,npm.config.get("logfd")) + } +} diff --git a/deps/npm/lib/bugs.js b/deps/npm/lib/bugs.js new file mode 100644 index 0000000000..1c52ffda0e --- /dev/null +++ b/deps/npm/lib/bugs.js @@ -0,0 +1,50 @@ + +module.exports = bugs + +bugs.usage = "npm bugs " + +bugs.completion = function (opts, cb) { + if (opts.conf.argv.remain.length > 2) return cb() + registry.get("/-/short", null, 60000, function (er, list) { + return cb(null, list || []) + }) +} + +var exec = require("./utils/exec.js") + , registry = require("./utils/npm-registry-client/index.js") + , npm = require("./npm.js") + , log = require("./utils/log.js") + +function bugs (args, cb) { + if (!args.length) return cb(bugs.usage) + var n = args[0].split("@").shift() + registry.get(n, "latest", 3600, function (er, d) { + if (er) return cb(er) + var bugs = d.bugs + , repo = d.repository || d.repositories + if (bugs) { + if (typeof bugs === "string") return open(bugs, cb) + if (bugs.url) return open(bugs.url, cb) + } + if (repo) { + if (Array.isArray(repo)) repo = repo.shift() + if (repo.url) repo = repo.url + log.verbose(repo, "repository") + if (repo && repo.match(/^(https?:\/\/|git(:\/\/|@))github.com/)) { + return open(repo.replace(/^git(@|:\/\/)/, "http://") + .replace(/^https?:\/\/github.com:/, "github.com/") + .replace(/\.git$/, '')+"/issues", cb) + } + } + return open("http://search.npmjs.org/#/" + d.name, cb) + }) +} + +function open (url, cb) { + exec(npm.config.get("browser"), [url], log.er(cb, + "Failed to open "+url+" in a browser. It could be that the\n"+ + "'browser' config is not set. Try doing this:\n"+ + " npm config set browser google-chrome\n"+ + "or:\n"+ + " npm config set browser lynx\n")) +} diff --git a/deps/npm/lib/build.js b/deps/npm/lib/build.js new file mode 100644 index 0000000000..864eb27cf6 --- /dev/null +++ b/deps/npm/lib/build.js @@ -0,0 +1,186 @@ + +// npm build command + +// everything about the installation after the creation of +// the .npm/{name}/{version}/package folder. +// linking the modules into the npm.root, +// resolving dependencies, etc. + +// This runs AFTER install or link are completed. + +var npm = require("./npm.js") + , log = require("./utils/log.js") + , chain = require("slide").chain + , fs = require("graceful-fs") + , path = require("path") + , lifecycle = require("./utils/lifecycle.js") + , readJson = require("./utils/read-json.js") + , link = require("./utils/link.js") + , linkIfExists = link.ifExists + , cmdShim = require("./utils/cmd-shim.js") + , cmdShimIfExists = cmdShim.ifExists + , asyncMap = require("slide").asyncMap + , output = require("./utils/output.js") + +module.exports = build +build.usage = "npm build \n(this is plumbing)" + +build._didBuild = {} +build._noLC = {} +function build (args, global, didPre, didRB, cb) { + if (typeof cb !== "function") cb = didRB, didRB = false + if (typeof cb !== "function") cb = didPre, didPre = false + if (typeof cb !== "function") { + cb = global, global = npm.config.get("global") + } + // it'd be nice to asyncMap these, but actually, doing them + // in parallel generally munges up the output from node-waf + var builder = build_(global, didPre, didRB) + chain(args.map(function (arg) { return function (cb) { + builder(arg, cb) + }}), cb) +} + +function build_ (global, didPre, didRB) { return function (folder, cb) { + folder = path.resolve(folder) + build._didBuild[folder] = true + log.info(folder, "build") + readJson(path.resolve(folder, "package.json"), function (er, pkg) { + if (er) return cb(er) + chain + ( [ !didPre && [lifecycle, pkg, "preinstall", folder] + , [linkStuff, pkg, folder, global, didRB] + , pkg.name === "npm" && [writeBuiltinConf, folder] + , didPre !== build._noLC && [lifecycle, pkg, "install", folder] + , didPre !== build._noLC && [lifecycle, pkg, "postinstall", folder] + , didPre !== build._noLC + && npm.config.get("npat") + && [lifecycle, pkg, "test", folder] ] + , cb ) + }) +}} + +function writeBuiltinConf (folder, cb) { + // the builtin config is "sticky". Any time npm installs itself, + // it puts its builtin config file there, as well. + var ini = require("./utils/ini.js") + ini.saveConfig("builtin", path.resolve(folder, "npmrc"), cb) +} + +function linkStuff (pkg, folder, global, didRB, cb) { + // if it's global, and folder is in {prefix}/node_modules, + // then bins are in {prefix}/bin + // otherwise, then bins are in folder/../.bin + var parent = path.dirname(folder) + , gnm = global && npm.globalDir + , top = parent === npm.dir + , gtop = parent === gnm + + log.verbose([global, gnm, gtop, parent], "linkStuff") + log(pkg._id, "linkStuff") + + if (top && pkg.preferGlobal && !global) { + log.warn(pkg._id + " should be installed with -g", "prefer global") + } + + asyncMap( [linkBins, linkMans, !didRB && rebuildBundles] + , function (fn, cb) { + if (!fn) return cb() + log.verbose(pkg._id, fn.name) + fn(pkg, folder, parent, gtop, cb) + }, cb) +} + +function rebuildBundles (pkg, folder, parent, gtop, cb) { + if (!npm.config.get("rebuild-bundle")) return cb() + + var deps = Object.keys(pkg.dependencies || {}) + .concat(Object.keys(pkg.devDependencies || {})) + , bundles = pkg.bundleDependencies || pkg.bundledDependencies || [] + + fs.readdir(path.resolve(folder, "node_modules"), function (er, files) { + // error means no bundles + if (er) return cb() + + log.verbose(files, "rebuildBundles") + // don't asyncMap these, because otherwise build script output + // gets interleaved and is impossible to read + chain(files.filter(function (file) { + // rebuild if: + // not a .folder, like .bin or .hooks + return file.charAt(0) !== "." + // not some old 0.x style bundle + && file.indexOf("@") === -1 + // either not a dep, or explicitly bundled + && (deps.indexOf(file) === -1 || bundles.indexOf(file) !== -1) + }).map(function (file) { + file = path.resolve(folder, "node_modules", file) + return function (cb) { + if (build._didBuild[file]) return cb() + log.verbose(file, "rebuild bundle") + // if file is not a package dir, then don't do it. + fs.lstat(path.resolve(file, "package.json"), function (er, st) { + if (er) return cb() + build_(false)(file, cb) + }) + }}), cb) + }) +} + +function linkBins (pkg, folder, parent, gtop, cb) { + if (!pkg.bin || !gtop && path.basename(parent) !== "node_modules") { + return cb() + } + var binRoot = gtop ? npm.globalBin + : path.resolve(parent, ".bin") + log.verbose([pkg.bin, binRoot, gtop], "bins linking") + + asyncMap(Object.keys(pkg.bin), function (b, cb) { + linkBin( path.resolve(folder, pkg.bin[b]) + , path.resolve(binRoot, b) + , gtop && folder + , function (er) { + if (er) return cb(er) + // bins should always be executable. + // XXX skip chmod on windows? + fs.chmod(path.resolve(folder, pkg.bin[b]), npm.modes.exec, function (er) { + if (er || !gtop) return cb(er) + var dest = path.resolve(binRoot, b) + , src = path.resolve(folder, pkg.bin[b]) + , out = npm.config.get("parseable") + ? dest + "::" + src + ":BINFILE" + : dest + " -> " + src + output.write(out, cb) + }) + }) + }, cb) +} + +function linkBin (from, to, gently, cb) { + if (process.platform !== "win32") { + return linkIfExists(from, to, gently, cb) + } else { + return cmdShimIfExists(from, to, cb) + } +} + +function linkMans (pkg, folder, parent, gtop, cb) { + if (!pkg.man || !gtop || process.platform === "win32") return cb() + var manRoot = path.resolve(npm.config.get("prefix"), "share", "man") + asyncMap(pkg.man, function (man, cb) { + if (typeof man !== "string") return cb() + var parseMan = man.match(/(.*)\.([0-9]+)(\.gz)?$/) + , stem = parseMan[1] + , sxn = parseMan[2] + , gz = parseMan[3] || "" + , bn = path.basename(stem) + , manSrc = path.join( folder, man ) + , manDest = path.join( manRoot + , "man"+sxn + , (bn.indexOf(pkg.name) === 0 ? bn + : pkg.name + "-" + bn) + + "." + sxn + gz + ) + linkIfExists(manSrc, manDest, gtop && folder, cb) + }, cb) +} diff --git a/deps/npm/lib/cache.js b/deps/npm/lib/cache.js new file mode 100644 index 0000000000..bde72c4f95 --- /dev/null +++ b/deps/npm/lib/cache.js @@ -0,0 +1,735 @@ +// XXX lib/utils/tar.js and this file need to be rewritten. + +/* +adding a folder: +1. tar into tmp/random/package.tgz +2. untar into tmp/random/contents/{blah} +3. rename {blah} to "package" +4. tar tmp/random/contents/package to cache/n/v/package.tgz +5. untar cache/n/v/package.tgz into cache/n/v/package +6. rm tmp/random + +Adding a url: +1. fetch to tmp/random/package.tgz +2. goto folder(2) + +adding a name@version: +1. registry.get(name, version) +2. if response isn't 304, add url(dist.tarball) + +adding a name@range: +1. registry.get(name) +2. Find a version that satisfies +3. add name@version + +adding a local tarball: +1. untar to tmp/random/{blah} +2. goto folder(2) +*/ + +exports = module.exports = cache +exports.read = read +exports.clean = clean +exports.unpack = unpack + +var mkdir = require("./utils/mkdir-p.js") + , exec = require("./utils/exec.js") + , fetch = require("./utils/fetch.js") + , npm = require("./npm.js") + , fs = require("graceful-fs") + , rm = require("rimraf") + , readJson = require("./utils/read-json.js") + , registry = require("./utils/npm-registry-client/index.js") + , log = require("./utils/log.js") + , path = require("path") + , output + , sha = require("./utils/sha.js") + , find = require("./utils/find.js") + , asyncMap = require("slide").asyncMap + , semver = require("semver") + , tar = require("./utils/tar.js") + , fileCompletion = require("./utils/completion/file-completion.js") + , url = require("url") + +cache.usage = "npm cache add " + + "\nnpm cache add " + + "\nnpm cache add " + + "\nnpm cache add " + + "\nnpm cache add @" + + "\nnpm cache ls []" + + "\nnpm cache clean [[@]]" + +cache.completion = function (opts, cb) { + + var argv = opts.conf.argv.remain + if (argv.length === 2) { + return cb(null, ["add", "ls", "clean"]) + } + + switch (argv[2]) { + case "clean": + case "ls": + // cache and ls are easy, because the completion is + // what ls_ returns anyway. + // just get the partial words, minus the last path part + var p = path.dirname(opts.partialWords.slice(3).join("/")) + if (p === ".") p = "" + return ls_(p, 2, cb) + case "add": + // Same semantics as install and publish. + return npm.commands.install.completion(opts, cb) + } +} + +function cache (args, cb) { + var cmd = args.shift() + switch (cmd) { + case "rm": case "clear": case "clean": return clean(args, cb) + case "list": case "sl": case "ls": return ls(args, cb) + case "add": return add(args, cb) + default: return cb(new Error( + "Invalid cache action: "+cmd)) + } +} + +// if the pkg and ver are in the cache, then +// just do a readJson and return. +// if they're not, then fetch them from the registry. +var cacheSeen = {} +function read (name, ver, forceBypass, cb) { + if (typeof cb !== "function") cb = forceBypass, forceBypass = true + var jsonFile = path.join(npm.cache, name, ver, "package", "package.json") + function c (er, data) { + if (!er) cacheSeen[data._id] = data + if (data) deprCheck(data) + return cb(er, data) + } + + if (forceBypass + && (npm.config.get("force") + || process.platform === "cygwin")) { + log.verbose(true, "force found, skipping cache") + return addNamed(name, ver, c) + } + + if (name+"@"+ver in cacheSeen) { + return cb(null, cacheSeen[name+"@"+ver]) + } + + readJson(jsonFile, function (er, data) { + if (er) return addNamed(name, ver, c) + deprCheck(data) + c(er, data) + }) +} + +// npm cache ls [] +function ls (args, cb) { + output = output || require("./utils/output.js") + args = args.join("/").split("@").join("/") + if (args.substr(-1) === "/") args = args.substr(0, args.length - 1) + ls_(args, npm.config.get("depth"), function(er, files) { + output.write(files.map(function (f) { + return path.join("~/.npm", f) + }).join("\n").trim(), function (er) { + return cb(er, files) + }) + }) +} + +// Calls cb with list of cached pkgs matching show. +function ls_ (req, depth, cb) { + return fileCompletion(npm.cache, req, depth, cb) +} + +// npm cache clean [] +function clean (args, cb) { + if (!cb) cb = args, args = [] + if (!args) args = [] + args = args.join("/").split("@").join("/") + if (args.substr(-1) === "/") args = args.substr(0, args.length - 1) + var f = path.join(npm.cache, path.normalize(args)) + if (f === npm.cache) { + fs.readdir(npm.cache, function (er, files) { + if (er) return cb() + asyncMap( files.filter(function (f) { + return npm.config.get("force") || f !== "-" + }).map(function (f) { + return path.join(npm.cache, f) + }) + , rm, cb ) + }) + } else rm(path.join(npm.cache, path.normalize(args)), cb) +} + +// npm cache add +// npm cache add +// npm cache add +// npm cache add +exports.add = function (pkg, ver, scrub, cb) { + if (typeof cb !== "function") cb = scrub, scrub = false + if (typeof cb !== "function") cb = ver, ver = null + if (scrub) { + return clean([], function (er) { + if (er) return cb(er) + add([pkg, ver], cb) + }) + } + log.verbose([pkg, ver], "cache add") + return add([pkg, ver], cb) +} + +function add (args, cb) { + // this is hot code. almost everything passes through here. + // the args can be any of: + // ["url"] + // ["pkg", "version"] + // ["pkg@version"] + // ["pkg", "url"] + // This is tricky, because urls can contain @ + // Also, in some cases we get [name, null] rather + // that just a single argument. + + var usage = "Usage:\n" + + " npm cache add \n" + + " npm cache add @\n" + + " npm cache add \n" + + " npm cache add \n" + , name + , spec + + if (args[1] === undefined) args[1] = null + + // at this point the args length must ==2 + if (args[1] !== null) { + name = args[0] + spec = args[1] + } else if (args.length === 2) { + spec = args[0] + } + + log.silly([name, spec, args], "cache add: name, spec, args") + + if (!name && !spec) return cb(usage) + + // see if the spec is a url + // otherwise, treat as name@version + var p = url.parse(spec.replace(/^git\+/, "git")) || {} + log.verbose(p, "parsed url") + + // it could be that we got name@http://blah + // in that case, we will not have a protocol now, but if we + // split and check, we will. + if (!name && !p.protocol && spec.indexOf("@") !== -1) { + spec = spec.split("@") + name = spec.shift() + spec = spec.join("@") + return add([name, spec], cb) + } + + switch (p.protocol) { + case "http:": + case "https:": + return addRemoteTarball(spec, null, name, cb) + case "git:": + case "githttp:": + case "githttps:": + case "gitrsync:": + case "gitftp:": + case "gitssh:": + //p.protocol = p.protocol.replace(/^git([^:])/, "$1") + return addRemoteGit(spec, p, name, cb) + default: + // if we have a name and a spec, then try name@spec + // if not, then try just spec (which may try name@"" if not found) + return name ? addNamed(name, spec, cb) : addLocal(spec, cb) + } +} + +// Only have a single download action at once for a given url +// additional calls stack the callbacks. +var inFlightURLs = {} +function addRemoteTarball (u, shasum, name, cb_) { + if (typeof cb_ !== "function") cb_ = name, name = "" + if (typeof cb_ !== "function") cb_ = shasum, shasum = null + + if (!inFlightURLs[u]) inFlightURLs[u] = [] + var iF = inFlightURLs[u] + iF.push(cb_) + if (iF.length > 1) return + + function cb (er, data) { + var c + while (c = iF.shift()) c(er, data) + delete inFlightURLs[u] + } + + log.verbose([u, shasum], "addRemoteTarball") + var tmp = path.join(npm.tmp, Date.now()+"-"+Math.random(), "tmp.tgz") + mkdir(path.dirname(tmp), function (er) { + if (er) return cb(er) + fetch(u, tmp, function (er) { + if (er) return log.er(cb, "failed to fetch "+u)(er) + if (!shasum) return done() + // validate that the url we just downloaded matches the expected shasum. + sha.check(tmp, shasum, done) + }) + }) + function done (er) { + if (er) return cb(er) + addLocalTarball(tmp, name, cb) + } +} + +// For now, this is kind of dumb. Just basically treat git as +// yet another "fetch and scrub" kind of thing. +// Clone to temp folder, then proceed with the addLocal stuff. +function addRemoteGit (u, parsed, name, cb_) { + if (typeof cb_ !== "function") cb_ = name, name = null + + if (!inFlightURLs[u]) inFlightURLs[u] = [] + var iF = inFlightURLs[u] + iF.push(cb_) + if (iF.length > 1) return + + function cb (er, data) { + var c + while (c = iF.shift()) c(er, data) + delete inFlightURLs[u] + } + + // figure out what we should check out. + var co = parsed.hash && parsed.hash.substr(1) || "master" + u = u.replace(/^git\+/, "") + .replace(/#.*$/, "") + .replace(/^ssh:\/\//, "") // ssh is the default anyway + log.verbose([u, co], "addRemoteGit") + + var tmp = path.join(npm.tmp, Date.now()+"-"+Math.random()) + mkdir(path.dirname(tmp), function (er) { + if (er) return cb(er) + exec( "git", ["clone", u, tmp], null, false + , function (er, code, stdout, stderr) { + stdout = (stdout + "\n" + stderr).trim() + if (er) { + log.error(stdout, "git clone "+u) + return cb(er) + } + log.verbose(stdout, "git clone "+u) + exec( "git", ["checkout", co], null, false, tmp + , function (er, code, stdout, stderr) { + stdout = (stdout + "\n" + stderr).trim() + if (er) { + log.error(stdout, "git checkout "+co) + return cb(er) + } + log.verbose(stdout, "git checkout "+co) + addLocalDirectory(tmp, cb) + }) + }) + }) +} + + +// only have one request in flight for a given +// name@blah thing. +var inFlightNames = {} +function addNamed (name, x, cb_) { + log.info([name, x], "addNamed") + var k = name + "@" + x + if (!inFlightNames[k]) inFlightNames[k] = [] + var iF = inFlightNames[k] + iF.push(cb_) + if (iF.length > 1) return + + function cb (er, data) { + var c + while (c = iF.shift()) c(er, data) + delete inFlightNames[k] + } + + log.verbose([semver.valid(x), semver.validRange(x)], "addNamed") + return ( null !== semver.valid(x) ? addNameVersion + : null !== semver.validRange(x) ? addNameRange + : addNameTag + )(name, x, cb) +} + +function addNameTag (name, tag, cb) { + log([name, tag], "addNameTag") + var explicit = true + if (!tag) { + explicit = false + tag = npm.config.get("tag") + } + + registry.get(name, function (er, data, json, response) { + if (er) return cb(er) + engineFilter(data) + if (data["dist-tags"] && data["dist-tags"][tag] + && data.versions[data["dist-tags"][tag]]) { + return addNameVersion(name, data["dist-tags"][tag], cb) + } + if (!explicit && Object.keys(data.versions).length) { + return addNameRange(name, "*", cb) + } + return cb(installTargetsError(tag, data)) + }) +} + +function engineFilter (data) { + var npmv = npm.version + , nodev = npm.config.get("node-version") + + if (!nodev || npm.config.get("force")) return data + + Object.keys(data.versions || {}).forEach(function (v) { + var eng = data.versions[v].engines + if (!eng) return + if (eng.node && !semver.satisfies(nodev, eng.node) + || eng.npm && !semver.satisfies(npmv, eng.npm)) { + delete data.versions[v] + } + }) +} + +function addNameRange (name, range, cb) { + range = semver.validRange(range) + if (range === null) return cb(new Error( + "Invalid version range: "+range)) + registry.get(name, function (er, data, json, response) { + if (er) return cb(er) + engineFilter(data) + // if the tagged version satisfies, then use that. + var tagged = data["dist-tags"][npm.config.get("tag")] + if (tagged && data.versions[tagged] && semver.satisfies(tagged, range)) { + return addNameVersion(name, tagged, cb) + } + // find the max satisfying version. + var ms = semver.maxSatisfying(Object.keys(data.versions || {}), range) + if (!ms) { + return cb(installTargetsError(range, data)) + } + addNameVersion(name, ms, cb) + }) +} + +function installTargetsError (requested, data) { + var targets = Object.keys(data["dist-tags"]).filter(function (f) { + return (data.versions || {}).hasOwnProperty(f) + }).concat(Object.keys(data.versions || {})) + + requested = data.name + (requested ? "@'" + requested + "'" : "") + + targets = targets.length + ? "Valid install targets:\n" + JSON.stringify(targets) + : "No valid targets found.\n" + + "Perhaps not compatible with your version of node?" + + return new Error( "No compatible version found: " + + requested + "\n" + targets) +} + +function addNameVersion (name, ver, cb) { + ver = semver.valid(ver) + if (ver === null) return cb(new Error("Invalid version: "+ver)) + registry.get(name, ver, function (er, data, json, response) { + if (er) return cb(er) + deprCheck(data) + var dist = data.dist + + if (!dist) return cb(new Error("No dist in "+data._id+" package")) + + var bd = npm.config.get("bindist") + , b = dist.bin && bd && dist.bin[bd] + log.verbose([bd, dist], "bin dist") + if (b && b.tarball && b.shasum) { + log.info(data._id, "prebuilt") + log.verbose(b, "prebuilt "+data._id) + dist = b + } + + if (!dist.tarball) return cb(new Error( + "No dist.tarball in " + data._id + " package")) + + if (response.statusCode !== 304 || npm.config.get("force") + || process.platform === "cygwin") { + return fetchit() + } + + // we got cached data, so let's see if we have a tarball. + fs.stat(path.join(npm.cache, name, ver, "package.tgz"), function (er, s) { + if (!er) readJson( path.join( npm.cache, name, ver + , "package", "package.json" ) + , function (er, data) { + if (er) return fetchit() + return cb(null, data) + }) + else return fetchit() + }) + + function fetchit () { + // use the same protocol as the registry. + // https registry --> https tarballs. + var tb = url.parse(dist.tarball) + tb.protocol = url.parse(npm.config.get("registry")).protocol + delete tb.href + tb = url.format(tb) + return addRemoteTarball( tb + , dist.shasum + , name+"-"+ver + , cb ) + } + }) +} + +function addLocal (p, name, cb_) { + if (typeof cb_ !== "function") cb_ = name, name = "" + + function cb (er, data) { + if (er) { + // if it doesn't have a / in it, it might be a + // remote thing. + if (p.indexOf("/") === -1 && p.charAt(0) !== ".") { + return addNamed(p, "", cb_) + } + return log.er(cb_, "Could not install: "+p)(er) + } + return cb_(er, data) + } + + // figure out if this is a folder or file. + fs.stat(p, function (er, s) { + if (er) return cb(er) + if (s.isDirectory()) addLocalDirectory(p, name, cb) + else addLocalTarball(p, name, cb) + }) +} + +function addLocalTarball (p, name, cb) { + if (typeof cb !== "function") cb = name, name = "" + // if it's a tar, and not in place, + // then unzip to .tmp, add the tmp folder, and clean up tmp + if (p.indexOf(npm.tmp) === 0) return addTmpTarball(p, name, cb) + + if (p.indexOf(npm.cache) === 0) { + if (path.basename(p) !== "package.tgz") return cb(new Error( + "Not a valid cache tarball name: "+p)) + return addPlacedTarball(p, name, cb) + } + + // just copy it over and then add the temp tarball file. + var tmp = path.join(npm.tmp, name + Date.now() + + "-" + Math.random(), "tmp.tgz") + mkdir(path.dirname(tmp), function (er) { + if (er) return cb(er) + var from = fs.createReadStream(p) + , to = fs.createWriteStream(tmp) + , errState = null + function errHandler (er) { + if (errState) return + return cb(errState = er) + } + from.on("error", errHandler) + to.on("error", errHandler) + to.on("close", function () { + if (errState) return + log.verbose(npm.modes.file.toString(8), "chmod "+tmp) + fs.chmod(tmp, npm.modes.file, function (er) { + if (er) return cb(er) + addTmpTarball(tmp, name, cb) + }) + }) + from.pipe(to) + }) +} + +// to maintain the cache dir's permissions consistently. +var cacheStat = null +function getCacheStat (cb) { + if (cacheStat) return cb(null, cacheStat) + fs.stat(npm.cache, function (er, st) { + if (er) return makeCacheDir(cb) + if (!st.isDirectory()) { + return log.er(cb, "invalid cache directory: "+npm.cache)(er) + } + return cb(null, cacheStat = st) + }) +} + +function makeCacheDir (cb) { + if (!process.getuid) return mkdir(npm.cache, npm.modes.exec, cb) + + var uid = +process.getuid() + , gid = +process.getgid() + + if (uid === 0) { + if (process.env.SUDO_UID) uid = +process.env.SUDO_UID + if (process.env.SUDO_GID) gid = +process.env.SUDO_GID + } + if (uid !== 0 || !process.env.HOME) { + cacheStat = {uid: uid, gid: gid} + return mkdir(npm.cache, npm.modes.exec, uid, gid, function (er) { + return cb(er, cacheStat) + }) + } + fs.stat(process.env.HOME, function (er, st) { + if (er) return log.er(cb, "homeless?")(er) + cacheStat = st + log.silly([st.uid, st.gid], "uid, gid for cache dir") + return mkdir(npm.cache, npm.modes.exec, st.uid, st.gid, function (er) { + return cb(er, cacheStat) + }) + }) +} + + + + +function addPlacedTarball (p, name, cb) { + if (!cb) cb = name, name = "" + getCacheStat(function (er, cs) { + if (er) return cb(er) + return addPlacedTarball_(p, name, cs.uid, cs.gid, cb) + }) +} + +function addPlacedTarball_ (p, name, uid, gid, cb) { + // now we know it's in place already as .cache/name/ver/package.tgz + // unpack to .cache/name/ver/package/, read the package.json, + // and fire cb with the json data. + var target = path.dirname(p) + , folder = path.join(target, "package") + + rm(folder, function (er) { + if (er) return log.er(cb, "Could not remove "+folder)(er) + tar.unpack(p, folder, null, null, uid, gid, function (er) { + if (er) return log.er(cb, "Could not unpack "+p+" to "+target)(er) + // calculate the sha of the file that we just unpacked. + // this is so that the data is available when publishing. + sha.get(p, function (er, shasum) { + if (er) return log.er(cb, "couldn't validate shasum of "+p)(er) + readJson(path.join(folder, "package.json"), function (er, data) { + if (er) return log.er(cb, "couldn't read json in "+folder)(er) + data.dist = data.dist || {} + if (shasum) data.dist.shasum = shasum + deprCheck(data) + asyncMap([p], function (f, cb) { + log.verbose(npm.modes.file.toString(8), "chmod "+f) + fs.chmod(f, npm.modes.file, cb) + }, function (f, cb) { + if (process.platform === "win32") { + log.silly(f, "skipping chown for windows") + cb() + } else if (typeof uid === "number" + && typeof gid === "number" + && parseInt(uid, 10) === uid + && parseInt(gid, 10) === gid) { + log.verbose([f, uid, gid], "chown") + fs.chown(f, uid, gid, cb) + } else { + log.verbose([f, uid, gid], "not chowning, invalid uid/gid") + cb() + } + }, function (er) { + cb(er, data) + }) + }) + }) + }) + }) +} + +function addLocalDirectory (p, name, cb) { + if (typeof cb !== "function") cb = name, name = "" + // if it's a folder, then read the package.json, + // tar it to the proper place, and add the cache tar + if (p.indexOf(npm.cache) === 0) return cb(new Error( + "Adding a cache directory to the cache will make the world implode.")) + readJson(path.join(p, "package.json"), function (er, data) { + if (er) return cb(er) + deprCheck(data) + var random = Date.now() + "-" + Math.random() + , tmp = path.join(npm.tmp, random) + , tmptgz = path.resolve(tmp, "tmp.tgz") + , placed = path.resolve( npm.cache, data.name + , data.version, "package.tgz" ) + , placeDirect = path.basename(p) === "package" + , tgz = placeDirect ? placed : tmptgz + , doFancyCrap = p.indexOf(npm.tmp) !== 0 + && p.indexOf(npm.cache) !== 0 + tar.pack(tgz, p, data, doFancyCrap, function (er) { + if (er) return log.er(cb,"couldn't pack "+p+ " to "+tgz)(er) + addLocalTarball(tgz, name, cb) + }) + }) +} + +function addTmpTarball (tgz, name, cb) { + if (!cb) cb = name, name = "" + getCacheStat(function (er, cs) { + if (er) return cb(er) + return addTmpTarball_(tgz, name, cs.uid, cs.gid, cb) + }) +} + +function addTmpTarball_ (tgz, name, uid, gid, cb) { + var contents = path.resolve(path.dirname(tgz)) // , "contents") + tar.unpack( tgz, path.resolve(contents, "package") + , null, null + , uid, gid + , function (er) { + if (er) return log.er(cb, "couldn't unpack "+tgz+" to "+contents)(er) + fs.readdir(contents, function (er, folder) { + if (er) return log.er(cb, "couldn't readdir "+contents)(er) + log.verbose(folder, "tarball contents") + if (folder.length > 1) { + folder = folder.filter(function (f) { + return !f.match(/^\.|^tmp\.tgz$/) + }) + } + if (folder.length > 1) { + log.warn(folder.slice(1).join("\n") + ,"extra junk in folder, ignoring") + } + if (!folder.length) return cb(new Error("Empty package tarball")) + folder = path.join(contents, folder[0]) + var newName = path.join(contents, "package") + fs.rename(folder, newName, function (er) { + if (er) return log.er(cb, "couldn't rename "+folder+" to package")(er) + addLocalDirectory(newName, name, cb) + }) + }) + }) +} + +function unpack (pkg, ver, unpackTarget, dMode, fMode, uid, gid, cb) { + if (typeof cb !== "function") cb = gid, gid = null + if (typeof cb !== "function") cb = uid, uid = null + if (typeof cb !== "function") cb = fMode, fMode = null + if (typeof cb !== "function") cb = dMode, dMode = null + + read(pkg, ver, false, function (er, data) { + if (er) { + log.error("Could not read data for "+pkg+"@"+ver) + return cb(er) + } + tar.unpack( path.join(npm.cache, pkg, ver, "package.tgz") + , unpackTarget + , dMode, fMode + , uid, gid + , cb ) + }) +} + +var deprecated = {} + , deprWarned = {} +function deprCheck (data) { + if (deprecated[data._id]) data.deprecated = deprecated[data._id] + if (data.deprecated) deprecated[data._id] = data.deprecated + else return + if (!deprWarned[data._id]) { + deprWarned[data._id] = true + log.warn(data._id+": "+data.deprecated, "deprecated") + } +} diff --git a/deps/npm/lib/completion.js b/deps/npm/lib/completion.js new file mode 100644 index 0000000000..bc97bf1af9 --- /dev/null +++ b/deps/npm/lib/completion.js @@ -0,0 +1,245 @@ + +module.exports = completion + +completion.usage = "npm completion >> ~/.bashrc\n" + + "npm completion >> ~/.zshrc\n" + + "source <(npm completion)" + +var output = require("./utils/output.js") + , configDefs = require("./utils/config-defs.js") + , configTypes = configDefs.types + , shorthands = configDefs.shorthands + , nopt = require("nopt") + , configNames = Object.keys(configTypes).filter(function (e) { + return e.charAt(0) !== "_" + }) + , shorthandNames = Object.keys(shorthands) + , allConfs = configNames.concat(shorthandNames) + , npm = require("./npm.js") + +completion.completion = function (opts, cb) { + if (opts.w > 3) return cb() + + var fs = require("fs") + , path = require("path") + , bashExists = null + , zshExists = null + , bashProfExists = null + fs.stat(path.resolve(process.env.HOME, ".bashrc"), function (er, b) { + bashExists = !er + next() + }) + fs.stat(path.resolve(process.env.HOME, ".zshrc"), function (er, b) { + zshExists = !er + next() + }) + function next () { + if (zshExists === null || bashExists === null) return + var out = [] + if (zshExists) out.push("~/.zshrc") + if (bashExists) out.push("~/.bashrc") + if (opts.w === 2) out = out.map(function (m) { + return [">>", m] + }) + cb(null, out) + } +} + +function completion (args, cb) { + if (process.platform === "win32") { + var e = new Error("npm completion not supported on windows") + e.code = "ENOTSUP" + e.errno = require("constants").ENOTSUP + return cb(e) + } + + // if the COMP_* isn't in the env, then just dump the script. + if (process.env.COMP_CWORD === undefined + ||process.env.COMP_LINE === undefined + ||process.env.COMP_POINT === undefined + ) return dumpScript(cb) + + console.error(process.env.COMP_CWORD) + console.error(process.env.COMP_LINE) + console.error(process.env.COMP_POINT) + + //console.log("abracadabrasauce\nabracad cat monger") + //if (Math.random() * 3 < 1) console.log("man\\ bear\\ pig") + //else if (Math.random() * 3 < 1) + // console.log("porkchop\\ sandwiches\nporkman") + //else console.log("encephylophagy") + + // get the partial line and partial word, + // if the point isn't at the end. + // ie, tabbing at: npm foo b|ar + var w = +process.env.COMP_CWORD + , words = args.map(unescape) + , word = words[w] + , line = process.env.COMP_LINE + , point = +process.env.COMP_POINT + , lineLength = line.length + , partialLine = line.substr(0, point) + , partialWords = words.slice(0, w) + + // figure out where in that last word the point is. + var partialWord = args[w] + , i = partialWord.length + while (partialWord.substr(0, i) !== partialLine.substr(-1*i) && i > 0) { + i -- + } + partialWord = unescape(partialWord.substr(0, i)) + partialWords.push(partialWord) + + var opts = { words : words + , w : w + , word : word + , line : line + , lineLength : line.length + , point : point + , partialLine : partialLine + , partialWords : partialWords + , partialWord : partialWord + , raw: args + } + + cb = wrapCb(cb, opts) + + console.error(opts) + + if (partialWords.slice(0, -1).indexOf("--") === -1) { + if (word.charAt(0) === "-") return configCompl(opts, cb) + if (words[w - 1] + && words[w - 1].charAt(0) === "-" + && !isFlag(words[w - 1])) { + // awaiting a value for a non-bool config. + // don't even try to do this for now + console.error("configValueCompl") + return configValueCompl(opts, cb) + } + } + + // try to find the npm command. + // it's the first thing after all the configs. + // take a little shortcut and use npm's arg parsing logic. + // don't have to worry about the last arg being implicitly + // boolean'ed, since the last block will catch that. + var parsed = opts.conf = + nopt(configTypes, shorthands, partialWords.slice(0, -1), 0) + // check if there's a command already. + console.error(parsed) + var cmd = parsed.argv.remain[1] + if (!cmd) return cmdCompl(opts, cb) + + Object.keys(parsed).forEach(function (k) { + npm.config.set(k, parsed[k]) + }) + + // at this point, if words[1] is some kind of npm command, + // then complete on it. + // otherwise, do nothing + cmd = npm.commands[cmd] + if (cmd && cmd.completion) return cmd.completion(opts, cb) + + // nothing to do. + cb() +} + +function dumpScript (cb) { + var fs = require("graceful-fs") + , path = require("path") + , p = path.resolve(__dirname, "utils/completion.sh") + + fs.readFile(p, "utf8", function (er, d) { + if (er) return cb(er) + d = d.replace(/^\#\!.*?\n/, "") + + process.stdout.write(d, function (n) { cb() }) + process.stdout.on("error", function (er) { + // Darwin is a real dick sometimes. + // + // This is necessary because the "source" or "." program in + // bash on OS X closes its file argument before reading + // from it, meaning that you get exactly 1 write, which will + // work most of the time, and will always raise an EPIPE. + // + // Really, one should not be tossing away EPIPE errors, or any + // errors, so casually. But, without this, `. <(npm completion)` + // can never ever work on OS X. + if (er.errno === require("constants").EPIPE) er = null + cb(er) + }) + + }) +} + +function unescape (w) { + if (w.charAt(0) === "\"") return w.replace(/^"|"$/g, "") + else return w.replace(/\\ /g, " ") +} + +function escape (w) { + if (!w.match(/\s+/)) return w + return "\"" + w + "\"" +} + +// The command should respond with an array. Loop over that, +// wrapping quotes around any that have spaces, and writing +// them to stdout. Use console.log, not the outfd config. +// If any of the items are arrays, then join them with a space. +// Ie, returning ["a", "b c", ["d", "e"]] would allow it to expand +// to: "a", "b c", or "d" "e" +function wrapCb (cb, opts) { return function (er, compls) { + if (!Array.isArray(compls)) compls = compls ? [compls] : [] + compls = compls.map(function (c) { + if (Array.isArray(c)) c = c.map(escape).join(" ") + else c = escape(c) + return c + }) + if (opts.partialWord) compls = compls.filter(function (c) { + return c.indexOf(opts.partialWord) === 0 + }) + console.error([er && er.stack, compls, opts.partialWord]) + if (er || compls.length === 0) return cb(er) + output.write(compls.join("\n"), 1, false, cb) +}} + +// the current word has a dash. Return the config names, +// with the same number of dashes as the current word has. +function configCompl (opts, cb) { + var word = opts.word + , split = word.match(/^(-+)((?:no-)*)(.*)$/) + , dashes = split[1] + , no = split[2] + , conf = split[3] + , confs = allConfs + , flags = configNames.filter(isFlag) + console.error(flags) + + return cb(null, allConfs.map(function (c) { + return dashes + c + }).concat(flags.map(function (f) { + return dashes + (no || "no-") + f + }))) +} + +// expand with the valid values of various config values. +// not yet implemented. +function configValueCompl (opts, cb) { + console.error('configValue', opts) + return cb(null, []) +} + +// check if the thing is a flag or not. +function isFlag (word) { + // shorthands never take args. + var split = word.match(/^(-*)((?:no-)+)?(.*)$/) + , dashes = split[1] + , no = split[2] + , conf = split[3] + return no || configTypes[conf] === Boolean || shorthands[conf] +} + +// complete against the npm commands +function cmdCompl (opts, cb) { + return cb(null, npm.fullList) +} diff --git a/deps/npm/lib/config.js b/deps/npm/lib/config.js new file mode 100644 index 0000000000..6781679f1d --- /dev/null +++ b/deps/npm/lib/config.js @@ -0,0 +1,286 @@ + +module.exports = config + +config.usage = "npm config set " + + "\nnpm config get " + + "\nnpm config delete " + + "\nnpm config list" + + "\nnpm config edit" + +var ini = require("./utils/ini.js") + , log = require("./utils/log.js") + , npm = require("./npm.js") + , exec = require("./utils/exec.js") + , fs = require("graceful-fs") + , dc + , output = require("./utils/output.js") + , types = require("./utils/config-defs.js").types + +config.completion = function (opts, cb) { + var argv = opts.conf.argv.remain + if (argv[1] !== "config") argv.unshift("config") + if (argv.length === 2) { + var cmds = ["get", "set", "delete", "ls", "rm", "edit"] + if (opts.partialWord !== "l") cmds.push("list") + return cb(null, cmds) + } + + var action = argv[2] + switch (action) { + case "set": + // todo: complete with valid values, if possible. + if (argv.length > 3) return cb(null, []) + // fallthrough + case "get": + case "delete": + case "rm": + return cb(null, Object.keys(types)) + case "edit": + case "list": case "ls": + return cb(null, []) + default: return cb(null, []) + } +} + +// npm config set key value +// npm config get key +// npm config list +function config (args, cb) { + var action = args.shift() + switch (action) { + case "set": return set(args[0], args[1], cb) + case "get": return get(args[0], cb) + case "delete": case "rm": case "del": return del(args[0], cb) + case "list": case "ls": return list(cb) + case "edit": return edit(cb) + default: return unknown(action, cb) + } +} + +function edit (cb) { + var e = ini.get("editor") + , which = ini.get("global") ? "global" : "user" + , f = ini.get(which + "config") + , eol = process.platform === "win32" ? "\r\n" : "\n" + if (!e) return cb(new Error("No EDITOR config or environ set.")) + ini.save(which, function (er) { + if (er) return cb(er) + fs.readFile(f, "utf8", function (er, data) { + if (er) data = "" + dc = dc || require("./utils/config-defs.js").defaults + data = [ ";;;;" + , "; npm "+(ini.get("global") ? + "globalconfig" : "userconfig")+" file" + , "; this is a simple ini-formatted file" + , "; lines that start with semi-colons are comments." + , "; read `npm help config` for help on the various options" + , ";;;;" + , "" + , data + ].concat( [ ";;;;" + , "; all options with default values" + , ";;;;" + ] + ) + .concat(Object.keys(dc).map(function (k) { + return "; " + k + " = " + ini.unParseField(dc[k],k) + })) + .concat([""]) + .join(eol) + fs.writeFile + ( f + , data + , "utf8" + , function (er) { + if (er) return cb(er) + exec(e, [f], function (er) { + if (er) return cb(er) + ini.resolveConfigs(function (er) { + if (er) return cb(er) + ini.save(which, cb) + }) + }) + } + ) + }) + }) +} + +function del (key, cb) { + if (!key) return cb(new Error("no key provided")) + ini.del(key) + ini.save(ini.get("global") ? "global" : "user", cb) +} + +function set (key, val, cb) { + if (val === undefined) { + if (key.indexOf("=") !== -1) { + var k = key.split("=") + key = k.shift() + val = k.join("=") + } else { + val = "" + } + } + key = key.trim() + val = val.trim() + log("set "+key+" "+val, "config") + var where = ini.get("global") ? "global" : "user" + ini.set(key, val, where) + ini.save(where, cb) +} + +function get (key, cb) { + if (!key) return list(cb) + if (key.charAt(0) === "_") { + return cb(new Error("---sekretz---")) + } + output.write(npm.config.get(key), cb) +} + +function sort (a, b) { + return a > b ? 1 : -1 +} + +function reverse (a, b) { + return a > b ? -1 : 1 +} + +function list (cb) { + var msg = "" + , long = npm.config.get("long") + + // cli configs. + // show any that aren't secret + var cli = ini.configList.list[ini.TRANS.cli] + , eol = process.platform === "win32" ? "\r\n" : "\n" + , cliKeys = Object.keys(cli).filter(function (k) { + return !(k.charAt(0) === "_" || types[k] !== types[k]) + }).sort(function (a, b) { + return a > b ? 1 : -1 + }) + if (cliKeys.length) { + msg += "; cli configs" + eol + cliKeys.forEach(function (k) { + if (k === "argv") return + msg += k + " = " + JSON.stringify(cli[k]) + eol + }) + msg += eol + } + + // env configs + var env = ini.configList.list[ini.TRANS.env] + , envKeys = Object.keys(env).filter(function (k) { + return !(k.charAt(0) === "_" || types[k] !== types[k]) + }).sort(function (a, b) { + return a > b ? 1 : -1 + }) + if (envKeys.length) { + msg += "; environment configs" + eol + envKeys.forEach(function (k) { + if (env[k] !== ini.get(k)) { + if (!long) return + msg += "; " + k + " = " + JSON.stringify(env[k]) + + " (overridden)" + eol + } else msg += k + " = " + JSON.stringify(env[k]) + eol + }) + msg += eol + } + + // user config file + var uconf = ini.configList.list[ini.TRANS.user] + , uconfKeys = Object.keys(uconf).filter(function (k) { + return types[k] === types[k] + }).sort(function (a, b) { + return a > b ? 1 : -1 + }) + if (uconfKeys.length) { + msg += "; userconfig " + ini.get("userconfig") + eol + uconfKeys.forEach(function (k) { + var val = (k.charAt(0) === "_") + ? "---sekretz---" + : JSON.stringify(uconf[k]) + if (uconf[k] !== ini.get(k)) { + if (!long) return + msg += "; " + k + " = " + val + + " (overridden)" + eol + } else msg += k + " = " + val + eol + }) + msg += eol + } + + // global config file + var gconf = ini.configList.list[ini.TRANS.global] + , gconfKeys = Object.keys(gconf).filter(function (k) { + return types[k] === types[k] + }).sort(function (a, b) { + return a > b ? 1 : -1 + }) + if (gconfKeys.length) { + msg += "; globalconfig " + ini.get("globalconfig") + eol + gconfKeys.forEach(function (k) { + var val = (k.charAt(0) === "_") + ? "---sekretz---" + : JSON.stringify(gconf[k]) + if (gconf[k] !== ini.get(k)) { + if (!long) return + msg += "; " + k + " = " + val + + " (overridden)" + eol + } else msg += k + " = " + val + eol + }) + msg += eol + } + + // builtin config file + var bconf = ini.configList.list[ini.TRANS.builtin] + , bconfKeys = Object.keys(bconf).filter(function (k) { + return types[k] === types[k] + }).sort(function (a, b) { + return a > b ? 1 : -1 + }) + if (bconfKeys.length) { + var path = require("path") + msg += "; builtin config " + path.resolve(__dirname, "../npmrc") + eol + bconfKeys.forEach(function (k) { + var val = (k.charAt(0) === "_") + ? "---sekretz---" + : JSON.stringify(bconf[k]) + if (bconf[k] !== ini.get(k)) { + if (!long) return + msg += "; " + k + " = " + val + + " (overridden)" + eol + } else msg += k + " = " + val + eol + }) + msg += eol + } + + // only show defaults if --long + if (!long) { + msg += "; node install prefix = " + process.installPrefix + eol + + "; node bin location = " + process.execPath + eol + + "; cwd = " + process.cwd() + eol + + "; HOME = " + process.env.HOME + eol + + "; 'npm config ls -l' to show all defaults." + eol + + return output.write(msg, cb) + } + + var defaults = ini.defaultConfig + , defKeys = Object.keys(defaults) + msg += "; default values" + eol + defKeys.forEach(function (k) { + var val = JSON.stringify(defaults[k]) + if (defaults[k] !== ini.get(k)) { + if (!long) return + msg += "; " + k + " = " + val + + " (overridden)" + eol + } else msg += k + " = " + val + eol + }) + msg += eol + + return output.write(msg, cb) +} + +function unknown (action, cb) { + cb("Usage:\n" + config.usage) +} diff --git a/deps/npm/lib/deprecate.js b/deps/npm/lib/deprecate.js new file mode 100644 index 0000000000..6f18879e4d --- /dev/null +++ b/deps/npm/lib/deprecate.js @@ -0,0 +1,48 @@ + +module.exports = deprecate + +deprecate.usage = "npm deprecate [@] " + +deprecate.completion = function (opts, cb) { + // first, get a list of remote packages this user owns. + // once we have a user account, then don't complete anything. + var un = npm.config.get("username") + if (!npm.config.get("username")) return cb() + if (opts.conf.argv.remain.length > 2) return cb() + // get the list of packages by user + var uri = "/-/by-user/"+encodeURIComponent(un) + registry.get(uri, null, 60000, function (er, list) { + if (er) return cb() + console.error(list) + return cb(null, list[un]) + }) +} + +var registry = require("./utils/npm-registry-client/index.js") + , semver = require("semver") + , log = require("./utils/log.js") + , npm = require("./npm.js") + +function deprecate (args, cb) { + var pkg = args[0] + , msg = args[1] + if (msg === undefined) return cb(new Error(deprecate.usage)) + // fetch the data and make sure it exists. + pkg = pkg.split(/@/) + var name = pkg.shift() + , ver = pkg.join("@") + if (semver.validRange(ver) === null) { + return cb(new Error("invalid version range: "+ver)) + } + registry.get(name, function (er, data) { + if (er) return cb(er) + // filter all the versions that match + Object.keys(data.versions).filter(function (v) { + return semver.satisfies(v, ver) + }).forEach(function (v) { + data.versions[v].deprecated = msg + }) + // now update the doc on the registry + registry.request.PUT(data._id, data, cb) + }) +} diff --git a/deps/npm/lib/docs.js b/deps/npm/lib/docs.js new file mode 100644 index 0000000000..de9f71c654 --- /dev/null +++ b/deps/npm/lib/docs.js @@ -0,0 +1,46 @@ + +module.exports = docs + +docs.usage = "npm docs " + +docs.completion = function (opts, cb) { + if (opts.conf.argv.remain.length > 2) return cb() + registry.get("/-/short", null, 60000, function (er, list) { + return cb(null, list || []) + }) +} + +var exec = require("./utils/exec.js") + , registry = require("./utils/npm-registry-client/index.js") + , npm = require("./npm.js") + , log = require("./utils/log.js") + +function docs (args, cb) { + if (!args.length) return cb(docs.usage) + var n = args[0].split("@").shift() + registry.get(n, "latest", 3600, function (er, d) { + if (er) return cb(er) + var homepage = d.homepage + , repo = d.repository || d.repositories + if (homepage) return open(homepage, cb) + if (repo) { + if (Array.isArray(repo)) repo = repo.shift() + if (repo.url) repo = repo.url + log.verbose(repo, "repository") + if (repo) { + return open(repo.replace(/^git(@|:\/\/)/, 'http://') + .replace(/\.git$/, '')+"#readme", cb) + } + } + return open("http://search.npmjs.org/#/" + d.name, cb) + }) +} + +function open (url, cb) { + exec(npm.config.get("browser"), [url], log.er(cb, + "Failed to open "+url+" in a browser. It could be that the\n"+ + "'browser' config is not set. Try doing this:\n"+ + " npm config set browser google-chrome\n"+ + "or:\n"+ + " npm config set browser lynx\n")) +} diff --git a/deps/npm/lib/edit.js b/deps/npm/lib/edit.js new file mode 100644 index 0000000000..df103d4df4 --- /dev/null +++ b/deps/npm/lib/edit.js @@ -0,0 +1,31 @@ +// npm edit [@] +// open the package folder in the $EDITOR + +module.exports = edit +edit.usage = "npm edit " + +edit.completion = require("./utils/completion/installed-shallow.js") + +var npm = require("./npm.js") + , exec = require("./utils/exec.js") + , path = require("path") + , fs = require("graceful-fs") + , log = require("./utils/log.js") + +function edit (args, cb) { + var p = args[0] + if (args.length !== 1 || !p) return cb(edit.usage) + var editor = npm.config.get("editor") + if (!editor) return cb(new Error( + "No editor set. Set the 'editor' config, or $EDITOR environ.")) + p = p.split("/") + .join("/node_modules/") + .replace(/(\/node_modules)+/, "/node_modules") + fs.lstat(path.resolve(npm.dir, p), function (er) { + if (er) return cb(er) + exec(editor, [path.resolve(npm.dir, p)], function (er) { + if (er) return cb(er) + npm.commands.rebuild(args, cb) + }) + }) +} diff --git a/deps/npm/lib/explore.js b/deps/npm/lib/explore.js new file mode 100644 index 0000000000..8392c91728 --- /dev/null +++ b/deps/npm/lib/explore.js @@ -0,0 +1,34 @@ +// npm expore [@] +// open a subshell to the package folder. + +module.exports = explore +explore.usage = "npm explore [ -- ]" +explore.completion = require("./utils/completion/installed-shallow.js") + +var npm = require("./npm.js") + , exec = require("./utils/exec.js") + , path = require("path") + , fs = require("graceful-fs") + +function explore (args, cb) { + if (args.length < 1 || !args[0]) return cb(explore.usage) + var p = args.shift() + args = args.join(" ").trim() + if (args) args = ["-c", args] + else args = [] + + var editor = npm.config.get("editor") + , cwd = path.resolve(npm.dir, p) + fs.stat(cwd, function (er, s) { + if (er || !s.isDirectory()) return cb(new Error( + "It doesn't look like "+p+" is installed.")) + if (!args.length) console.log( + "\nExploring "+cwd+"\n"+ + "Type 'exit' or ^D when finished\n") + exec(npm.config.get("shell"), args, null, true, cwd, function (er) { + // only fail if non-interactive. + if (!args.length) return cb() + cb(er) + }) + }) +} diff --git a/deps/npm/lib/faq.js b/deps/npm/lib/faq.js new file mode 100644 index 0000000000..912db0072f --- /dev/null +++ b/deps/npm/lib/faq.js @@ -0,0 +1,8 @@ + +module.exports = faq + +faq.usage = "npm faq" + +var npm = require("./npm.js") + +function faq (args, cb) { npm.commands.help(["faq"], cb) } diff --git a/deps/npm/lib/get.js b/deps/npm/lib/get.js new file mode 100644 index 0000000000..aa058002ec --- /dev/null +++ b/deps/npm/lib/get.js @@ -0,0 +1,12 @@ + +module.exports = get + +get.usage = "npm get (See `npm config`)" + +var npm = require("./npm.js") + +get.completion = npm.commands.config.completion + +function get (args, cb) { + npm.commands.config(["get"].concat(args), cb) +} diff --git a/deps/npm/lib/help-search.js b/deps/npm/lib/help-search.js new file mode 100644 index 0000000000..a4ef667c08 --- /dev/null +++ b/deps/npm/lib/help-search.js @@ -0,0 +1,183 @@ + +module.exports = helpSearch + +var fs = require("graceful-fs") + , output = require("./utils/output.js") + , path = require("path") + , asyncMap = require("slide").asyncMap + , cliDocsPath = path.join(__dirname, "..", "doc", "cli") + , apiDocsPath = path.join(__dirname, "..", "doc", "api") + , log = require("./utils/log.js") + , npm = require("./npm.js") + +helpSearch.usage = "npm help-search " + +function helpSearch (args, silent, cb) { + if (typeof cb !== "function") cb = silent, silent = false + if (!args.length) return cb(helpSearch.usage) + + // see if we're actually searching the api docs. + var argv = npm.config.get("argv").cooked + , docsPath = cliDocsPath + , cmd = "help" + if (argv.length && argv[0].indexOf("api") !== -1) { + docsPath = apiDocsPath + cmd = "apihelp" + } + + fs.readdir(docsPath, function(er, files) { + if (er) return log.er(cb, "Could not load documentation")(er) + + var search = args.join(" ") + , results = [] + asyncMap(files, function (file, cb) { + fs.lstat(path.resolve(docsPath, file), function (er, st) { + if (er) return cb(er) + if (!st.isFile()) return cb(null, []) + + fs.readFile(path.resolve(docsPath, file), "utf8", function (er, data) { + if (er) return cb(er) + + var match = false + for (var a = 0, l = args.length; a < l && !match; a ++) { + match = data.toLowerCase().indexOf(args[a].toLowerCase()) !== -1 + } + if (!match) return cb(null, []) + + var lines = data.split(/\n+/) + , context = [] + + // if a line has a search term, then skip it and the next line. + // if the next line has a search term, then skip all 3 + // otherwise, set the line to null. + for (var i = 0, l = lines.length; i < l; i ++) { + var line = lines[i] + , nextLine = lines[i + 1] + , match = false + if (nextLine) { + for (var a = 0, ll = args.length; a < ll && !match; a ++) { + match = nextLine.toLowerCase() + .indexOf(args[a].toLowerCase()) !== -1 + } + if (match) { + // skip over the next line, and the line after it. + i += 2 + continue + } + } + + match = false + for (var a = 0, ll = args.length; a < ll && !match; a ++) { + match = line.toLowerCase().indexOf(args[a].toLowerCase()) !== -1 + } + if (match) { + // skip over the next line + i ++ + continue + } + + lines[i] = null + } + + // now squish any string of nulls into a single null + lines = lines.reduce(function (l, r) { + if (!(r === null && l[l.length-1] === null)) l.push(r) + return l + }, []) + + if (lines[lines.length - 1] === null) lines.pop() + if (lines[0] === null) lines.shift() + + // now see how many args were found at all. + var found = {} + , totalHits = 0 + lines.forEach(function (line) { + args.forEach(function (arg) { + var hit = (line || "").toLowerCase() + .split(arg.toLowerCase()).length - 1 + if (hit > 0) { + found[arg] = (found[arg] || 0) + hit + totalHits += hit + } + }) + }) + + return cb(null, { file: file, lines: lines, found: Object.keys(found) + , hits: found, totalHits: totalHits }) + }) + }) + }, function (er, results) { + if (er) return cb(er) + + // if only one result, then just show that help section. + if (results.length === 1) { + return npm.commands.help([results[0].file.replace(/\.md$/, "")], cb) + } + + if (results.length === 0) { + return output.write("No results for " + + args.map(JSON.stringify).join(" "), cb) + } + + // sort results by number of results found, then by number of hits + // then by number of matching lines + results = results.sort(function (a, b) { + return a.found.length > b.found.length ? -1 + : a.found.length < b.found.length ? 1 + : a.totalHits > b.totalHits ? -1 + : a.totalHits < b.totalHits ? 1 + : a.lines.length > b.lines.length ? -1 + : a.lines.length < b.lines.length ? 1 + : 0 + }) + + var out = results.map(function (res, i, results) { + var out = "npm " + cmd + " "+res.file.replace(/\.md$/, "") + , r = Object.keys(res.hits).map(function (k) { + return k + ":" + res.hits[k] + }).sort(function (a, b) { + return a > b ? 1 : -1 + }).join(" ") + + out += ((new Array(Math.max(1, 81 - out.length - r.length))) + .join (" ")) + r + + if (!npm.config.get("long")) return out + + var out = "\n\n" + out + + "\n" + (new Array(81)).join("—") + "\n" + + res.lines.map(function (line, i) { + if (line === null || i > 3) return "" + for (var out = line, a = 0, l = args.length; a < l; a ++) { + var finder = out.toLowerCase().split(args[a].toLowerCase()) + , newOut = [] + , p = 0 + finder.forEach(function (f) { + newOut.push( out.substr(p, f.length) + , "\1" + , out.substr(p + f.length, args[a].length) + , "\2" ) + p += f.length + args[a].length + }) + out = newOut.join("") + } + out = out.split("\1").join("\033[31;40m") + .split("\2").join("\033[0m") + return out + }).join("\n").trim() + return out + }).join("\n") + + if (results.length && !npm.config.get("long")) { + out = "Top hits for "+(args.map(JSON.stringify).join(" ")) + + "\n" + (new Array(81)).join("—") + "\n" + + out + + "\n" + (new Array(81)).join("—") + "\n" + + "(run with -l or --long to see more context)" + } + + output.write(out.trim(), function (er) { cb(er, results) }) + }) + + }) +} diff --git a/deps/npm/lib/help.js b/deps/npm/lib/help.js new file mode 100644 index 0000000000..854a7d6bab --- /dev/null +++ b/deps/npm/lib/help.js @@ -0,0 +1,150 @@ + +module.exports = help + +help.completion = function (opts, cb) { + if (opts.conf.argv.remain.length > 2) return cb(null, []) + var num = 1 + if (-1 !== opts.conf.argv.remain[1].indexOf("api")) num = 3 + getSections(num, cb) +} + +var fs = require("graceful-fs") + , path = require("path") + , exec = require("./utils/exec.js") + , npm = require("./npm.js") + , output = require("./utils/output.js") + , log = require("./utils/log.js") + +function help (args, cb) { + var num = 1 + , argv = npm.config.get("argv").cooked + if (argv.length && -1 !== argv[0].indexOf("api")) { + num = 3 + } + + if (args.length > 1 && args[0]) { + return npm.commands["help-search"](args, num, cb) + } + + var section = args[0] + + if (section) { + if ( npm.config.get("usage") + && npm.commands[section] + && npm.commands[section].usage + ) { + npm.config.set("loglevel", "silent") + return output.write(npm.commands[section].usage, cb) + } + var sectionPath = path.join( __dirname, "..", "man", "man" + num + , section + "." + num) + , htmlPath = path.resolve( __dirname, "..", "html" + , num === 3 ? "api" : "doc" + , section+".html" ) + return fs.stat + ( sectionPath + , function (e, o) { + if (e) return npm.commands["help-search"](args, cb) + + var manpath = path.join(__dirname, "..", "man") + , env = {} + Object.keys(process.env).forEach(function (i) { + env[i] = process.env[i] + }) + env.MANPATH = manpath + var viewer = npm.config.get("viewer") + switch (viewer) { + case "woman": + var a = ["-e", "(woman-find-file \"" + sectionPath + "\")"] + exec("emacsclient", a, env, true, cb) + break + case "browser": + var b = npm.config.get("browser") + if (!b) { + return cb(new Error("viewer=browser and no browser set.")) + } + output.write("Opening HTML in default browser...", cb) + // windows is SO weird. + if (process.platform === "win32") { + exec("cmd", ["/c", htmlPath], env, false, function () {}) + } else { + exec(b, [htmlPath], env, false, function () {}) + } + break + default: + exec("man", [num, section], env, true, cb) + } + } + ) + } else getSections(function (er, sections) { + if (er) return cb(er) + npm.config.set("loglevel", "silent") + output.write + ( ["\nUsage: npm " + , "" + , "where is one of:" + , npm.config.get("long") ? usages() + : " " + wrap(Object.keys(npm.commands)) + , "" + , "npm -h quick help on " + , "npm -l display full usage info" + , "npm faq commonly asked questions" + , "npm help search for help on " + , "npm help npm involved overview" + , "" + , "Specify configs in the ini-formatted file:" + , " " + npm.config.get("userconfig") + , "or on the command line via: npm --key value" + , "Config info can be viewed via: npm help config" + ].join("\n"), function () { cb(er) }) + }) +} + +function usages () { + // return a string of : + var maxLen = 0 + return Object.keys(npm.commands).filter(function (c) { + return c === npm.deref(c) + }).reduce(function (set, c) { + set.push([c, npm.commands[c].usage || ""]) + maxLen = Math.max(maxLen, c.length) + return set + }, []).map(function (item) { + var c = item[0] + , usage = item[1] + return "\n " + c + (new Array(maxLen - c.length + 2).join(" ")) + + (usage.split("\n") + .join("\n" + (new Array(maxLen + 6).join(" ")))) + }).join("\n") + return out +} + + +function wrap (arr) { + var out = [''] + , l = 0 + arr.sort(function (a,b) { return a --save` afterwards to install a package and" + ,"save it as a dependency in the package.json file." + ,"" + ,"Press ^C at any time to quit." + ,"" + ].join("\n")) + promiseChain(cb) + ( prompt + , ["Package name: ", defaultName(folder, data)] + , function (n) { data.name = n } + ) + ( prompt + , ["Description: ", data.description] + , function (d) { data.description = d } + ) + ( defaultVersion, [folder, data], function (v) { data.version = v } ) + (function (cb) { + prompt("Package version: ", data.version, function (er, v) { + if (er) return cb(er) + data.version = v + cb() + }) + }, []) + ( prompt + , ["Project homepage: ", data.homepage || data.url || "none"] + , function (u) { + if (u === "none") return + data.homepage = u + delete data.url + } + ) + ( defaultRepo, [folder, data], function (r) { data.repository = r } ) + (function (cb) { + prompt( "Project git repository: " + , data.repository && data.repository.url || "none" + , function (er, r) { + if (er) return cb(er) + if (r !== "none") { + data.repository = (data.repository || {}).url = r + } + cb() + } + ) + }, []) + ( prompt + , ["Author name: ", data.author && data.author.name] + , function (n) { + if (!n) return + (data.author = data.author || {}).name = n + } + ) + ( prompt + , ["Author email: ", data.author && data.author.email || "none"] + , function (n) { + if (n === "none") return + (data.author = data.author || {}).email = n + } + ) + ( prompt + , ["Author url: ", data.author && data.author.url || "none"] + , function (n) { + if (n === "none") return + (data.author = data.author || {}).url = n + } + ) + ( prompt + , ["Main module/entry point: ", data.main || "none"] + , function (m) { + if (m === "none") { + delete data.main + return + } + data.main = m + } + ) + ( prompt + , ["Test command: ", data.scripts && data.scripts.test || "none"] + , function (t) { + if (t === "none") return + (data.scripts = data.scripts || {}).test = t + } + ) + ( prompt + , [ "What versions of node does it run on? " + , data.engines && data.engines.node + || (eng) + ] + , function (nodever) { + (data.engines = data.engines || {}).node = nodever + } + ) + (cleanupPaths, [data, folder]) + (function (cb) { + try { data = readJson.processJson(data) } + catch (er) { return cb(er) } + Object.keys(data) + .filter(function (k) { return k.match(/^_/) }) + .forEach(function (k) { delete data[k] }) + readJson.unParsePeople(data) + var str = JSON.stringify(data, null, 2) + , msg = "About to write to " + + path.join(folder, "package.json") + + "\n\n" + + str + + "\n\n" + output.write(msg, cb) + }) + (function (cb) { + prompt("\nIs this ok? ", "yes", function (er, ok) { + if (er) return cb(er) + if (ok.toLowerCase().charAt(0) !== "y") { + return cb(new Error("cancelled")) + } + return cb() + }) + }) + (function (cb) { + fs.writeFile( path.join(folder, "package.json") + , JSON.stringify(data, null, 2) + "\n" + , cb ) + }) + () +} + +// sync - no io +function defaultName (folder, data) { + if (data.name) return data.name + return path.basename(folder) + .replace(/^node[-\._]?|([-\._]node)[-\._]?(js)?$/g, "") +} + +function defaultVersion (folder, data, cb) { + if (data.version) return cb(null, data.version) + exec("git", ["describe", "--tags"], process.env, false, folder, + function (er, code, out) { + out = (out || "").trim() + if (semver.valid(out)) return cb(null, out) + out = npm.config.get("init.version") + if (semver.valid(out)) return cb(null, out) + return cb(null, "0.0.0") + }) +} + +function defaultRepo (folder, data, cb) { + if (data.repository) return cb(null, data.repository) + exec( "git", ["remote", "-v"], process.env, false, folder + , function (er, code, out) { + out = (out || "") + .trim() + .split("\n").filter(function (line) { + return line.search(/^origin/) !== -1 + })[0] + if (!out) return cb(null, {}) + var repo = + { type: "git" + , url: out.split(/\s/)[1] + .replace("git@github.com:", "git://github.com/") + } + return cb(null, repo) + }) +} + +function cleanupPaths (data, folder, cb) { + if (data.main) { + data.main = cleanupPath(data.main, folder) + } + var dirs = data.directories + if (dirs) { + Object.keys(dirs).forEach(function (dir) { + dirs[dir] = cleanupPath(dirs[dir], folder) + }) + } + cb() +} + +function cleanupPath (m, folder) { + if (m.indexOf(folder) === 0) m = path.join(".", m.substr(folder.length)) + return m +} diff --git a/deps/npm/lib/install.js b/deps/npm/lib/install.js new file mode 100644 index 0000000000..4d71b9a512 --- /dev/null +++ b/deps/npm/lib/install.js @@ -0,0 +1,608 @@ + +// npm install +// +// See doc/install.md for more description + +// Managing "family" lists... +// every time we dive into a deeper node_modules folder, the "family" +// list that gets passed along uses the previous "family" list as +// it's __proto__. Any "resolved precise dependency" things that aren't +// already on this object get added, and then that's passed to the next +// generation of installation. + +module.exports = install + +install.usage = "npm install " + + "\nnpm install " + + "\nnpm install " + + "\nnpm install " + + "\nnpm install @" + + "\nnpm install @" + + "\nnpm install @" + + "\n\nCan specify one or more: npm install ./foo.tgz bar@stable /some/folder" + + "\nInstalls dependencies in ./package.json if no argument supplied" + +install.completion = function (opts, cb) { + // install can complete to a folder with a package.json, or any package. + // if it has a slash, then it's gotta be a folder + // if it starts with https?://, then just give up, because it's a url + // for now, not yet implemented. + var registry = require("./utils/npm-registry-client/index.js") + registry.get("/-/short", function (er, pkgs) { + if (er) return cb() + if (!opts.partialWord) return cb(null, pkgs) + + var name = opts.partialWord.split("@").shift() + pkgs = pkgs.filter(function (p) { + return p.indexOf(name) === 0 + }) + + if (pkgs.length !== 1 && opts.partialWord === name) { + return cb(null, pkgs) + } + + registry.get(pkgs[0], function (er, d) { + if (er) return cb() + return cb(null, Object.keys(d["dist-tags"] || {}) + .concat(Object.keys(d.versions || {})) + .map(function (t) { + return pkgs[0] + "@" + t + })) + }) + }) +} + +var npm = require("./npm.js") + , semver = require("semver") + , readJson = require("./utils/read-json.js") + , log = require("./utils/log.js") + , path = require("path") + , fs = require("graceful-fs") + , cache = require("./cache.js") + , asyncMap = require("slide").asyncMap + , chain = require("slide").chain + , relativize = require("./utils/relativize.js") + , output + , url = require("url") + , mkdir = require("./utils/mkdir-p.js") + , lifecycle = require("./utils/lifecycle.js") + +function install (args, cb_) { + + function cb (er, installed) { + if (er) return cb_(er) + + output = output || require("./utils/output.js") + + var tree = treeify(installed) + , pretty = prettify(tree, installed) + + output.write(pretty, function (er) { + if (er) return cb_(er) + save(where, installed, tree, pretty, cb_) + }) + } + + // the /path/to/node_modules/.. + var where = path.resolve(npm.dir, "..") + + // internal api: install(where, what, cb) + if (arguments.length === 3) { + where = args + args = [].concat(cb_) // pass in [] to do default dep-install + cb_ = arguments[2] + log.verbose([where, args], "install(where, what)") + } + + if (!npm.config.get("global")) { + args = args.filter(function (a) { + return path.resolve(a) !== where + }) + } + + mkdir(where, function (er) { + if (er) return cb(er) + // install dependencies locally by default, + // or install current folder globally + if (!args.length) { + if (npm.config.get("global")) args = ["."] + else return readJson( path.resolve(where, "package.json") + , { dev: !npm.config.get("production") } + , function (er, data) { + if (er) return log.er(cb, "Couldn't read dependencies.")(er) + var deps = Object.keys(data.dependencies || {}) + log.verbose([where, deps], "where, deps") + var family = {} + , ancestors = {} + family[data.name] = ancestors[data.name] = data.version + installManyTop(deps.map(function (dep) { + var target = data.dependencies[dep] + , parsed = url.parse(target.replace(/^git\+/, "git")) + if (!parsed.protocol) { + target = dep + "@" + target + } + return target + }), where, family, ancestors, false, cb) + }) + } + + // initial "family" is the name:version of the root, if it's got + // a pacakge.json file. + readJson(path.resolve(where, "package.json"), function (er, data) { + if (er) data = null + var family = {} + , ancestors = {} + if (data) family[data.name] = ancestors[data.name] = data.version + var fn = npm.config.get("global") ? installMany : installManyTop + fn(args, where, family, ancestors, true, cb) + }) + }) +} + +// if the -S|--save option is specified, then write installed packages +// as dependencies to a package.json file. +// This is experimental. +function save (where, installed, tree, pretty, cb) { + if (!npm.config.get("save") || npm.config.get("global")) { + return cb(null, installed, tree, pretty) + } + // each item in the tree is a top-level thing that should be saved + // to the package.json file. + // The relevant tree shape is { : {what:} } + var saveTarget = path.resolve(where, "package.json") + , things = Object.keys(tree).map(function (k) { + //log.warn(k, "k") + return tree[k].what.split("@") + }).reduce(function (set, k) { + var rangeDescriptor = semver.gte(k[1], "0.1.0") ? "~" : "" + set[k[0]] = rangeDescriptor + k[1] + return set + }, {}) + + //log.warn(things, "things") + + // don't use readJson, because we don't want to do all the other + // tricky npm-specific stuff that's in there. + fs.readFile(saveTarget, function (er, data) { + // ignore errors here, just don't save it. + try { + data = JSON.parse(data.toString("utf8")) + } catch (ex) { + er = ex + } + if (er) return cb(null, installed, tree, pretty) + + var deps = npm.config.get("dev") ? "devDependencies" : "dependencies" + deps = data[deps] = data[deps] || {} + + Object.keys(things).forEach(function (t) { + deps[t] = things[t] + }) + data = JSON.stringify(data, null, 2) + "\n" + fs.writeFile(saveTarget, data, function (er) { + cb(er, installed, tree, pretty) + }) + }) +} + + +// Outputting *all* the installed modules is a bit confusing, +// because the length of the path does not make it clear +// that the submodules are not immediately require()able. +// TODO: Show the complete tree, ls-style. +function prettify (tree, installed) { + if (npm.config.get("parseable")) return parseable(installed) + return Object.keys(tree).map(function (p) { + p = tree[p] + var c = "" + if (p.children && p.children.length) { + pref = "\n" + var l = p.children.pop() + c = p.children.map(function (c) { + var gc = c.children && c.children.length + ? " (" + c.children.map(function (gc) { + return gc.what + }).join(" ") + ")" + : "" + return "\n├── " + c.what + gc + }).join("") + "\n└── " + l.what + } + return [p.what, p.where, c].join(" ") + + }).join("\n") +} + +function parseable (installed) { + var long = npm.config.get("long") + , cwd = process.cwd() + return installed.map(function (item) { + return path.resolve(cwd, item[1]) + + ( long ? ":" + item[0] : "" ) + }).join("\n") +} + +function treeify (installed) { + // each item is [what, where, parent, parentDir] + // If no parent, then report it. + // otherwise, tack it into the parent's children list. + // If the parent isn't a top-level then ignore it. + var whatWhere = installed.reduce(function (l, r) { + var parentDir = r[3] + , parent = r[2] + , where = r[1] + , what = r[0] + l[where] = { parentDir: parentDir + , parent: parent + , children: [] + , where: where + , what: what } + return l + }, {}) + + //log.warn(whatWhere, "whatWhere") + return Object.keys(whatWhere).reduce(function (l, r) { + var ww = whatWhere[r] + //log.warn([r, ww], "r, ww") + if (!ww.parent) { + l[r] = ww + } else { + var p = whatWhere[ww.parentDir] + if (p) p.children.push(ww) + else l[r] = ww + } + return l + }, {}) +} + + +// just like installMany, but also add the existing packages in +// where/node_modules to the family object. +function installManyTop (what, where, family, ancestors, explicit, cb_) { + + function cb (er, d) { + if (explicit || er) return cb_(er, d) + // since this wasn't an explicit install, let's build the top + // folder, so that `npm install` also runs the lifecycle scripts. + npm.commands.build([where], false, true, function (er) { + return cb_(er, d) + }) + } + + if (explicit) return next() + + readJson(path.join(where, "package.json"), function (er, data) { + if (er) return next(er) + lifecycle(data, "preinstall", where, next) + }) + + function next (er) { + if (er) return cb(er) + installManyTop_(what, where, family, ancestors, explicit, cb) + } +} + +function installManyTop_ (what, where, family, ancestors, explicit, cb) { + var nm = path.resolve(where, "node_modules") + , names = explicit + ? what.map(function (w) { return w.split(/@/).shift() }) + : [] + + fs.readdir(nm, function (er, pkgs) { + if (er) return installMany(what, where, family, ancestors, explicit, cb) + pkgs = pkgs.filter(function (p) { + return !p.match(/^[\._-]/) + && (!explicit || names.indexOf(p) === -1) + }) + asyncMap(pkgs.map(function (p) { + return path.resolve(nm, p, "package.json") + }), function (jsonfile, cb) { + readJson(jsonfile, function (er, data) { + if (er) return cb(null, []) + return cb(null, [[data.name, data.version]]) + }) + }, function (er, packages) { + // add all the existing packages to the family list. + // however, do not add to the ancestors list. + packages.forEach(function (p) { + family[p[0]] = p[1] + }) + return installMany(what, where, family, ancestors, explicit, cb) + }) + }) +} + +function installMany (what, where, family, ancestors, explicit, cb) { + // 'npm install foo' should install the version of foo + // that satisfies the dep in the current folder. + // This will typically return immediately, since we already read + // this file family, and it'll be cached. + readJson(path.resolve(where, "package.json"), function (er, data) { + if (er) data = {} + + d = data.dependencies || {} + var parent = data._id + + log.verbose(what, "into "+where) + // what is a list of things. + // resolve each one. + asyncMap( what + , targetResolver(where, family, ancestors, explicit, d) + , function (er, targets) { + if (er) return cb(er) + // each target will be a data object corresponding + // to a package, folder, or whatever that is in the cache now. + var newPrev = Object.create(family) + , newAnc = Object.create(ancestors) + + newAnc[data.name] = data.version + targets.forEach(function (t) { + newPrev[t.name] = t.version + }) + log.silly(targets, "resolved") + targets.filter(function (t) { return t }).forEach(function (t) { + log.info(t._id, "into "+where) + }) + asyncMap(targets, function (target, cb) { + log(target._id, "installOne") + installOne(target, where, newPrev, newAnc, parent, cb) + }, cb) + }) + }) +} + +function targetResolver (where, family, ancestors, explicit, deps) { + var alreadyInstalledManually = explicit ? [] : null + , nm = path.resolve(where, "node_modules") + + if (!explicit) fs.readdir(nm, function (er, inst) { + if (er) return alreadyInstalledManually = [] + asyncMap(inst, function (pkg, cb) { + readJson(path.resolve(nm, pkg, "package.json"), function (er, d) { + if (er) return cb(null, []) + if (semver.satisfies(d.version, deps[d.name] || "*")) { + return cb(null, d.name) + } + return cb(null, []) + }) + }, function (er, inst) { + // this is the list of things that are valid and should be ignored. + alreadyInstalledManually = inst + }) + }) + + var to = 0 + return function resolver (what, cb) { + if (!alreadyInstalledManually) return setTimeout(function () { + resolver(what, cb) + }, to++) + // now we know what's been installed here manually, + // or tampered with in some way that npm doesn't want to overwrite. + if (alreadyInstalledManually.indexOf(what.split("@").shift()) !== -1) { + log.verbose("skipping "+what, "already installed in "+where) + return cb(null, []) + } + + if (family[what] && semver.satisfies(family[what], deps[what] || "")) { + return cb(null, []) + } + + if (deps[what]) { + what = what + "@" + deps[what] + } + log.verbose(what, "cache add") + cache.add(what, function (er, data) { + if (!er && data && family[data.name] === data.version) { + return cb(null, []) + } + return cb(er, data) + }) + } +} + +// we've already decided to install this. if anything's in the way, +// then uninstall it first. +function installOne (target, where, family, ancestors, parent, cb) { + // the --link flag makes this a "link" command if it's at the + // the top level. + if (where === npm.prefix && npm.config.get("link") + && !npm.config.get("global")) { + return localLink(target, where, family, ancestors, parent, cb) + } + installOne_(target, where, family, ancestors, parent, cb) +} + +function localLink (target, where, family, ancestors, parent, cb) { + log.verbose(target._id, "try to link") + var jsonFile = path.resolve( npm.dir, target.name + , "package.json" ) + + readJson(jsonFile, function (er, data) { + if (er || data._id === target._id) { + if (er) { + install( path.resolve(npm.globalDir, "..") + , target._id + , function (er) { + if (er) return cb(er, []) + thenLink() + }) + } else thenLink() + + function thenLink () { + npm.commands.link([target.name], function (er, d) { + log.silly([er, d], "back from link") + cb(er, [resultList(target, where, parent)]) + }) + } + + } else { + log.verbose(target._id, "install locally (no link)") + installOne_(target, where, family, ancestors, parent, cb) + } + }) +} + +function resultList (target, where, parent) { + var nm = path.resolve(where, "node_modules") + , targetFolder = path.resolve(nm, target.name) + , prettyWhere = relativize(where, process.cwd() + "/x") + + if (prettyWhere === ".") prettyWhere = null + + if (!npm.config.get("global")) { + // print out the folder relative to where we are right now. + // relativize isn't really made for dirs, so you need this hack + targetFolder = relativize(targetFolder, process.cwd()+"/x") + } + + return [ target._id + , targetFolder + , prettyWhere && parent + , parent && prettyWhere ] +} + +function installOne_ (target, where, family, ancestors, parent, cb) { + var nm = path.resolve(where, "node_modules") + , targetFolder = path.resolve(nm, target.name) + , prettyWhere = relativize(where, process.cwd() + "/x") + + if (prettyWhere === ".") prettyWhere = null + + chain + ( [ [checkEngine, target] + , [checkCycle, target, ancestors] + , [checkGit, targetFolder] + , [write, target, targetFolder, family, ancestors] ] + , function (er, d) { + log.verbose(target._id, "installOne cb") + if (er) return cb(er) + + d.push(resultList(target, where, parent)) + cb(er, d) + } + ) +} + +function checkEngine (target, cb) { + var npmv = npm.version + , force = npm.config.get("force") + , nodev = force ? null : npm.config.get("node-version") + , eng = target.engines + if (!eng) return cb() + if (nodev && eng.node && !semver.satisfies(nodev, eng.node) + || eng.npm && !semver.satisfies(npmv, eng.npm)) { + var er = new Error("Unsupported") + er.errno = npm.ENOTSUP + er.required = eng + er.pkgid = target._id + return cb(er) + } + return cb() +} + + +function checkCycle (target, ancestors, cb) { + // there are some very rare and pathological edge-cases where + // a cycle can cause npm to try to install a never-ending tree + // of stuff. + // Simplest: + // + // A -> B -> A' -> B' -> A -> B -> A' -> B' -> A -> ... + // + // Solution: Simply flat-out refuse to install any name@version + // that is already in the prototype tree of the ancestors object. + // A more correct, but more complex, solution would be to symlink + // the deeper thing into the new location. + // Will do that if anyone whines about this irl. + + var p = Object.getPrototypeOf(ancestors) + , name = target.name + , version = target.version + while (p && p !== Object.prototype && p[name] !== version) { + p = Object.getPrototypeOf(p) + } + if (p[name] !== version) return cb() + + var er = new Error("Unresolvable cycle detected") + var tree = [target._id, JSON.parse(JSON.stringify(ancestors))] + , t = Object.getPrototypeOf(ancestors) + while (t && t !== Object.prototype) { + if (t === p) t.THIS_IS_P = true + tree.push(JSON.parse(JSON.stringify(t))) + t = Object.getPrototypeOf(t) + } + log.verbose(tree, "unresolvable dependency tree") + er.pkgid = target._id + er.errno = npm.ECYCLE + return cb(er) +} + +function checkGit (folder, cb) { + // if it's a git repo then don't touch it! + fs.lstat(folder, function (er, s) { + if (er || !s.isDirectory()) return cb() + else checkGit_(folder, cb) + }) +} + +function checkGit_ (folder, cb) { + fs.stat(path.resolve(folder, ".git"), function (er, s) { + if (!er && s.isDirectory()) { + var e = new Error("Appears to be a git repo or submodule.") + e.path = folder + e.errno = npm.EISGIT + return cb(e) + } + cb() + }) +} + +function write (target, targetFolder, family, ancestors, cb_) { + var up = npm.config.get("unsafe-perm") + , user = up ? null : npm.config.get("user") + , group = up ? null : npm.config.get("group") + + function cb (er, data) { + // cache.unpack returns the data object, and all we care about + // is the list of installed packages from that last thing. + if (!er) return cb_(er, data) + log.error(er, "error installing "+target._id) + if (false === npm.config.get("rollback")) return cb_(er) + npm.commands.unbuild([targetFolder], function (er2) { + if (er2) log.error(er2, "error rolling back "+target._id) + return cb_(er, data) + }) + } + + chain + ( [ [ npm.commands.unbuild, [targetFolder] ] + , [ cache.unpack, target.name, target.version, targetFolder + , null, null, user, group ] + , [ lifecycle, target, "preinstall", targetFolder ] ] + + // nest the chain so that we can throw away the results returned + // up until this point, since we really don't care about it. + , function (er) { + if (er) return cb(er) + var deps = Object.keys(target.dependencies || {}) + installMany(deps.filter(function (d) { + // prefer to not install things that are satisfied by + // something in the "family" list. + return !semver.satisfies(family[d], target.dependencies[d]) + }).map(function (d) { + var t = target.dependencies[d] + , parsed = url.parse(t.replace(/^git\+/, "git")) + if (!parsed.protocol) { + t = d + "@" + t + } + return t + }), targetFolder, family, ancestors, false, function (er, d) { + //log.warn(d, "write installMany cb") + log.verbose(targetFolder, "about to build") + if (er) return cb(er) + npm.commands.build( [targetFolder] + , npm.config.get("global") + , true + , function (er) { return cb(er, d) }) + }) + } ) +} diff --git a/deps/npm/lib/link.js b/deps/npm/lib/link.js new file mode 100644 index 0000000000..fea6606666 --- /dev/null +++ b/deps/npm/lib/link.js @@ -0,0 +1,158 @@ + +// link with no args: symlink the folder to the global location +// link with package arg: symlink the global to the local + +var npm = require("./npm.js") + , symlink = require("./utils/link.js") + , fs = require("graceful-fs") + , log = require("./utils/log.js") + , asyncMap = require("slide").asyncMap + , chain = require("slide").chain + , path = require("path") + , relativize = require("./utils/relativize.js") + , rm = require("rimraf") + , output = require("./utils/output.js") + , build = require("./build.js") + +module.exports = link + +link.usage = "npm link (in package dir)" + + "\nnpm link (link global into local)" + +link.completion = function (opts, cb) { + var dir = npm.globalDir + fs.readdir(dir, function (er, files) { + cb(er, files.filter(function (f) { + return f.charAt(0) !== "." + })) + }) +} + +function link (args, cb) { + if (process.platform === "win32") { + var e = new Error("npm link not supported on windows") + e.code = "ENOTSUP" + e.errno = require("constants").ENOTSUP + return cb(e) + } + + if (npm.config.get("global")) { + return cb(new Error("link should never be --global.\n" + +"Please re-run this command with --local")) + } + if (args.length === 1 && args[0] === ".") args = [] + if (args.length) return linkInstall(args, cb) + linkPkg(npm.prefix, cb) +} + +function linkInstall (pkgs, cb) { + asyncMap(pkgs, function (pkg, cb) { + function n (er, data) { + if (er) return cb(er, data) + // install returns [ [folder, pkgId], ... ] + // but we definitely installed just one thing. + var d = data.filter(function (d) { return !d[3] }) + pp = d[0][1] + pkg = path.basename(pp) + target = path.resolve(npm.dir, pkg) + next() + } + + var t = path.resolve(npm.globalDir, "..") + , pp = path.resolve(npm.globalDir, pkg) + , rp = null + , target = path.resolve(npm.dir, pkg) + + // if it's a folder or a random not-installed thing, then + // link or install it first + if (pkg.indexOf("/") !== -1 || pkg.indexOf("\\") !== -1) { + return fs.lstat(path.resolve(pkg), function (er, st) { + if (er || !st.isDirectory()) { + npm.commands.install(t, pkg, n) + } else { + rp = path.resolve(pkg) + linkPkg(rp, n) + } + }) + } + + fs.lstat(pp, function (er, st) { + if (er) { + rp = pp + return npm.commands.install(t, pkg, n) + } else if (!st.isSymbolicLink()) { + rp = pp + next() + } else { + return fs.realpath(pp, function (er, real) { + if (er) log.warn(pkg, "invalid symbolic link") + else rp = real + next() + }) + } + }) + + function next () { + chain + ( [ [npm.commands, "unbuild", [target]] + , [log.verbose, "symlinking " + pp + " to "+target, "link"] + , [symlink, pp, target] + // do run lifecycle scripts - full build here. + , rp && [build, [target]] + , [ resultPrinter, pkg, pp, target, rp ] ] + , cb ) + } + }, cb) +} + +function linkPkg (folder, cb_) { + var me = folder || npm.prefix + , readJson = require("./utils/read-json.js") + readJson( path.resolve(me, "package.json") + , { dev: true } + , function (er, d) { + function cb (er) { + return cb_(er, [[d && d._id, target, null, null]]) + } + if (er) return cb(er) + var target = path.resolve(npm.globalDir, d.name) + rm(target, function (er) { + if (er) return cb(er) + symlink(me, target, function (er) { + if (er) return cb(er) + log.verbose(target, "link: build target") + // also install missing dependencies. + npm.commands.install(me, [], function (er, installed) { + if (er) return cb(er) + // build the global stuff. Don't run *any* scripts, because + // install command already will have done that. + build([target], true, build._noLC, true, function (er) { + if (er) return cb(er) + resultPrinter(path.basename(me), me, target, cb) + }) + }) + }) + }) + }) +} + +function resultPrinter (pkg, src, dest, rp, cb) { + if (typeof cb !== "function") cb = rp, rp = null + var where = relativize(dest, path.resolve(process.cwd(),"x")) + rp = (rp || "").trim() + src = (src || "").trim() + if (npm.config.get("parseable")) { + return parseableOutput(dest, rp || src, cb) + } + if (rp === src) rp = null + output.write(where+" -> " + src + +(rp ? " -> " + rp: ""), cb) +} + +function parseableOutput (dest, rp, cb) { + // link is always effectively "long", since it doesn't help much to + // *just* print the target folder. + // However, we don't actually ever read the version number, so + // the second field is always blank. + output.write(dest + "::" + rp, cb) +} diff --git a/deps/npm/lib/ls.js b/deps/npm/lib/ls.js new file mode 100644 index 0000000000..33336cf63e --- /dev/null +++ b/deps/npm/lib/ls.js @@ -0,0 +1,169 @@ + +// show the installed versions of packages +// +// --parseable creates output like this: +// ::: +// Flags are a :-separated list of zero or more indicators + +module.exports = exports = ls + +var npm = require("./npm.js") + , readInstalled = require("./utils/read-installed.js") + , output = require("./utils/output.js") + , log = require("./utils/log.js") + , relativize = require("./utils/relativize.js") + , path = require("path") + +ls.usage = "npm ls" + +function ls (args, silent, cb) { + if (typeof cb !== "function") cb = silent, silent = false + + if (args.length) { + log.warn("ls doesn't take positional args. Try the 'search' command") + } + + var dir = path.resolve(npm.dir, "..") + + readInstalled(dir, function (er, data) { + if (er || silent) return cb(er, data) + var long = npm.config.get("long") + var out = makePretty(bfsify(data), long, dir).join("\n") + output.write(out, function (er) { cb(er, data) }) + }) +} + +function bfsify (root, current, queue, seen) { + // walk over the data, and turn it from this: + // +-- a + // | `-- b + // | `-- a (truncated) + // `--b (truncated) + // into this: + // +-- a + // `-- b + // which looks nicer + current = current || root + queue = queue || [] + seen = seen || [] + var deps = current.dependencies = current.dependencies || {} + Object.keys(deps).forEach(function (d) { + var dep = deps[d] + if (typeof dep !== "object") return + if (seen.indexOf(dep) !== -1) { + if (npm.config.get("parseable") || !npm.config.get("long")) { + delete deps[d] + return + } else { + dep = deps[d] = Object.create(dep) + dep.dependencies = {} + } + } + queue.push(dep) + seen.push(dep) + }) + if (!queue.length) return root + return bfsify(root, queue.shift(), queue, seen) +} + + +function makePretty (data, long, dir, prefix, list) { + var top = !list + list = list || [] + prefix = prefix || "" + list.push(format(data, long, prefix, dir)) + var deps = data.dependencies || {} + , childPref = prefix.split("├─").join("│ ") + .split("└─").join(" ") + , depList = Object.keys(deps) + , depLast = depList.length - 1 + , maxDepth = npm.config.get("depth") + Object.keys(deps).sort(function (a, b) { + return a > b ? 1 : -1 + }).forEach(function (d, i) { + var depData = deps[d] + if (typeof depData === "string") { + if (data.depth < maxDepth) { + var p = data.link || data.path + log.warn("Unmet dependency in "+p, d+" "+deps[d]) + depData = npm.config.get("parseable") + ? ( npm.config.get("long") + ? path.resolve(data.path, "node_modules", d) + + ":"+d+"@"+JSON.stringify(depData)+":INVALID:MISSING" + : "" ) + : "─ \033[31;40mUNMET DEPENDENCY\033[0m "+d+" "+depData + } else { + if (npm.config.get("parseable")) { + depData = path.resolve(data.path, "node_modules", d) + + (npm.config.get("long") + ? ":" + d + "@" + JSON.stringify(depData) + + ":" // no realpath resolved + + ":MAXDEPTH" + : "") + } else { + depData = "─ "+d+"@'"+depData +"' (max depth reached)" + } + } + } + var c = i === depLast ? "└─" : "├─" + makePretty(depData, long, dir, childPref + c, list) + }) + if (top && list.length === 1 && !data._id) { + if (!npm.config.get("parseable")) { + list.push("(empty)") + } else if (npm.config.get("long")) list[0] += ":EMPTY" + } + return list.filter(function (l) { return l && l.trim() }) +} + +function ugly (data) { + if (typeof data === "string") { + return data + } + if (!npm.config.get("long")) return data.path + + return data.path + + ":" + (data._id || "") + + ":" + (data.realPath !== data.path ? data.realPath : "") + + (data.extraneous ? ":EXTRANEOUS" : "") + + (data.invalid ? ":INVALID" : "") +} + +function format (data, long, prefix, dir) { + if (npm.config.get("parseable")) return ugly(data) + if (typeof data === "string") { + return prefix + data + } +// console.log([data.path, dir], "relativize") + var depLen = Object.keys(data.dependencies).length + , space = prefix.split("├─").join("│ ") + .split("└─").join(" ") + + (depLen ? "" : " ") + , rel = relativize(data.path || "", dir) + , l = prefix + + (rel === "." ? "" : depLen ? "┬ " : "─ ") + + (data._id ? data._id + " " : "") + + (data.link ? "-> " + data.link : "") + "" + + (rel === "." && !(long && data._id) ? dir : "") + if (data.invalid) { + if (data.realName !== data.name) l += " ("+data.realName+")" + l += " \033[31;40minvalid\033[0m" + } + if (data.extraneous && rel !== ".") { + l += " \033[32;40mextraneous\033[0m" + } + if (!long || !data._id) return l + var extras = [] + if (rel !== ".") extras.push(rel) + else extras.push(dir) + if (data.description) extras.push(data.description) + if (data.repository) extras.push(data.repository.url) + if (data.homepage) extras.push(data.homepage) + extras = extras.filter(function (e) { return e }) + var lastExtra = !depLen && extras.length - 1 + l += extras.map(function (e, i) { + var indent = !depLen ? " " : "│ " + return "\n" + space + indent + e + }).join("") + return l +} diff --git a/deps/npm/lib/npm.js b/deps/npm/lib/npm.js new file mode 100644 index 0000000000..51f7d64b88 --- /dev/null +++ b/deps/npm/lib/npm.js @@ -0,0 +1,448 @@ +;(function(){ +// windows: running "npm blah" in this folder will invoke WSH, not node. +if (typeof WScript !== "undefined") { + WScript.echo("npm does not work when run\n" + +"with the Windows Scripting Host\n\n" + +"'cd' to a different directory,\n" + +"or type 'npm.cmd ',\n" + +"or type 'node npm '.") + WScript.quit(1) + return +} + + +process.title = "npm" + + +// FIXME there really ought to be a path.split in node core +require("path").SPLIT_CHAR = process.platform === "win32" ? "\\" : "/" + +var EventEmitter = require("events").EventEmitter + , npm = module.exports = new EventEmitter + , config = require("./config.js") + , set = require("./utils/set.js") + , get = require("./utils/get.js") + , ini = require("./utils/ini.js") + , log = require("./utils/log.js") + , fs = require("graceful-fs") + , path = require("path") + , abbrev = require("abbrev") + , which = require("which") + , semver = require("semver") + , findPrefix = require("./utils/find-prefix.js") + , getUid = require("./utils/uid-number.js") + , mkdir = require("./utils/mkdir-p.js") + +npm.commands = {} +npm.ELIFECYCLE = {} +npm.E404 = {} +npm.EPUBLISHCONFLICT = {} +npm.EJSONPARSE = {} +npm.EISGIT = {} +npm.ECYCLE = {} +npm.ENOTSUP = {} + +// HACK for windows +if (process.platform === "win32") { + // stub in unavailable methods from process and fs binding + if (!process.getuid) process.getuid = function() {} + if (!process.getgid) process.getgid = function() {} + var fsBinding = process.binding("fs") + if (!fsBinding.chown) fsBinding.chown = function() { + var cb = arguments[arguments.length - 1] + if (typeof cb == "function") cb() + } + + // patch rename/renameSync, but this should really be fixed in node + var _fsRename = fs.rename + , _fsPathPatch + _fsPathPatch = function(p) { + return p && p.replace(/\\/g, "/") || p; + } + fs.rename = function(p1, p2) { + arguments[0] = _fsPathPatch(p1) + arguments[1] = _fsPathPatch(p2) + return _fsRename.apply(fs, arguments); + } +} + +try { + // startup, ok to do this synchronously + var j = JSON.parse(fs.readFileSync( + path.join(__dirname, "../package.json"))+"") + npm.version = j.version + npm.nodeVersionRequired = j.engines.node + if (!semver.satisfies(process.version, j.engines.node)) { + log.error(["" + ,"npm requires node version: "+j.engines.node + ,"And you have: "+process.version + ,"which is not satisfactory." + ,"" + ,"Bad things will likely happen. You have been warned." + ,""].join("\n"), "unsupported version") + } +} catch (ex) { + try { + log(ex, "error reading version") + } catch (er) {} + npm.version = ex +} + +var commandCache = {} + // short names for common things + , aliases = { "rm" : "uninstall" + , "r" : "uninstall" + , "un" : "uninstall" + , "unlink" : "uninstall" + , "remove" : "uninstall" + , "rb" : "rebuild" + , "list" : "ls" + , "la" : "ls" + , "ll" : "ls" + , "ln" : "link" + , "i" : "install" + , "up" : "update" + , "c" : "config" + , "info" : "view" + , "show" : "view" + , "find" : "search" + , "s" : "search" + , "se" : "search" + , "author" : "owner" + , "home" : "docs" + , "unstar": "star" // same function + , "apihelp" : "help" + } + + , aliasNames = Object.keys(aliases) + // these are filenames in . + , cmdList = [ "install" + , "uninstall" + , "cache" + , "config" + , "set" + , "get" + , "update" + , "outdated" + , "prune" + , "submodule" + , "pack" + + , "rebuild" + , "link" + + , "publish" + , "star" + , "tag" + , "adduser" + , "unpublish" + , "owner" + , "deprecate" + + , "help" + , "help-search" + , "ls" + , "search" + , "view" + , "init" + , "version" + , "edit" + , "explore" + , "docs" + , "bugs" + , "faq" + , "root" + , "prefix" + , "bin" + , "whoami" + + , "test" + , "stop" + , "start" + , "restart" + , "run-script" + , "completion" + ] + , plumbing = [ "build" + , "unbuild" + , "xmas" + ] + , fullList = npm.fullList = cmdList.concat(aliasNames).filter(function (c) { + return plumbing.indexOf(c) === -1 + }) + , abbrevs = abbrev(fullList) + +Object.keys(abbrevs).concat(plumbing).forEach(function addCommand (c) { + Object.defineProperty(npm.commands, c, { get : function () { + if (!loaded) throw new Error( + "Call npm.load(conf, cb) before using this command.\n"+ + "See the README.md or cli.js for example usage.") + var a = npm.deref(c) + if (c === "la" || c === "ll") { + npm.config.set("long", true) + } + npm.command = c + if (commandCache[a]) return commandCache[a] + var cmd = require(__dirname+"/"+a+".js") + commandCache[a] = function () { + var args = Array.prototype.slice.call(arguments, 0) + if (typeof args[args.length - 1] !== "function") { + args.push(defaultCb) + } + if (args.length === 1) args.unshift([]) + cmd.apply(npm, args) + } + Object.keys(cmd).forEach(function (k) { + commandCache[a][k] = cmd[k] + }) + return commandCache[a] + }, enumerable: fullList.indexOf(c) !== -1 }) + + // make css-case commands callable via camelCase as well + if (c.match(/\-([a-z])/)) { + addCommand(c.replace(/\-([a-z])/g, function (a, b) { + return b.toUpperCase() + })) + } +}) + +function defaultCb (er, data) { + if (er) console.error(er.stack || er.message) + else console.log(data) +} + +npm.deref = function (c) { + if (!c) return "" + if (c.match(/[A-Z]/)) c = c.replace(/([A-Z])/g, function (m) { + return "-" + m.toLowerCase() + }) + if (plumbing.indexOf(c) !== -1) return c + var a = abbrevs[c] + if (aliases[a]) a = aliases[a] + return a +} + +var loaded = false + , loading = false + , loadErr = null + , loadListeners = [] + +function loadCb (er) { + loadListeners.forEach(function (cb) { + process.nextTick(cb.bind(npm, er, npm)) + }) + loadListeners.length = 0 +} + + +npm.load = function (conf, cb_) { + if (!cb_ && typeof conf === "function") cb_ = conf , conf = {} + if (!cb_) cb_ = function () {} + if (!conf) conf = {} + loadListeners.push(cb_) + if (loaded || loadErr) return cb(loadErr) + if (loading) return + loading = true + var onload = true + + function cb (er) { + if (loadErr) return + loaded = true + loadCb(loadErr = er) + if (onload = onload && npm.config.get("onload-script")) { + require(onload) + onload = false + } + } + + log.waitForConfig() + + load(npm, conf, cb) +} + + +function load (npm, conf, cb) { + which(process.argv[0], function (er, node) { + //console.error("back from which") + if (!er && node.toUpperCase() !== process.execPath.toUpperCase()) { + log.verbose("node symlink", node) + process.execPath = node + process.installPrefix = path.resolve(node, "..", "..") + } + + // look up configs + //console.error("about to look up configs") + + ini.resolveConfigs(conf, function (er) { + //console.error("back from config lookup", er && er.stack) + if (er) return cb(er) + + var n = 2 + , errState + + var umask = parseInt(conf.umask, 8) + npm.modes = { exec: 0777 & (~umask) + , file: 0666 & (~umask) + , umask: umask } + + loadPrefix(npm, conf, next) + loadUid(npm, conf, next) + + function next (er) { + //console.error("next", er && er.stack) + if (errState) return + if (er) return cb(errState = er) + if (-- n <= 0) return cb() + } + }) + }) +} + +function loadPrefix (npm, conf, cb) { + // try to guess at a good node_modules location. + var p + , gp + if (!conf.hasOwnProperty("prefix")) { + p = process.cwd() + } else { + p = npm.config.get("prefix") + } + gp = npm.config.get("prefix") + + findPrefix(p, function (er, p) { + //console.log("Back from findPrefix", er && er.stack, p) + Object.defineProperty(npm, "localPrefix", + { get : function () { return p } + , set : function (r) { return p = r } + , enumerable : true + }) + // the prefix MUST exist, or else nothing works. + mkdir(p, npm.modes.exec, null, null, true, next) + }) + + findPrefix(gp, function (er, gp) { + Object.defineProperty(npm, "globalPrefix", + { get : function () { return gp } + , set : function (r) { return gp = r } + , enumerable : true + }) + // the prefix MUST exist, or else nothing works. + mkdir(gp, npm.modes.exec, null, null, true, next) + }) + + var i = 2 + , errState = null + function next (er) { + if (errState) return + if (er) return cb(errState = er) + if (--i === 0) return cb() + } +} + + +function loadUid (npm, conf, cb) { + // if we're not in unsafe-perm mode, then figure out who + // to run stuff as. Do this first, to support `npm update npm -g` + if (!npm.config.get("unsafe-perm")) { + getUid(npm.config.get("user"), npm.config.get("group"), cb) + } else { + //console.error("skipping loadUid") + process.nextTick(cb) + } +} + + +npm.config = + { get : function (key) { return ini.get(key) } + , set : function (key, val) { return ini.set(key, val, "cli") } + , del : function (key, val) { return ini.del(key, val, "cli") } + } + +Object.defineProperty(npm, "prefix", + { get : function () { + return npm.config.get("global") ? npm.globalPrefix : npm.localPrefix + } + , set : function (r) { + var k = npm.config.get("global") ? "globalPrefix" : "localPrefix" + return npm[k] = r + } + , enumerable : true + }) + +Object.defineProperty(npm, "bin", + { get : function () { + if (npm.config.get("global")) return npm.globalBin + return path.resolve(npm.root, ".bin") + } + , enumerable : true + }) + +Object.defineProperty(npm, "globalBin", + { get : function () { + var b = npm.globalPrefix + if (process.platform !== "win32") b = path.resolve(b, "bin") + return b + } + }) + +Object.defineProperty(npm, "dir", + { get : function () { + if (npm.config.get("global")) return npm.globalDir + return path.resolve(npm.prefix, "node_modules") + } + , enumerable : true + }) + +Object.defineProperty(npm, "globalDir", + { get : function () { + return (process.platform !== "win32") + ? path.resolve(npm.globalPrefix, "lib", "node_modules") + : path.resolve(npm.globalPrefix, "node_modules") + } + , enumerable : true + }) + +Object.defineProperty(npm, "root", + { get : function () { return npm.dir } }) + +Object.defineProperty(npm, "cache", + { get : function () { return npm.config.get("cache") } + , set : function (r) { return npm.config.set("cache", r) } + , enumerable : true + }) + +var tmpFolder +Object.defineProperty(npm, "tmp", + { get : function () { + if (!tmpFolder) tmpFolder = "npm-"+Date.now() + return path.resolve(npm.config.get("tmp"), tmpFolder) + } + , enumerable : true + }) + +// the better to repl you with +Object.getOwnPropertyNames(npm.commands).forEach(function (n) { + if (npm.hasOwnProperty(n)) return + + Object.defineProperty(npm, n, { get: function () { + return function () { + var args = Array.prototype.slice.call(arguments, 0) + , cb = defaultCb + + if (args.length === 1 && Array.isArray(args[0])) { + args = args[0] + } + + if (typeof args[args.length - 1] === "function") { + cb = args.pop() + } + + npm.commands[n](args, cb) + } + }, enumerable: false, configurable: true }) +}) + +if (require.main === module) { + require("../bin/npm-cli.js") +} +})() diff --git a/deps/npm/lib/outdated.js b/deps/npm/lib/outdated.js new file mode 100644 index 0000000000..496dfbd426 --- /dev/null +++ b/deps/npm/lib/outdated.js @@ -0,0 +1,146 @@ +/* + +npm outdated [pkg] + +Does the following: + +1. check for a new version of pkg + +If no packages are specified, then run for all installed +packages. + +*/ + +module.exports = outdated + +outdated.usage = "npm outdated [ [ ...]]" + +outdated.completion = require("./utils/completion/installed-deep.js") + + +var path = require("path") + , fs = require("graceful-fs") + , readJson = require("./utils/read-json.js") + , cache = require("./cache.js") + , asyncMap = require("slide").asyncMap + , npm = require("./npm.js") + , log = require("./utils/log.js") + , semver = require("semver") + , relativize = require("./utils/relativize.js") + +function outdated (args, silent, cb) { + if (typeof cb !== "function") cb = silent, silent = false + var dir = path.resolve(npm.dir, "..") + outdated_(args, dir, {}, function (er, list) { + function cb_ (er) { cb(er, list) } + + if (er || silent) return cb_(er) + var outList = list.map(makePretty) + require("./utils/output.js").write(outList.join("\n"), cb_) + }) +} + +// [[ dir, dep, has, want ]] +function makePretty (p) { + var parseable = npm.config.get("parseable") + , long = npm.config.get("long") + , dep = p[1] + , dir = path.resolve(p[0], "node_modules", dep) + , has = p[2] + , want = p[3] + if (parseable) { + var str = dir + if (npm.config.get("long")) { + str += ":" + dep + "@" + want + + ":" + (has ? (dep + "@" + has) : "MISSING") + } + return str + } + + if (!npm.config.get("global")) { + dir = relativize(dir, process.cwd()+"/x") + } + return dep + "@" + want + " " + dir + + " current=" + (has || "MISSING") +} + +function outdated_ (args, dir, parentHas, cb) { + // get the deps from package.json, or {