From 05a8d231bb7784e18560280e5183f9efe82319e7 Mon Sep 17 00:00:00 2001 From: Michael Yin Date: Mon, 29 Sep 2014 13:35:20 +0800 Subject: [PATCH] 0.2.0 --- .gitignore | 2 + .npmignore | 3 +- README.md | 19 +- app.iced | 36 +- client.iced | 272 +- node_modules/.bin/express | 1 - node_modules/.bin/icake | 1 - node_modules/.bin/iced | 1 - node_modules/.bin/jade | 1 - node_modules/express/.npmignore | 9 - node_modules/express/.travis.yml | 4 - node_modules/express/History.md | 1106 -- node_modules/express/LICENSE | 22 - node_modules/express/Makefile | 33 - node_modules/express/Readme.md | 180 - node_modules/express/bin/express | 422 - node_modules/express/client.js | 25 - node_modules/express/index.js | 4 - node_modules/express/lib/application.js | 530 - node_modules/express/lib/express.js | 92 - node_modules/express/lib/middleware.js | 33 - node_modules/express/lib/request.js | 496 - node_modules/express/lib/response.js | 756 -- node_modules/express/lib/router/index.js | 273 - node_modules/express/lib/router/route.js | 72 - node_modules/express/lib/utils.js | 302 - node_modules/express/lib/view.js | 76 - .../node_modules/buffer-crc32/.npmignore | 1 - .../node_modules/buffer-crc32/.travis.yml | 8 - .../node_modules/buffer-crc32/README.md | 47 - .../node_modules/buffer-crc32/index.js | 88 - .../node_modules/buffer-crc32/package.json | 39 - .../buffer-crc32/tests/crc.test.js | 89 - .../express/node_modules/commander/.npmignore | 4 - .../node_modules/commander/.travis.yml | 4 - .../express/node_modules/commander/History.md | 107 - .../express/node_modules/commander/Makefile | 7 - .../express/node_modules/commander/Readme.md | 262 - .../express/node_modules/commander/index.js | 2 - .../node_modules/commander/lib/commander.js | 1026 -- .../node_modules/commander/package.json | 38 - .../express/node_modules/connect/.npmignore | 12 - .../express/node_modules/connect/.travis.yml | 5 - .../express/node_modules/connect/LICENSE | 24 - .../express/node_modules/connect/Readme.md | 133 - .../express/node_modules/connect/index.js | 4 - .../express/node_modules/connect/lib/cache.js | 81 - .../node_modules/connect/lib/connect.js | 92 - .../express/node_modules/connect/lib/index.js | 50 - .../connect/lib/middleware/basicAuth.js | 103 - .../connect/lib/middleware/bodyParser.js | 61 - .../connect/lib/middleware/compress.js | 152 - .../connect/lib/middleware/cookieParser.js | 62 - .../connect/lib/middleware/cookieSession.js | 117 - .../connect/lib/middleware/csrf.js | 73 - .../connect/lib/middleware/directory.js | 229 - .../connect/lib/middleware/errorHandler.js | 86 - .../connect/lib/middleware/favicon.js | 80 - .../connect/lib/middleware/json.js | 86 - .../connect/lib/middleware/limit.js | 78 - .../connect/lib/middleware/logger.js | 339 - .../connect/lib/middleware/methodOverride.js | 40 - .../connect/lib/middleware/multipart.js | 133 - .../connect/lib/middleware/query.js | 46 - .../connect/lib/middleware/responseTime.js | 32 - .../connect/lib/middleware/session.js | 356 - .../connect/lib/middleware/session/cookie.js | 140 - .../connect/lib/middleware/session/memory.js | 129 - .../connect/lib/middleware/session/session.js | 116 - .../connect/lib/middleware/session/store.js | 84 - .../connect/lib/middleware/static.js | 95 - .../connect/lib/middleware/staticCache.js | 231 - .../connect/lib/middleware/timeout.js | 55 - .../connect/lib/middleware/urlencoded.js | 78 - .../connect/lib/middleware/vhost.js | 40 - .../express/node_modules/connect/lib/patch.js | 79 - .../express/node_modules/connect/lib/proto.js | 230 - .../connect/lib/public/directory.html | 81 - .../connect/lib/public/error.html | 14 - .../connect/lib/public/favicon.ico | Bin 1406 -> 0 bytes .../connect/lib/public/icons/page.png | Bin 635 -> 0 bytes .../connect/lib/public/icons/page_add.png | Bin 739 -> 0 bytes .../connect/lib/public/icons/page_attach.png | Bin 794 -> 0 bytes .../connect/lib/public/icons/page_code.png | Bin 818 -> 0 bytes .../connect/lib/public/icons/page_copy.png | Bin 663 -> 0 bytes .../connect/lib/public/icons/page_delete.png | Bin 740 -> 0 bytes .../connect/lib/public/icons/page_edit.png | Bin 807 -> 0 bytes .../connect/lib/public/icons/page_error.png | Bin 793 -> 0 bytes .../connect/lib/public/icons/page_excel.png | Bin 817 -> 0 bytes .../connect/lib/public/icons/page_find.png | Bin 879 -> 0 bytes .../connect/lib/public/icons/page_gear.png | Bin 833 -> 0 bytes .../connect/lib/public/icons/page_go.png | Bin 779 -> 0 bytes .../connect/lib/public/icons/page_green.png | Bin 621 -> 0 bytes .../connect/lib/public/icons/page_key.png | Bin 801 -> 0 bytes .../lib/public/icons/page_lightning.png | Bin 839 -> 0 bytes .../connect/lib/public/icons/page_link.png | Bin 830 -> 0 bytes .../lib/public/icons/page_paintbrush.png | Bin 813 -> 0 bytes .../connect/lib/public/icons/page_paste.png | Bin 703 -> 0 bytes .../connect/lib/public/icons/page_red.png | Bin 641 -> 0 bytes .../connect/lib/public/icons/page_refresh.png | Bin 858 -> 0 bytes .../connect/lib/public/icons/page_save.png | Bin 774 -> 0 bytes .../connect/lib/public/icons/page_white.png | Bin 294 -> 0 bytes .../lib/public/icons/page_white_acrobat.png | Bin 591 -> 0 bytes .../public/icons/page_white_actionscript.png | Bin 664 -> 0 bytes .../lib/public/icons/page_white_add.png | Bin 512 -> 0 bytes .../connect/lib/public/icons/page_white_c.png | Bin 587 -> 0 bytes .../lib/public/icons/page_white_camera.png | Bin 656 -> 0 bytes .../lib/public/icons/page_white_cd.png | Bin 666 -> 0 bytes .../lib/public/icons/page_white_code.png | Bin 603 -> 0 bytes .../lib/public/icons/page_white_code_red.png | Bin 587 -> 0 bytes .../public/icons/page_white_coldfusion.png | Bin 592 -> 0 bytes .../public/icons/page_white_compressed.png | Bin 724 -> 0 bytes .../lib/public/icons/page_white_copy.png | Bin 309 -> 0 bytes .../lib/public/icons/page_white_cplusplus.png | Bin 621 -> 0 bytes .../lib/public/icons/page_white_csharp.png | Bin 700 -> 0 bytes .../lib/public/icons/page_white_cup.png | Bin 639 -> 0 bytes .../lib/public/icons/page_white_database.png | Bin 579 -> 0 bytes .../lib/public/icons/page_white_delete.png | Bin 536 -> 0 bytes .../lib/public/icons/page_white_dvd.png | Bin 638 -> 0 bytes .../lib/public/icons/page_white_edit.png | Bin 618 -> 0 bytes .../lib/public/icons/page_white_error.png | Bin 623 -> 0 bytes .../lib/public/icons/page_white_excel.png | Bin 663 -> 0 bytes .../lib/public/icons/page_white_find.png | Bin 676 -> 0 bytes .../lib/public/icons/page_white_flash.png | Bin 582 -> 0 bytes .../lib/public/icons/page_white_freehand.png | Bin 639 -> 0 bytes .../lib/public/icons/page_white_gear.png | Bin 402 -> 0 bytes .../lib/public/icons/page_white_get.png | Bin 516 -> 0 bytes .../lib/public/icons/page_white_go.png | Bin 612 -> 0 bytes .../connect/lib/public/icons/page_white_h.png | Bin 603 -> 0 bytes .../public/icons/page_white_horizontal.png | Bin 296 -> 0 bytes .../lib/public/icons/page_white_key.png | Bin 616 -> 0 bytes .../lib/public/icons/page_white_lightning.png | Bin 669 -> 0 bytes .../lib/public/icons/page_white_link.png | Bin 614 -> 0 bytes .../lib/public/icons/page_white_magnify.png | Bin 554 -> 0 bytes .../lib/public/icons/page_white_medal.png | Bin 706 -> 0 bytes .../lib/public/icons/page_white_office.png | Bin 779 -> 0 bytes .../lib/public/icons/page_white_paint.png | Bin 688 -> 0 bytes .../public/icons/page_white_paintbrush.png | Bin 618 -> 0 bytes .../lib/public/icons/page_white_paste.png | Bin 620 -> 0 bytes .../lib/public/icons/page_white_php.png | Bin 538 -> 0 bytes .../lib/public/icons/page_white_picture.png | Bin 650 -> 0 bytes .../public/icons/page_white_powerpoint.png | Bin 588 -> 0 bytes .../lib/public/icons/page_white_put.png | Bin 523 -> 0 bytes .../lib/public/icons/page_white_ruby.png | Bin 626 -> 0 bytes .../lib/public/icons/page_white_stack.png | Bin 317 -> 0 bytes .../lib/public/icons/page_white_star.png | Bin 565 -> 0 bytes .../lib/public/icons/page_white_swoosh.png | Bin 634 -> 0 bytes .../lib/public/icons/page_white_text.png | Bin 342 -> 0 bytes .../public/icons/page_white_text_width.png | Bin 315 -> 0 bytes .../lib/public/icons/page_white_tux.png | Bin 668 -> 0 bytes .../lib/public/icons/page_white_vector.png | Bin 644 -> 0 bytes .../public/icons/page_white_visualstudio.png | Bin 702 -> 0 bytes .../lib/public/icons/page_white_width.png | Bin 309 -> 0 bytes .../lib/public/icons/page_white_word.png | Bin 651 -> 0 bytes .../lib/public/icons/page_white_world.png | Bin 734 -> 0 bytes .../lib/public/icons/page_white_wrench.png | Bin 613 -> 0 bytes .../lib/public/icons/page_white_zip.png | Bin 386 -> 0 bytes .../connect/lib/public/icons/page_word.png | Bin 777 -> 0 bytes .../connect/lib/public/icons/page_world.png | Bin 903 -> 0 bytes .../node_modules/connect/lib/public/style.css | 141 - .../express/node_modules/connect/lib/utils.js | 404 - .../node_modules/buffer-crc32/.npmignore | 1 - .../node_modules/buffer-crc32/.travis.yml | 8 - .../node_modules/buffer-crc32/README.md | 33 - .../node_modules/buffer-crc32/index.js | 84 - .../node_modules/buffer-crc32/package.json | 34 - .../buffer-crc32/tests/crc.test.js | 52 - .../connect/node_modules/bytes/.npmignore | 1 - .../connect/node_modules/bytes/History.md | 10 - .../connect/node_modules/bytes/Makefile | 7 - .../connect/node_modules/bytes/Readme.md | 51 - .../connect/node_modules/bytes/component.json | 7 - .../connect/node_modules/bytes/index.js | 39 - .../connect/node_modules/bytes/package.json | 20 - .../node_modules/formidable/.npmignore | 4 - .../node_modules/formidable/.travis.yml | 4 - .../connect/node_modules/formidable/Makefile | 14 - .../connect/node_modules/formidable/Readme.md | 311 - .../connect/node_modules/formidable/TODO | 3 - .../benchmark/bench-multipart-parser.js | 70 - .../node_modules/formidable/example/post.js | 43 - .../node_modules/formidable/example/upload.js | 48 - .../connect/node_modules/formidable/index.js | 1 - .../node_modules/formidable/lib/file.js | 73 - .../formidable/lib/incoming_form.js | 384 - .../node_modules/formidable/lib/index.js | 3 - .../formidable/lib/multipart_parser.js | 312 - .../formidable/lib/querystring_parser.js | 25 - .../node_modules/formidable/lib/util.js | 6 - .../formidable/node-gently/Makefile | 4 - .../formidable/node-gently/Readme.md | 167 - .../formidable/node-gently/example/dog.js | 22 - .../node-gently/example/event_emitter.js | 11 - .../formidable/node-gently/index.js | 1 - .../node-gently/lib/gently/gently.js | 184 - .../node-gently/lib/gently/index.js | 1 - .../formidable/node-gently/package.json | 14 - .../formidable/node-gently/test/common.js | 8 - .../node-gently/test/simple/test-gently.js | 348 - .../node_modules/formidable/package.json | 28 - .../node_modules/formidable/test/common.js | 19 - .../test/fixture/file/funkyfilename.txt | 1 - .../formidable/test/fixture/file/plain.txt | 1 - .../http/special-chars-in-filename/info.md | 3 - .../formidable/test/fixture/js/no-filename.js | 3 - .../fixture/js/special-chars-in-filename.js | 21 - .../formidable/test/fixture/multipart.js | 72 - .../test/integration/test-fixtures.js | 89 - .../formidable/test/legacy/common.js | 24 - .../integration/test-multipart-parser.js | 80 - .../test/legacy/simple/test-file.js | 104 - .../test/legacy/simple/test-incoming-form.js | 727 -- .../legacy/simple/test-multipart-parser.js | 50 - .../legacy/simple/test-querystring-parser.js | 45 - .../legacy/system/test-multi-video-upload.js | 75 - .../node_modules/formidable/test/run.js | 2 - .../test/unit/test-incoming-form.js | 63 - .../node_modules/formidable/tool/record.js | 47 - .../connect/node_modules/pause/.npmignore | 4 - .../connect/node_modules/pause/History.md | 5 - .../connect/node_modules/pause/Makefile | 7 - .../connect/node_modules/pause/Readme.md | 29 - .../connect/node_modules/pause/index.js | 29 - .../connect/node_modules/pause/package.json | 20 - .../connect/node_modules/qs/.gitmodules | 6 - .../connect/node_modules/qs/.npmignore | 1 - .../connect/node_modules/qs/.travis.yml | 4 - .../connect/node_modules/qs/History.md | 83 - .../connect/node_modules/qs/Makefile | 12 - .../connect/node_modules/qs/Readme.md | 58 - .../connect/node_modules/qs/benchmark.js | 17 - .../connect/node_modules/qs/component.json | 6 - .../connect/node_modules/qs/examples.js | 51 - .../connect/node_modules/qs/index.js | 2 - .../connect/node_modules/qs/lib/head.js | 1 - .../node_modules/qs/lib/querystring.js | 262 - .../connect/node_modules/qs/lib/tail.js | 1 - .../connect/node_modules/qs/package.json | 39 - .../connect/node_modules/qs/querystring.js | 254 - .../node_modules/qs/test/browser/expect.js | 1202 --- .../node_modules/qs/test/browser/index.html | 18 - .../node_modules/qs/test/browser/jquery.js | 8981 ----------------- .../node_modules/qs/test/browser/mocha.css | 163 - .../node_modules/qs/test/browser/mocha.js | 4201 -------- .../node_modules/qs/test/browser/qs.css | 0 .../node_modules/qs/test/browser/qs.js | 351 - .../connect/node_modules/qs/test/parse.js | 147 - .../connect/node_modules/qs/test/stringify.js | 73 - .../express/node_modules/connect/package.json | 53 - .../express/node_modules/connect/test.js | 40 - .../node_modules/cookie-signature/.npmignore | 4 - .../node_modules/cookie-signature/History.md | 5 - .../node_modules/cookie-signature/Makefile | 7 - .../node_modules/cookie-signature/Readme.md | 42 - .../node_modules/cookie-signature/index.js | 43 - .../cookie-signature/package.json | 24 - .../express/node_modules/cookie/.npmignore | 1 - .../express/node_modules/cookie/.travis.yml | 4 - .../express/node_modules/cookie/README.md | 44 - .../express/node_modules/cookie/index.js | 61 - .../express/node_modules/cookie/package.json | 36 - .../node_modules/cookie/test/mocha.opts | 1 - .../express/node_modules/cookie/test/parse.js | 28 - .../node_modules/cookie/test/serialize.js | 59 - .../express/node_modules/debug/.npmignore | 4 - .../express/node_modules/debug/History.md | 62 - .../express/node_modules/debug/Readme.md | 115 - .../express/node_modules/debug/component.json | 9 - .../express/node_modules/debug/debug.js | 124 - .../express/node_modules/debug/example/app.js | 19 - .../node_modules/debug/example/browser.html | 24 - .../node_modules/debug/example/wildcards.js | 10 - .../node_modules/debug/example/worker.js | 22 - .../express/node_modules/debug/index.js | 5 - .../express/node_modules/debug/lib/debug.js | 134 - .../express/node_modules/debug/package.json | 40 - .../express/node_modules/fresh/.npmignore | 1 - .../express/node_modules/fresh/Makefile | 7 - .../express/node_modules/fresh/Readme.md | 32 - .../express/node_modules/fresh/index.js | 49 - .../express/node_modules/fresh/package.json | 20 - .../express/node_modules/methods/index.js | 26 - .../express/node_modules/methods/package.json | 20 - .../express/node_modules/mkdirp/.npmignore | 2 - .../express/node_modules/mkdirp/.travis.yml | 5 - .../express/node_modules/mkdirp/LICENSE | 21 - .../node_modules/mkdirp/examples/pow.js | 6 - .../express/node_modules/mkdirp/index.js | 82 - .../express/node_modules/mkdirp/package.json | 33 - .../node_modules/mkdirp/readme.markdown | 63 - .../express/node_modules/mkdirp/test/chmod.js | 38 - .../node_modules/mkdirp/test/clobber.js | 37 - .../node_modules/mkdirp/test/mkdirp.js | 28 - .../express/node_modules/mkdirp/test/perm.js | 32 - .../node_modules/mkdirp/test/perm_sync.js | 39 - .../express/node_modules/mkdirp/test/race.js | 41 - .../express/node_modules/mkdirp/test/rel.js | 32 - .../node_modules/mkdirp/test/return.js | 25 - .../node_modules/mkdirp/test/return_sync.js | 24 - .../express/node_modules/mkdirp/test/root.js | 18 - .../express/node_modules/mkdirp/test/sync.js | 32 - .../express/node_modules/mkdirp/test/umask.js | 28 - .../node_modules/mkdirp/test/umask_sync.js | 32 - .../node_modules/range-parser/.npmignore | 1 - .../node_modules/range-parser/History.md | 15 - .../node_modules/range-parser/Makefile | 7 - .../node_modules/range-parser/Readme.md | 28 - .../node_modules/range-parser/index.js | 49 - .../node_modules/range-parser/package.json | 20 - .../express/node_modules/send/.npmignore | 4 - .../express/node_modules/send/History.md | 25 - .../express/node_modules/send/Makefile | 8 - .../express/node_modules/send/Readme.md | 123 - .../express/node_modules/send/index.js | 2 - .../express/node_modules/send/lib/send.js | 473 - .../express/node_modules/send/lib/utils.js | 47 - .../send/node_modules/mime/LICENSE | 19 - .../send/node_modules/mime/README.md | 63 - .../send/node_modules/mime/mime.js | 104 - .../send/node_modules/mime/package.json | 35 - .../send/node_modules/mime/test.js | 55 - .../send/node_modules/mime/types/mime.types | 1510 --- .../send/node_modules/mime/types/node.types | 65 - .../express/node_modules/send/package.json | 34 - node_modules/express/package.json | 84 - node_modules/express/test.js | 39 - node_modules/iced-coffee-script/.npmignore | 11 - node_modules/iced-coffee-script/CNAME | 1 - .../iced-coffee-script/CONTRIBUTING.md | 9 - node_modules/iced-coffee-script/LICENSE | 22 - node_modules/iced-coffee-script/README | 56 - node_modules/iced-coffee-script/Rakefile | 79 - node_modules/iced-coffee-script/UPDATING.md | 167 - node_modules/iced-coffee-script/bin/cake | 7 - node_modules/iced-coffee-script/bin/coffee | 7 - .../extras/coffee-script-iced-large.js | 8 - .../extras/coffee-script-iced.js | 8 - node_modules/iced-coffee-script/iced.md | 709 -- .../lib/coffee-script/browser.js | 120 - .../lib/coffee-script/cake.js | 116 - .../lib/coffee-script/coffee-script.js | 373 - .../lib/coffee-script/command.js | 546 - .../lib/coffee-script/grammar.js | 641 -- .../lib/coffee-script/helpers.js | 227 - .../lib/coffee-script/iced.js | 256 - .../lib/coffee-script/icedlib.js | 288 - .../lib/coffee-script/index.js | 13 - .../lib/coffee-script/lexer.js | 907 -- .../lib/coffee-script/nodes.js | 4280 -------- .../lib/coffee-script/optparse.js | 141 - .../lib/coffee-script/parser.js | 620 -- .../lib/coffee-script/repl.js | 194 - .../lib/coffee-script/rewriter.js | 494 - .../lib/coffee-script/scope.js | 150 - .../lib/coffee-script/sourcemap.js | 163 - .../iced-coffee-script/media/detail.png | Bin 42586 -> 0 bytes .../iced-coffee-script/media/post-rotate.png | Bin 104388 -> 0 bytes .../iced-coffee-script/media/rotate.graffle | 6497 ------------ .../iced-coffee-script/media/rotate1.png | Bin 67189 -> 0 bytes .../iced-coffee-script/media/rotate2.png | Bin 90053 -> 0 bytes .../iced-coffee-script/media/rotate3.png | Bin 111758 -> 0 bytes .../iced-coffee-script/media/rotate4.png | Bin 149621 -> 0 bytes .../iced-coffee-script/media/rotate5.png | Bin 74346 -> 0 bytes .../iced-coffee-script/media/sketch.pdf | Bin 99754 -> 0 bytes node_modules/iced-coffee-script/package.json | 50 - node_modules/jade/.npmignore | 14 - node_modules/jade/LICENSE | 22 - node_modules/jade/Readme.md | 1300 --- node_modules/jade/Readme_zh-cn.md | 921 -- node_modules/jade/bin/jade | 168 - node_modules/jade/index.js | 4 - node_modules/jade/jade.js | 3654 ------- node_modules/jade/jade.md | 510 - node_modules/jade/jade.min.js | 2 - node_modules/jade/lib/compiler.js | 655 -- node_modules/jade/lib/doctypes.js | 18 - node_modules/jade/lib/filters.js | 105 - node_modules/jade/lib/inline-tags.js | 28 - node_modules/jade/lib/jade.js | 253 - node_modules/jade/lib/lexer.js | 775 -- node_modules/jade/lib/nodes/attrs.js | 77 - node_modules/jade/lib/nodes/block-comment.js | 33 - node_modules/jade/lib/nodes/block.js | 122 - node_modules/jade/lib/nodes/case.js | 43 - node_modules/jade/lib/nodes/code.js | 35 - node_modules/jade/lib/nodes/comment.js | 32 - node_modules/jade/lib/nodes/doctype.js | 29 - node_modules/jade/lib/nodes/each.js | 35 - node_modules/jade/lib/nodes/filter.js | 34 - node_modules/jade/lib/nodes/index.js | 20 - node_modules/jade/lib/nodes/literal.js | 32 - node_modules/jade/lib/nodes/mixin.js | 36 - node_modules/jade/lib/nodes/node.js | 25 - node_modules/jade/lib/nodes/tag.js | 95 - node_modules/jade/lib/nodes/text.js | 36 - node_modules/jade/lib/parser.js | 699 -- node_modules/jade/lib/runtime.js | 174 - node_modules/jade/lib/self-closing.js | 19 - node_modules/jade/lib/utils.js | 68 - .../jade/node_modules/commander/.npmignore | 4 - .../jade/node_modules/commander/.travis.yml | 4 - .../jade/node_modules/commander/History.md | 107 - .../jade/node_modules/commander/Makefile | 7 - .../jade/node_modules/commander/Readme.md | 262 - .../jade/node_modules/commander/index.js | 2 - .../node_modules/commander/lib/commander.js | 1026 -- .../jade/node_modules/commander/package.json | 38 - .../jade/node_modules/mkdirp/.npmignore | 2 - .../jade/node_modules/mkdirp/.travis.yml | 5 - node_modules/jade/node_modules/mkdirp/LICENSE | 21 - .../jade/node_modules/mkdirp/examples/pow.js | 6 - .../jade/node_modules/mkdirp/index.js | 82 - .../jade/node_modules/mkdirp/package.json | 33 - .../jade/node_modules/mkdirp/readme.markdown | 63 - .../jade/node_modules/mkdirp/test/chmod.js | 38 - .../jade/node_modules/mkdirp/test/clobber.js | 37 - .../jade/node_modules/mkdirp/test/mkdirp.js | 28 - .../jade/node_modules/mkdirp/test/perm.js | 32 - .../node_modules/mkdirp/test/perm_sync.js | 39 - .../jade/node_modules/mkdirp/test/race.js | 41 - .../jade/node_modules/mkdirp/test/rel.js | 32 - .../jade/node_modules/mkdirp/test/return.js | 25 - .../node_modules/mkdirp/test/return_sync.js | 24 - .../jade/node_modules/mkdirp/test/root.js | 18 - .../jade/node_modules/mkdirp/test/sync.js | 32 - .../jade/node_modules/mkdirp/test/umask.js | 28 - .../node_modules/mkdirp/test/umask_sync.js | 32 - node_modules/jade/package.json | 50 - node_modules/jade/runtime.js | 179 - node_modules/jade/runtime.min.js | 1 - node_modules/jade/testing/index.html | 5 - node_modules/jade/testing/index.jade | 4 - node_modules/jade/testing/layout.html | 1 - node_modules/jade/testing/layout.jade | 6 - node_modules/jade/testing/mobile.html | 0 node_modules/jade/testing/mobile.jade | 0 .../jade/testing/nested/something.html | 1 - .../jade/testing/nested/something.jade | 1 - node_modules/jade/testing/some.js | 4 - node_modules/jade/testing/test.md | 5 - node_modules/lazy/.npmignore | 1 - node_modules/lazy/README.md | 185 - node_modules/lazy/lazy.js | 349 - node_modules/lazy/lazy.js~ | 348 - node_modules/lazy/package.json | 38 - node_modules/lazy/package.json~ | 32 - node_modules/lazy/test/bucket.js | 37 - node_modules/lazy/test/complex.js | 52 - node_modules/lazy/test/custom.js | 32 - node_modules/lazy/test/em.js | 33 - node_modules/lazy/test/filter.js | 27 - node_modules/lazy/test/foldr.js | 26 - node_modules/lazy/test/forEach.js | 31 - node_modules/lazy/test/head.js | 26 - node_modules/lazy/test/join.js | 26 - node_modules/lazy/test/lines.js | 89 - node_modules/lazy/test/map.js | 29 - node_modules/lazy/test/pipe.js | 38 - node_modules/lazy/test/product.js | 26 - node_modules/lazy/test/range.js | 272 - node_modules/lazy/test/skip.js | 27 - node_modules/lazy/test/sum.js | 26 - node_modules/lazy/test/tail.js | 27 - node_modules/lazy/test/take.js | 26 - node_modules/lazy/test/takeWhile.js | 26 - package.json | 13 +- views/error.jade | 7 +- views/layout.jade | 3 - views/login.jade | 2 +- views/tasks.jade | 59 +- xunlei-lixian/.gitignore | 1 - xunlei-lixian/LICENSE | 23 - xunlei-lixian/README.md | 468 - xunlei-lixian/lixian.py | 1093 -- xunlei-lixian/lixian_alias.py | 20 - xunlei-lixian/lixian_batch.py | 15 - xunlei-lixian/lixian_cli.py | 65 - xunlei-lixian/lixian_cli_parser.py | 178 - xunlei-lixian/lixian_colors.py | 70 - xunlei-lixian/lixian_colors_console.py | 46 - xunlei-lixian/lixian_colors_linux.py | 62 - xunlei-lixian/lixian_colors_win32.py | 201 - xunlei-lixian/lixian_commands/__init__.py | 0 xunlei-lixian/lixian_commands/add.py | 27 - xunlei-lixian/lixian_commands/config.py | 36 - xunlei-lixian/lixian_commands/delete.py | 38 - xunlei-lixian/lixian_commands/download.py | 325 - xunlei-lixian/lixian_commands/help.py | 12 - xunlei-lixian/lixian_commands/info.py | 19 - xunlei-lixian/lixian_commands/list.py | 57 - xunlei-lixian/lixian_commands/login.py | 42 - xunlei-lixian/lixian_commands/logout.py | 18 - xunlei-lixian/lixian_commands/pause.py | 23 - xunlei-lixian/lixian_commands/readd.py | 40 - xunlei-lixian/lixian_commands/rename.py | 19 - xunlei-lixian/lixian_commands/restart.py | 23 - xunlei-lixian/lixian_commands/util.py | 112 - xunlei-lixian/lixian_config.py | 86 - xunlei-lixian/lixian_download_asyn.py | 358 - xunlei-lixian/lixian_download_tools.py | 128 - xunlei-lixian/lixian_encoding.py | 27 - xunlei-lixian/lixian_filter_expr.py | 69 - xunlei-lixian/lixian_hash.py | 90 - xunlei-lixian/lixian_hash_bt.py | 249 - xunlei-lixian/lixian_hash_ed2k.py | 79 - xunlei-lixian/lixian_help.py | 300 - xunlei-lixian/lixian_logging.py | 91 - xunlei-lixian/lixian_nodes.py | 149 - xunlei-lixian/lixian_plugins/__init__.py | 20 - xunlei-lixian/lixian_plugins/api/__init__.py | 68 - .../lixian_plugins/commands/__init__.py | 36 - .../lixian_plugins/commands/aria2.py | 91 - .../lixian_plugins/commands/decode_url.py | 12 - .../lixian_plugins/commands/diagnostics.py | 16 - xunlei-lixian/lixian_plugins/commands/echo.py | 11 - .../commands/export_download_urls.py | 37 - .../lixian_plugins/commands/extend_links.py | 26 - .../lixian_plugins/commands/get_torrent.py | 44 - xunlei-lixian/lixian_plugins/commands/hash.py | 27 - xunlei-lixian/lixian_plugins/commands/kuai.py | 16 - .../lixian_plugins/commands/list_torrent.py | 65 - .../lixian_plugins/commands/speed_test.py | 96 - .../lixian_plugins/filters/__init__.py | 67 - xunlei-lixian/lixian_plugins/filters/date.py | 7 - xunlei-lixian/lixian_plugins/filters/name.py | 7 - xunlei-lixian/lixian_plugins/filters/raw.py | 7 - .../lixian_plugins/filters/regexp.py | 8 - xunlei-lixian/lixian_plugins/filters/size.py | 23 - xunlei-lixian/lixian_plugins/filters/sort.py | 11 - .../lixian_plugins/filters/total_size.py | 27 - .../lixian_plugins/parsers/__init__.py | 69 - xunlei-lixian/lixian_plugins/parsers/icili.py | 19 - xunlei-lixian/lixian_plugins/parsers/kuai.py | 52 - xunlei-lixian/lixian_plugins/parsers/qjwm.py | 22 - .../lixian_plugins/parsers/simplecd.py | 30 - .../lixian_plugins/parsers/verycd.py | 21 - .../lixian_plugins/queries/__init__.py | 2 - .../lixian_plugins/queries/torrentz.py | 5 - xunlei-lixian/lixian_progress.py | 30 - xunlei-lixian/lixian_queries.py | 326 - xunlei-lixian/lixian_query.py | 476 - xunlei-lixian/lixian_url.py | 75 - xunlei-lixian/lixian_util.py | 29 - xunlei-lixian/lixian_verification_code.py | 13 - xunlei-lixian/tests/123.txt | 1 - xunlei-lixian/tests/123456.txt | 1 - ...uick-brown-fox-jumps-over-the-lazy-dog.txt | 1 - xunlei-lixian/tests/a.txt | 1 - xunlei-lixian/tests/abc.txt | 1 - xunlei-lixian/tests/empty.txt | 0 550 files changed, 187 insertions(+), 73215 deletions(-) delete mode 120000 node_modules/.bin/express delete mode 120000 node_modules/.bin/icake delete mode 120000 node_modules/.bin/iced delete mode 120000 node_modules/.bin/jade delete mode 100644 node_modules/express/.npmignore delete mode 100644 node_modules/express/.travis.yml delete mode 100644 node_modules/express/History.md delete mode 100644 node_modules/express/LICENSE delete mode 100644 node_modules/express/Makefile delete mode 100644 node_modules/express/Readme.md delete mode 100755 node_modules/express/bin/express delete mode 100644 node_modules/express/client.js delete mode 100644 node_modules/express/index.js delete mode 100644 node_modules/express/lib/application.js delete mode 100644 node_modules/express/lib/express.js delete mode 100644 node_modules/express/lib/middleware.js delete mode 100644 node_modules/express/lib/request.js delete mode 100644 node_modules/express/lib/response.js delete mode 100644 node_modules/express/lib/router/index.js delete mode 100644 node_modules/express/lib/router/route.js delete mode 100644 node_modules/express/lib/utils.js delete mode 100644 node_modules/express/lib/view.js delete mode 100644 node_modules/express/node_modules/buffer-crc32/.npmignore delete mode 100644 node_modules/express/node_modules/buffer-crc32/.travis.yml delete mode 100644 node_modules/express/node_modules/buffer-crc32/README.md delete mode 100644 node_modules/express/node_modules/buffer-crc32/index.js delete mode 100644 node_modules/express/node_modules/buffer-crc32/package.json delete mode 100644 node_modules/express/node_modules/buffer-crc32/tests/crc.test.js delete mode 100644 node_modules/express/node_modules/commander/.npmignore delete mode 100644 node_modules/express/node_modules/commander/.travis.yml delete mode 100644 node_modules/express/node_modules/commander/History.md delete mode 100644 node_modules/express/node_modules/commander/Makefile delete mode 100644 node_modules/express/node_modules/commander/Readme.md delete mode 100644 node_modules/express/node_modules/commander/index.js delete mode 100644 node_modules/express/node_modules/commander/lib/commander.js delete mode 100644 node_modules/express/node_modules/commander/package.json delete mode 100644 node_modules/express/node_modules/connect/.npmignore delete mode 100644 node_modules/express/node_modules/connect/.travis.yml delete mode 100644 node_modules/express/node_modules/connect/LICENSE delete mode 100644 node_modules/express/node_modules/connect/Readme.md delete mode 100644 node_modules/express/node_modules/connect/index.js delete mode 100644 node_modules/express/node_modules/connect/lib/cache.js delete mode 100644 node_modules/express/node_modules/connect/lib/connect.js delete mode 100644 node_modules/express/node_modules/connect/lib/index.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/basicAuth.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/bodyParser.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/compress.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/cookieParser.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/cookieSession.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/csrf.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/directory.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/errorHandler.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/favicon.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/json.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/limit.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/logger.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/methodOverride.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/multipart.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/query.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/responseTime.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/session.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/session/cookie.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/session/memory.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/session/session.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/session/store.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/static.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/staticCache.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/timeout.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/urlencoded.js delete mode 100644 node_modules/express/node_modules/connect/lib/middleware/vhost.js delete mode 100644 node_modules/express/node_modules/connect/lib/patch.js delete mode 100644 node_modules/express/node_modules/connect/lib/proto.js delete mode 100644 node_modules/express/node_modules/connect/lib/public/directory.html delete mode 100644 node_modules/express/node_modules/connect/lib/public/error.html delete mode 100644 node_modules/express/node_modules/connect/lib/public/favicon.ico delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_add.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_attach.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_code.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_copy.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_delete.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_edit.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_error.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_excel.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_find.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_gear.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_go.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_green.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_key.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_lightning.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_link.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_paintbrush.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_paste.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_red.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_refresh.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_save.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_acrobat.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_actionscript.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_add.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_c.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_camera.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_cd.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_code.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_code_red.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_coldfusion.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_compressed.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_copy.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_cplusplus.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_csharp.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_cup.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_database.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_delete.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_dvd.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_edit.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_error.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_excel.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_find.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_flash.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_freehand.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_gear.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_get.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_go.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_h.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_horizontal.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_key.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_lightning.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_link.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_magnify.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_medal.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_office.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_paint.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_paintbrush.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_paste.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_php.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_picture.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_powerpoint.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_put.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_ruby.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_stack.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_star.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_swoosh.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_text.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_text_width.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_tux.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_vector.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_visualstudio.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_width.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_word.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_world.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_wrench.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_white_zip.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_word.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/icons/page_world.png delete mode 100644 node_modules/express/node_modules/connect/lib/public/style.css delete mode 100644 node_modules/express/node_modules/connect/lib/utils.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/buffer-crc32/.npmignore delete mode 100644 node_modules/express/node_modules/connect/node_modules/buffer-crc32/.travis.yml delete mode 100644 node_modules/express/node_modules/connect/node_modules/buffer-crc32/README.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/buffer-crc32/index.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/buffer-crc32/package.json delete mode 100644 node_modules/express/node_modules/connect/node_modules/buffer-crc32/tests/crc.test.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/bytes/.npmignore delete mode 100644 node_modules/express/node_modules/connect/node_modules/bytes/History.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/bytes/Makefile delete mode 100644 node_modules/express/node_modules/connect/node_modules/bytes/Readme.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/bytes/component.json delete mode 100644 node_modules/express/node_modules/connect/node_modules/bytes/index.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/bytes/package.json delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/.npmignore delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/.travis.yml delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/Makefile delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/Readme.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/TODO delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/benchmark/bench-multipart-parser.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/example/post.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/example/upload.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/index.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/lib/file.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/lib/incoming_form.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/lib/index.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/lib/multipart_parser.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/lib/querystring_parser.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/lib/util.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/Makefile delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/Readme.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/example/dog.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/example/event_emitter.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/index.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/lib/gently/gently.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/lib/gently/index.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/package.json delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/test/common.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/node-gently/test/simple/test-gently.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/package.json delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/common.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/file/funkyfilename.txt delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/file/plain.txt delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/http/special-chars-in-filename/info.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/js/no-filename.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/js/special-chars-in-filename.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/multipart.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/integration/test-fixtures.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/common.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/integration/test-multipart-parser.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-file.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-incoming-form.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-multipart-parser.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-querystring-parser.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/system/test-multi-video-upload.js delete mode 100755 node_modules/express/node_modules/connect/node_modules/formidable/test/run.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/test/unit/test-incoming-form.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/formidable/tool/record.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/pause/.npmignore delete mode 100644 node_modules/express/node_modules/connect/node_modules/pause/History.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/pause/Makefile delete mode 100644 node_modules/express/node_modules/connect/node_modules/pause/Readme.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/pause/index.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/pause/package.json delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/.gitmodules delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/.npmignore delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/.travis.yml delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/History.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/Makefile delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/Readme.md delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/benchmark.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/component.json delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/examples.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/index.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/lib/head.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/lib/querystring.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/lib/tail.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/package.json delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/querystring.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/browser/expect.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/browser/index.html delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/browser/jquery.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/browser/mocha.css delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/browser/mocha.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/browser/qs.css delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/browser/qs.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/parse.js delete mode 100644 node_modules/express/node_modules/connect/node_modules/qs/test/stringify.js delete mode 100644 node_modules/express/node_modules/connect/package.json delete mode 100644 node_modules/express/node_modules/connect/test.js delete mode 100644 node_modules/express/node_modules/cookie-signature/.npmignore delete mode 100644 node_modules/express/node_modules/cookie-signature/History.md delete mode 100644 node_modules/express/node_modules/cookie-signature/Makefile delete mode 100644 node_modules/express/node_modules/cookie-signature/Readme.md delete mode 100644 node_modules/express/node_modules/cookie-signature/index.js delete mode 100644 node_modules/express/node_modules/cookie-signature/package.json delete mode 100644 node_modules/express/node_modules/cookie/.npmignore delete mode 100644 node_modules/express/node_modules/cookie/.travis.yml delete mode 100644 node_modules/express/node_modules/cookie/README.md delete mode 100644 node_modules/express/node_modules/cookie/index.js delete mode 100644 node_modules/express/node_modules/cookie/package.json delete mode 100644 node_modules/express/node_modules/cookie/test/mocha.opts delete mode 100644 node_modules/express/node_modules/cookie/test/parse.js delete mode 100644 node_modules/express/node_modules/cookie/test/serialize.js delete mode 100644 node_modules/express/node_modules/debug/.npmignore delete mode 100644 node_modules/express/node_modules/debug/History.md delete mode 100644 node_modules/express/node_modules/debug/Readme.md delete mode 100644 node_modules/express/node_modules/debug/component.json delete mode 100644 node_modules/express/node_modules/debug/debug.js delete mode 100644 node_modules/express/node_modules/debug/example/app.js delete mode 100644 node_modules/express/node_modules/debug/example/browser.html delete mode 100644 node_modules/express/node_modules/debug/example/wildcards.js delete mode 100644 node_modules/express/node_modules/debug/example/worker.js delete mode 100644 node_modules/express/node_modules/debug/index.js delete mode 100644 node_modules/express/node_modules/debug/lib/debug.js delete mode 100644 node_modules/express/node_modules/debug/package.json delete mode 100644 node_modules/express/node_modules/fresh/.npmignore delete mode 100644 node_modules/express/node_modules/fresh/Makefile delete mode 100644 node_modules/express/node_modules/fresh/Readme.md delete mode 100644 node_modules/express/node_modules/fresh/index.js delete mode 100644 node_modules/express/node_modules/fresh/package.json delete mode 100644 node_modules/express/node_modules/methods/index.js delete mode 100644 node_modules/express/node_modules/methods/package.json delete mode 100644 node_modules/express/node_modules/mkdirp/.npmignore delete mode 100644 node_modules/express/node_modules/mkdirp/.travis.yml delete mode 100644 node_modules/express/node_modules/mkdirp/LICENSE delete mode 100644 node_modules/express/node_modules/mkdirp/examples/pow.js delete mode 100644 node_modules/express/node_modules/mkdirp/index.js delete mode 100644 node_modules/express/node_modules/mkdirp/package.json delete mode 100644 node_modules/express/node_modules/mkdirp/readme.markdown delete mode 100644 node_modules/express/node_modules/mkdirp/test/chmod.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/clobber.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/mkdirp.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/perm.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/perm_sync.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/race.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/rel.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/return.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/return_sync.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/root.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/sync.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/umask.js delete mode 100644 node_modules/express/node_modules/mkdirp/test/umask_sync.js delete mode 100644 node_modules/express/node_modules/range-parser/.npmignore delete mode 100644 node_modules/express/node_modules/range-parser/History.md delete mode 100644 node_modules/express/node_modules/range-parser/Makefile delete mode 100644 node_modules/express/node_modules/range-parser/Readme.md delete mode 100644 node_modules/express/node_modules/range-parser/index.js delete mode 100644 node_modules/express/node_modules/range-parser/package.json delete mode 100644 node_modules/express/node_modules/send/.npmignore delete mode 100644 node_modules/express/node_modules/send/History.md delete mode 100644 node_modules/express/node_modules/send/Makefile delete mode 100644 node_modules/express/node_modules/send/Readme.md delete mode 100644 node_modules/express/node_modules/send/index.js delete mode 100644 node_modules/express/node_modules/send/lib/send.js delete mode 100644 node_modules/express/node_modules/send/lib/utils.js delete mode 100644 node_modules/express/node_modules/send/node_modules/mime/LICENSE delete mode 100644 node_modules/express/node_modules/send/node_modules/mime/README.md delete mode 100644 node_modules/express/node_modules/send/node_modules/mime/mime.js delete mode 100644 node_modules/express/node_modules/send/node_modules/mime/package.json delete mode 100644 node_modules/express/node_modules/send/node_modules/mime/test.js delete mode 100644 node_modules/express/node_modules/send/node_modules/mime/types/mime.types delete mode 100644 node_modules/express/node_modules/send/node_modules/mime/types/node.types delete mode 100644 node_modules/express/node_modules/send/package.json delete mode 100644 node_modules/express/package.json delete mode 100644 node_modules/express/test.js delete mode 100644 node_modules/iced-coffee-script/.npmignore delete mode 100644 node_modules/iced-coffee-script/CNAME delete mode 100644 node_modules/iced-coffee-script/CONTRIBUTING.md delete mode 100644 node_modules/iced-coffee-script/LICENSE delete mode 100644 node_modules/iced-coffee-script/README delete mode 100644 node_modules/iced-coffee-script/Rakefile delete mode 100644 node_modules/iced-coffee-script/UPDATING.md delete mode 100755 node_modules/iced-coffee-script/bin/cake delete mode 100755 node_modules/iced-coffee-script/bin/coffee delete mode 100644 node_modules/iced-coffee-script/extras/coffee-script-iced-large.js delete mode 100644 node_modules/iced-coffee-script/extras/coffee-script-iced.js delete mode 100644 node_modules/iced-coffee-script/iced.md delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/browser.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/cake.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/coffee-script.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/command.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/grammar.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/helpers.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/iced.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/icedlib.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/index.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/lexer.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/nodes.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/optparse.js delete mode 100755 node_modules/iced-coffee-script/lib/coffee-script/parser.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/repl.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/rewriter.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/scope.js delete mode 100644 node_modules/iced-coffee-script/lib/coffee-script/sourcemap.js delete mode 100644 node_modules/iced-coffee-script/media/detail.png delete mode 100644 node_modules/iced-coffee-script/media/post-rotate.png delete mode 100644 node_modules/iced-coffee-script/media/rotate.graffle delete mode 100644 node_modules/iced-coffee-script/media/rotate1.png delete mode 100644 node_modules/iced-coffee-script/media/rotate2.png delete mode 100644 node_modules/iced-coffee-script/media/rotate3.png delete mode 100644 node_modules/iced-coffee-script/media/rotate4.png delete mode 100644 node_modules/iced-coffee-script/media/rotate5.png delete mode 100644 node_modules/iced-coffee-script/media/sketch.pdf delete mode 100644 node_modules/iced-coffee-script/package.json delete mode 100644 node_modules/jade/.npmignore delete mode 100644 node_modules/jade/LICENSE delete mode 100644 node_modules/jade/Readme.md delete mode 100644 node_modules/jade/Readme_zh-cn.md delete mode 100755 node_modules/jade/bin/jade delete mode 100644 node_modules/jade/index.js delete mode 100644 node_modules/jade/jade.js delete mode 100644 node_modules/jade/jade.md delete mode 100644 node_modules/jade/jade.min.js delete mode 100644 node_modules/jade/lib/compiler.js delete mode 100644 node_modules/jade/lib/doctypes.js delete mode 100644 node_modules/jade/lib/filters.js delete mode 100644 node_modules/jade/lib/inline-tags.js delete mode 100644 node_modules/jade/lib/jade.js delete mode 100644 node_modules/jade/lib/lexer.js delete mode 100644 node_modules/jade/lib/nodes/attrs.js delete mode 100644 node_modules/jade/lib/nodes/block-comment.js delete mode 100644 node_modules/jade/lib/nodes/block.js delete mode 100644 node_modules/jade/lib/nodes/case.js delete mode 100644 node_modules/jade/lib/nodes/code.js delete mode 100644 node_modules/jade/lib/nodes/comment.js delete mode 100644 node_modules/jade/lib/nodes/doctype.js delete mode 100644 node_modules/jade/lib/nodes/each.js delete mode 100644 node_modules/jade/lib/nodes/filter.js delete mode 100644 node_modules/jade/lib/nodes/index.js delete mode 100644 node_modules/jade/lib/nodes/literal.js delete mode 100644 node_modules/jade/lib/nodes/mixin.js delete mode 100644 node_modules/jade/lib/nodes/node.js delete mode 100644 node_modules/jade/lib/nodes/tag.js delete mode 100644 node_modules/jade/lib/nodes/text.js delete mode 100644 node_modules/jade/lib/parser.js delete mode 100644 node_modules/jade/lib/runtime.js delete mode 100644 node_modules/jade/lib/self-closing.js delete mode 100644 node_modules/jade/lib/utils.js delete mode 100644 node_modules/jade/node_modules/commander/.npmignore delete mode 100644 node_modules/jade/node_modules/commander/.travis.yml delete mode 100644 node_modules/jade/node_modules/commander/History.md delete mode 100644 node_modules/jade/node_modules/commander/Makefile delete mode 100644 node_modules/jade/node_modules/commander/Readme.md delete mode 100644 node_modules/jade/node_modules/commander/index.js delete mode 100644 node_modules/jade/node_modules/commander/lib/commander.js delete mode 100644 node_modules/jade/node_modules/commander/package.json delete mode 100644 node_modules/jade/node_modules/mkdirp/.npmignore delete mode 100644 node_modules/jade/node_modules/mkdirp/.travis.yml delete mode 100644 node_modules/jade/node_modules/mkdirp/LICENSE delete mode 100644 node_modules/jade/node_modules/mkdirp/examples/pow.js delete mode 100644 node_modules/jade/node_modules/mkdirp/index.js delete mode 100644 node_modules/jade/node_modules/mkdirp/package.json delete mode 100644 node_modules/jade/node_modules/mkdirp/readme.markdown delete mode 100644 node_modules/jade/node_modules/mkdirp/test/chmod.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/clobber.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/mkdirp.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/perm.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/perm_sync.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/race.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/rel.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/return.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/return_sync.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/root.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/sync.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/umask.js delete mode 100644 node_modules/jade/node_modules/mkdirp/test/umask_sync.js delete mode 100644 node_modules/jade/package.json delete mode 100644 node_modules/jade/runtime.js delete mode 100644 node_modules/jade/runtime.min.js delete mode 100644 node_modules/jade/testing/index.html delete mode 100644 node_modules/jade/testing/index.jade delete mode 100644 node_modules/jade/testing/layout.html delete mode 100644 node_modules/jade/testing/layout.jade delete mode 100644 node_modules/jade/testing/mobile.html delete mode 100644 node_modules/jade/testing/mobile.jade delete mode 100644 node_modules/jade/testing/nested/something.html delete mode 100644 node_modules/jade/testing/nested/something.jade delete mode 100644 node_modules/jade/testing/some.js delete mode 100644 node_modules/jade/testing/test.md delete mode 100644 node_modules/lazy/.npmignore delete mode 100644 node_modules/lazy/README.md delete mode 100644 node_modules/lazy/lazy.js delete mode 100644 node_modules/lazy/lazy.js~ delete mode 100644 node_modules/lazy/package.json delete mode 100644 node_modules/lazy/package.json~ delete mode 100644 node_modules/lazy/test/bucket.js delete mode 100644 node_modules/lazy/test/complex.js delete mode 100644 node_modules/lazy/test/custom.js delete mode 100644 node_modules/lazy/test/em.js delete mode 100644 node_modules/lazy/test/filter.js delete mode 100644 node_modules/lazy/test/foldr.js delete mode 100644 node_modules/lazy/test/forEach.js delete mode 100644 node_modules/lazy/test/head.js delete mode 100644 node_modules/lazy/test/join.js delete mode 100644 node_modules/lazy/test/lines.js delete mode 100644 node_modules/lazy/test/map.js delete mode 100644 node_modules/lazy/test/pipe.js delete mode 100644 node_modules/lazy/test/product.js delete mode 100644 node_modules/lazy/test/range.js delete mode 100644 node_modules/lazy/test/skip.js delete mode 100644 node_modules/lazy/test/sum.js delete mode 100644 node_modules/lazy/test/tail.js delete mode 100644 node_modules/lazy/test/take.js delete mode 100644 node_modules/lazy/test/takeWhile.js delete mode 100644 xunlei-lixian/.gitignore delete mode 100644 xunlei-lixian/LICENSE delete mode 100644 xunlei-lixian/README.md delete mode 100644 xunlei-lixian/lixian.py delete mode 100644 xunlei-lixian/lixian_alias.py delete mode 100755 xunlei-lixian/lixian_batch.py delete mode 100755 xunlei-lixian/lixian_cli.py delete mode 100644 xunlei-lixian/lixian_cli_parser.py delete mode 100644 xunlei-lixian/lixian_colors.py delete mode 100644 xunlei-lixian/lixian_colors_console.py delete mode 100644 xunlei-lixian/lixian_colors_linux.py delete mode 100644 xunlei-lixian/lixian_colors_win32.py delete mode 100644 xunlei-lixian/lixian_commands/__init__.py delete mode 100644 xunlei-lixian/lixian_commands/add.py delete mode 100644 xunlei-lixian/lixian_commands/config.py delete mode 100644 xunlei-lixian/lixian_commands/delete.py delete mode 100644 xunlei-lixian/lixian_commands/download.py delete mode 100644 xunlei-lixian/lixian_commands/help.py delete mode 100644 xunlei-lixian/lixian_commands/info.py delete mode 100644 xunlei-lixian/lixian_commands/list.py delete mode 100644 xunlei-lixian/lixian_commands/login.py delete mode 100644 xunlei-lixian/lixian_commands/logout.py delete mode 100644 xunlei-lixian/lixian_commands/pause.py delete mode 100644 xunlei-lixian/lixian_commands/readd.py delete mode 100644 xunlei-lixian/lixian_commands/rename.py delete mode 100644 xunlei-lixian/lixian_commands/restart.py delete mode 100644 xunlei-lixian/lixian_commands/util.py delete mode 100644 xunlei-lixian/lixian_config.py delete mode 100644 xunlei-lixian/lixian_download_asyn.py delete mode 100644 xunlei-lixian/lixian_download_tools.py delete mode 100644 xunlei-lixian/lixian_encoding.py delete mode 100644 xunlei-lixian/lixian_filter_expr.py delete mode 100755 xunlei-lixian/lixian_hash.py delete mode 100644 xunlei-lixian/lixian_hash_bt.py delete mode 100644 xunlei-lixian/lixian_hash_ed2k.py delete mode 100644 xunlei-lixian/lixian_help.py delete mode 100644 xunlei-lixian/lixian_logging.py delete mode 100644 xunlei-lixian/lixian_nodes.py delete mode 100644 xunlei-lixian/lixian_plugins/__init__.py delete mode 100644 xunlei-lixian/lixian_plugins/api/__init__.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/__init__.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/aria2.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/decode_url.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/diagnostics.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/echo.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/export_download_urls.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/extend_links.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/get_torrent.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/hash.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/kuai.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/list_torrent.py delete mode 100644 xunlei-lixian/lixian_plugins/commands/speed_test.py delete mode 100644 xunlei-lixian/lixian_plugins/filters/__init__.py delete mode 100644 xunlei-lixian/lixian_plugins/filters/date.py delete mode 100644 xunlei-lixian/lixian_plugins/filters/name.py delete mode 100644 xunlei-lixian/lixian_plugins/filters/raw.py delete mode 100644 xunlei-lixian/lixian_plugins/filters/regexp.py delete mode 100644 xunlei-lixian/lixian_plugins/filters/size.py delete mode 100644 xunlei-lixian/lixian_plugins/filters/sort.py delete mode 100644 xunlei-lixian/lixian_plugins/filters/total_size.py delete mode 100644 xunlei-lixian/lixian_plugins/parsers/__init__.py delete mode 100644 xunlei-lixian/lixian_plugins/parsers/icili.py delete mode 100644 xunlei-lixian/lixian_plugins/parsers/kuai.py delete mode 100644 xunlei-lixian/lixian_plugins/parsers/qjwm.py delete mode 100644 xunlei-lixian/lixian_plugins/parsers/simplecd.py delete mode 100644 xunlei-lixian/lixian_plugins/parsers/verycd.py delete mode 100644 xunlei-lixian/lixian_plugins/queries/__init__.py delete mode 100644 xunlei-lixian/lixian_plugins/queries/torrentz.py delete mode 100644 xunlei-lixian/lixian_progress.py delete mode 100644 xunlei-lixian/lixian_queries.py delete mode 100644 xunlei-lixian/lixian_query.py delete mode 100644 xunlei-lixian/lixian_url.py delete mode 100644 xunlei-lixian/lixian_util.py delete mode 100644 xunlei-lixian/lixian_verification_code.py delete mode 100644 xunlei-lixian/tests/123.txt delete mode 100644 xunlei-lixian/tests/123456.txt delete mode 100644 xunlei-lixian/tests/The-quick-brown-fox-jumps-over-the-lazy-dog.txt delete mode 100644 xunlei-lixian/tests/a.txt delete mode 100644 xunlei-lixian/tests/abc.txt delete mode 100644 xunlei-lixian/tests/empty.txt diff --git a/.gitignore b/.gitignore index 714cebf..dbfd766 100644 --- a/.gitignore +++ b/.gitignore @@ -11,3 +11,5 @@ pids logs results npm-debug.log +node_modules +.lixian-portal.* diff --git a/.npmignore b/.npmignore index 07e6e47..53239ed 100644 --- a/.npmignore +++ b/.npmignore @@ -1 +1,2 @@ -/node_modules +node_modules +.lixian-portal.* diff --git a/README.md b/README.md index 1b45b81..6add88f 100644 --- a/README.md +++ b/README.md @@ -1,7 +1,7 @@ lixian-portal ============= -给`iambus/xunlei-lixian`做的一个简洁实用的webui。(不明真相的同学赶快先去膜拜了[iambus/xunlei-lixian](https://github.com/iambus/xunlei-lixian)了再回来) +一个简洁实用的 Web 版迅雷离线下载程序。 # 这是啥 @@ -9,9 +9,9 @@ lixian-portal # 典型使用场景 -1. 家里有个连着移动硬盘的树莓派 +1. 家里有个 HTPC,运行着这个程序 2. 我平常刷微博时发现先几个好看的电影,和想玩的游戏,然后再xxx上找到这些电影和游戏的ed2k链接,然后输入进去 -3. 周末我通过smb文件共享打开树莓派里已经下好的电影和游戏,看之且玩之 +3. 周末我通过 samba 文件共享打开这个程序已经下好的电影和游戏,看之且玩之 4. 室友也可以看(如果有室友的话) # 界面预览 @@ -22,18 +22,15 @@ lixian-portal # 环境 -* linux/osx (如果你想用windows来做下载服务器,你得先在windows上安装好wget) -* python2 -* nodejs +* [NodeJS](http://nodejs.org/) # 安装方法 -* 下载代码并解压缩 -* 运行命令启动`node /path/to/lixian-portal` +* 使用 NodeJS 自带的包管理器 npm 来安装该程序:`npm install lixian-portal -g` +* 运行命令启动:`lixian-portal` * 下载的位置为启动这个程序的目录(Current Working Directory) -* 如需下载到其他位置,可以设置环境变量`LIXIAN_PORTAL_HOME`,例如:可以这样启动程序`LIXIAN_PORTAL_HOME=/mnt/sdb1 node /path/to/lixan-portal` +* 如需下载到其他位置,可以设置环境变量`LIXIAN_PORTAL_HOME`,例如:可以这样启动程序`LIXIAN_PORTAL_HOME=/mnt/sdb1 lixian-portal` # Tricks -* 如果想让它一直在后运行,可以使用这个命令启动`nohup node /path/to/lixian-portal &` -* `lixian-portal`兼容`xunlei-lixian`的设置,按这样的命令格式设置即可`HOME=/path/to/lixian-portal/Downloads lx config output-dir /mnt/Downloads` +* 如果想让它一直在后运行,可以使用这个命令启动`nohup lixian-portal &` diff --git a/app.iced b/app.iced index 6f6af32..fe60b57 100644 --- a/app.iced +++ b/app.iced @@ -23,13 +23,6 @@ app.use express.methodOverride() client.startCron() queue = client.queue -autorefresh = -> - queue.append - name: "刷新任务列表" - func: queue.tasks.updateTasklist - setTimeout autorefresh, 60000 * (1 + Math.random() * 3) -autorefresh() - app.get '/', (req, res, n)-> return res.redirect '/login' if client.stats.requireLogin @@ -43,22 +36,16 @@ app.all '*', (req, res, n)-> ip = ip.split(',')[0].trim() return n 403 if process.env.ONLYFROM && -1 == process.env.ONLYFROM.indexOf ip n null -app.post '/refresh', (req, res, n)-> - - queue.append - name: "刷新任务列表" - func: queue.tasks.updateTasklist - res.redirect 'back' app.post '/', (req, res, n)-> if req.files && req.files.bt && req.files.bt.path && req.files.bt.length bt = req.files.bt await fs.rename bt.path, "#{bt.path}.torrent", defer e return n e if e - await queue.tasks.addBtTask bt.name, "#{bt.path}.torrent", defer e + await queue.execute 'addBtTask', bt.name, "#{bt.path}.torrent", defer e return n e if e - else - await queue.tasks.addTask req.body.url, defer e + if req.body.url && req.body.url.length + await queue.execute 'addTask', req.body.url, defer e return n e if e res.redirect '/' @@ -66,21 +53,17 @@ app.get '/login', (req, res)-> res.locals.vcode = null res.render 'login' app.post '/login', (req, res, n)-> - await queue.tasks.login req.body.username, req.body.password, req.body.vcode, defer e + await queue.execute 'login', req.body.username, req.body.password, defer e return n e if e res.redirect '/' app.get '/logout', (req, res, n)-> - await queue.tasks.logout defer e + await queue.execute 'logout', defer e return n e if e res.redirect '/' app.delete '/tasks/:id', (req, res, n)-> - if client.stats.retrieving?.task.id - client.stats.retrieving.kill() - queue.append - name: "删除任务 #{task.id}" - func: (fcb)-> - queue.tasks.deleteTask req.params.id, fcb + await queue.execute 'deleteTask', req.params.id, defer e + return cb e if e res.redirect '/' @@ -90,6 +73,11 @@ app.use (e, req, res, next)-> await client.init defer e throw e if e +autorefresh = -> + await queue.execute 'updateTasklist', defer(e) + setTimeout autorefresh, 60000 * (1 + Math.random() * 3) +autorefresh() + port = process.env.PORT || 3000 if isNaN Number port await exec "fuser #{port}", defer e diff --git a/client.iced b/client.iced index f45c0e4..e01c4e2 100644 --- a/client.iced +++ b/client.iced @@ -1,29 +1,12 @@ path = require 'path' fs = require 'fs' -lazy = require 'lazy' - -cli = path.join __dirname, 'xunlei-lixian', 'lixian_cli.py' - -{ - exec - execFile - spawn -} = require 'child_process' - -statusMap = - completed: 'success' - failed: 'error' - waiting: 'warn' - downloading: 'info' -statusMapLabel = - completed: '完成' - failed: '失败' - waiting: '等待' - downloading: '下载中' - -regexMG = /^([^ ]+) +(.+) +(completed|downloading|waiting|failed) *(http\:\/\/.+)?$/mg -regexQ = /^([^ ]+) +(.+) +(completed|downloading|waiting|failed) *(http\:\/\/.+)?$/m +mkdirp = require 'mkdirp' +request = require 'request' +StatusBar = require 'status-bar' + +Lixian = require 'node-lixian' +lixian = new Lixian() exports.stats = stats = task: null @@ -39,181 +22,146 @@ exports.stats = stats = exports.queue = queue = [] exports.log = log = [] -workingDirectory = process.env.LIXIAN_PORTAL_HOME || process.cwd() -process.env.HOME = workingDirectory +cwd = process.env.LIXIAN_PORTAL_HOME || process.cwd() -queue.append = (task)-> - @push task unless (@filter (t)->t.name==task.name).length -queue.prepend = (task)-> - @unshift task unless (@filter (t)->t.name==task.name).length +retrieves = [] exports.startCron = -> while true - if queue.length - stats.task = queue.shift() - log.unshift "#{stats.task.name} 启动" - console.log log[log.length - 1] - await stats.task.func defer e - log.unshift "#{stats.task.name} 完成" - console.log log[log.length - 1] - if e - log.unshift e.message - console.error e.message - + if retrieve = retrieves.shift() + await queue.execute 'retrieve', retrieve.task, retrieve.file, defer e + stats.retrieving = null await setTimeout defer(), 100 exports.init = (cb)-> + await lixian.init {}, defer e + return cb e if e + await fs.readFile (path.join cwd, '.lixian-portal.username'), 'utf8', defer e, username + await fs.readFile (path.join cwd, '.lixian-portal.password'), 'utf8', defer e, password + if username && password + console.log '正在尝试自动登录...' + await queue.execute 'login', username, password, defer e + if e + console.error e.message + else + console.log '自动登录成功.' cb null -getPythonBin = (cb)-> - await exec 'which python2', cwd: workingDirectory, defer e - return cb null, 'python2' unless e - await exec 'python --version', cwd: workingDirectory, defer e, out, err - return cb e if e - return cb new Error "invalid Python version: #{err}. (Python 2.x needed)" unless err.match /Python[\s]+2\./ - return cb null, 'python' +stats.executings = [] +queue.execute = (command, args..., cb)=> + commands = + retrieve: (task, file)-> "取回文件 #{task.name}/#{file.name}" + deleteTask: (id)-> "删除任务 #{id}" + updateTasklist: -> "刷新任务列表" + addTask: (url)-> "添加任务 #{url}" + addBtTask: (filename, torrent)-> "添加 BT 任务 #{filename}" + login: (username, password)-> "以 #{username} 登录" + logout: -> "登出" + command_name = commands[command] args... + log.unshift "#{command_name} 启动" + console.log log[0] + stats.executings.push command_name + await queue.tasks[command] args..., defer e, results... + stats.executings.splice (stats.executings.indexOf command_name), 1 + if e + log.unshift e.message + console.log log[0] + log.unshift "#{command_name} 失败" + console.log log[0] + else + log.unshift "#{command_name} 完成" + console.log log[0] + cb e, results... + + queue.tasks = - retrieve: (task, cb)-> - await getPythonBin defer e, pyothon_bin + retrieve: (task, file, cb)-> + await mkdirp (path.join cwd, task.name), defer e return cb e if e - stats.retrieving = spawn pyothon_bin, [cli, 'download', '--continue', '--no-hash', task.id], stdio: 'pipe', cwd: workingDirectory - errBuffer = [] - stats.retrieving.task = task - new lazy(stats.retrieving.stderr).lines.forEach (line)-> - line ?= [] - line = line.toString 'utf8' - errBuffer.push line - line = line.match /\s+(\d?\d%)\s+([^ ]{1,10})\s+([^ ]{1,10})\r?\n?$/ - [dummy, stats.progress, stats.speed, stats.time] = line if line - - await stats.retrieving.on 'exit', defer e - if e - stats.error[task.id] = errBuffer.join '' - stats.retrieving = null - queue.append - name: "刷新任务列表" - func: queue.tasks.updateTasklist - queue.append - name: "删除任务 #{task.id}" - func: (fcb)-> - queue.tasks.deleteTask task.id, fcb + req = request + url: file.url + headers: + 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8' + 'Cookie': stats.cookie + 'Referer': 'http://dynamic.cloud.vip.xunlei.com/user_task' + 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_9_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/37.0.2062.124 Safari/537.36' + 'Accept-Language': 'zh-CN,zh;q=0.8,it-IT;q=0.6,it;q=0.4,en-US;q=0.2,en;q=0.2' + proxy: process.env['http_proxy'] + writer = fs.createWriteStream path.join cwd, task.name, file.name + await req.on 'response', defer res + fileSize = Number res.headers['content-length'] + return cb new Error "Invalid Content-Length" if isNaN fileSize + statusBar = StatusBar.create total: fileSize + statusBar.on 'render', (progress)-> + stats.retrieving = + req: req + progress: progress + task: task + file: file + format: statusBar.format + req.pipe statusBar + req.pipe writer + await writer.on 'close', defer() + statusBar.cancel() cb() - updateTasklist: (cb)-> - await getPythonBin defer e, pyothon_bin - return cb e if e - await exec "#{pyothon_bin} #{cli} config encoding utf-8", cwd: workingDirectory, defer e + await lixian.list {}, defer e, data return cb e if e - await exec "#{pyothon_bin} #{cli} list --no-colors", cwd: workingDirectory, defer e, out, err - if e && err.match /user is not logged in|Verification code required/ - stats.requireLogin = true - return cb e - return cb e if e - _tasks = [] - if out.match regexMG - for task in out.match regexMG - task = task.match regexQ - _tasks.push - id: task[1] - filename: task[2] - status: statusMap[task[3]] - statusLabel: statusMapLabel[task[3]] - - stats.tasks = _tasks - - for task in _tasks - if task.status=='success' && !stats.error[task.id]? - queue.append - name: "取回 #{task.id}" - func: (fcb)-> - queue.tasks.retrieve task, fcb + stats.cookie = data.cookie + stats.tasks = data.tasks + for task in stats.tasks + for file in task.files + file.status = 'warning' + file.statusLabel = '未就绪' + if file.url + file.status = 'success' + file.statusLabel = '就绪' + unless retrieves.filter((r)-> r.task.id == task.id && r.file.name == file.name).length + retrieves.push + task: task + file: file cb() deleteTask: (id, cb)-> - await getPythonBin defer e, pyothon_bin + if stats.retrieving?.task.id == id + stats.retrieving.req.abort() + retrieves = retrieves.filter (retrieve)-> retrieve.task.id != id + + await lixian.delete_task delete: id, defer e return cb e if e - await exec "#{pyothon_bin} #{cli} delete #{id}", cwd: workingDirectory, defer e, out, err + await queue.execute 'updateTasklist', defer e return cb e if e - queue.append - name: "刷新任务列表" - func: queue.tasks.updateTasklist cb null - login: (username, password, vcode, cb)-> - await getPythonBin defer e, pyothon_bin + login: (username, password, cb)-> + await lixian.login username: username, password: password, defer e return cb e if e - vcode_path = (path.join workingDirectory, ".lixian-portal-vcode.jpg") - if vcode and stats.login_process? - console.log "vcode: #{vcode}" - stats.login_process.stdin.write vcode - stats.login_process.stdin.write "\n" - await stats.login_process.on 'exit', defer code - @login_result = code - else - await fs.unlink vcode_path, defer e - @login_output = "" - stats.login_process = spawn pyothon_bin, [ - cli - "login" - username - password - "--verification-code-path" - vcode_path - ], cwd: workingDirectory - stats.login_process.on 'exit', (code)=> - @login_result = code - console.log "login exited #{@login_result}" - stats.login_process = null - stats.login_process.stdout.on 'data', (data)=> @login_output+= data.toString 'utf8' - stats.login_process.stderr.on 'data', (data)=> @login_output+= data.toString 'utf8' - stats.login_process.stdin.setEncoding 'utf8' - lixian_code_fetched = false - while stats.login_process && !lixian_code_fetched - await setTimeout defer(), 100 - await fs.exists vcode_path, defer lixian_code_fetched - if lixian_code_fetched - e = true - while e - await setTimeout defer(), 100 - await fs.readFile vcode_path, 'base64', defer e, stats.requireVerificationCode - return cb null - if @login_result == 0 - stats.requireLogin = false - queue.append - name: "刷新任务列表" - func: queue.tasks.updateTasklist - cb null - else - console.error @login_output - @login_output = @login_output.match(/^[\w]+\:\s(.*)$/m)?[1] - @login_output = "用户名、密码或者验证码错误" if @login_output?.match /login failed/ - cb new Error "登录失败: \n\n#{@login_output}" + await queue.execute 'updateTasklist', defer e + return cb e if e + await fs.writeFile (path.join cwd, '.lixian-portal.username'), username, 'utf8', defer e + await fs.writeFile (path.join cwd, '.lixian-portal.password'), password, 'utf8', defer e + stats.requireLogin = false + cb null logout: (cb)-> - await getPythonBin defer e, pyothon_bin - return cb e if e - await exec "#{pyothon_bin} #{cli} logout", cwd: workingDirectory, defer e, out, err - await fs.unlink path.join(workingDirectory, '.xunlei.lixian.cookies'), defer e + await fs.unlink (path.join cwd, '.lixian-portal.username'), defer e + await fs.unlink (path.join cwd, '.lixian-portal.password'), defer e stats.requireLogin = true cb null addBtTask: (filename, torrent, cb)-> - await getPythonBin defer e, pyothon_bin + await lixian.add_torrent torrent: torrent, defer e return cb e if e - await exec "#{pyothon_bin} #{cli} add #{torrent}", cwd: workingDirectory, defer e, out, err - return cb e if e - await queue.tasks.updateTasklist defer e + await queue.execute 'updateTasklist', defer e return cb e if e cb null addTask: (url, cb)-> - await getPythonBin defer e, pyothon_bin - return cb e if e - await exec "#{pyothon_bin} #{cli} add \"#{url}\"", cwd: workingDirectory, defer e, out, err + await lixian.add_url url: url, defer e return cb e if e - await queue.tasks.updateTasklist defer e + await queue.execute 'updateTasklist', defer e return cb e if e cb null diff --git a/node_modules/.bin/express b/node_modules/.bin/express deleted file mode 120000 index b741d99..0000000 --- a/node_modules/.bin/express +++ /dev/null @@ -1 +0,0 @@ -../express/bin/express \ No newline at end of file diff --git a/node_modules/.bin/icake b/node_modules/.bin/icake deleted file mode 120000 index 9f13a4d..0000000 --- a/node_modules/.bin/icake +++ /dev/null @@ -1 +0,0 @@ -../iced-coffee-script/bin/cake \ No newline at end of file diff --git a/node_modules/.bin/iced b/node_modules/.bin/iced deleted file mode 120000 index 101c253..0000000 --- a/node_modules/.bin/iced +++ /dev/null @@ -1 +0,0 @@ -../iced-coffee-script/bin/coffee \ No newline at end of file diff --git a/node_modules/.bin/jade b/node_modules/.bin/jade deleted file mode 120000 index 571fae7..0000000 --- a/node_modules/.bin/jade +++ /dev/null @@ -1 +0,0 @@ -../jade/bin/jade \ No newline at end of file diff --git a/node_modules/express/.npmignore b/node_modules/express/.npmignore deleted file mode 100644 index caf574d..0000000 --- a/node_modules/express/.npmignore +++ /dev/null @@ -1,9 +0,0 @@ -.git* -docs/ -examples/ -support/ -test/ -testing.js -.DS_Store -coverage.html -lib-cov diff --git a/node_modules/express/.travis.yml b/node_modules/express/.travis.yml deleted file mode 100644 index cc4dba2..0000000 --- a/node_modules/express/.travis.yml +++ /dev/null @@ -1,4 +0,0 @@ -language: node_js -node_js: - - "0.8" - - "0.10" diff --git a/node_modules/express/History.md b/node_modules/express/History.md deleted file mode 100644 index 4885f6c..0000000 --- a/node_modules/express/History.md +++ /dev/null @@ -1,1106 +0,0 @@ - -3.1.2 / 2013-04-12 -================== - - * add support for custom Accept parameters - * update cookie-signature - -3.1.1 / 2013-04-01 -================== - - * add X-Forwarded-Host support to `req.host` - * fix relative redirects - * update mkdirp - * update buffer-crc32 - * remove legacy app.configure() method from app template. - -3.1.0 / 2013-01-25 -================== - - * add support for leading "." in "view engine" setting - * add array support to `res.set()` - * add node 0.8.x to travis.yml - * add "subdomain offset" setting for tweaking `req.subdomains` - * add `res.location(url)` implementing `res.redirect()`-like setting of Location - * use app.get() for x-powered-by setting for inheritance - * fix colons in passwords for `req.auth` - -3.0.6 / 2013-01-04 -================== - - * add http verb methods to Router - * update connect - * fix mangling of the `res.cookie()` options object - * fix jsonp whitespace escape. Closes #1132 - -3.0.5 / 2012-12-19 -================== - - * add throwing when a non-function is passed to a route - * fix: explicitly remove Transfer-Encoding header from 204 and 304 responses - * revert "add 'etag' option" - -3.0.4 / 2012-12-05 -================== - - * add 'etag' option to disable `res.send()` Etags - * add escaping of urls in text/plain in `res.redirect()` - for old browsers interpreting as html - * change crc32 module for a more liberal license - * update connect - -3.0.3 / 2012-11-13 -================== - - * update connect - * update cookie module - * fix cookie max-age - -3.0.2 / 2012-11-08 -================== - - * add OPTIONS to cors example. Closes #1398 - * fix route chaining regression. Closes #1397 - -3.0.1 / 2012-11-01 -================== - - * update connect - -3.0.0 / 2012-10-23 -================== - - * add `make clean` - * add "Basic" check to req.auth - * add `req.auth` test coverage - * add cb && cb(payload) to `res.jsonp()`. Closes #1374 - * add backwards compat for `res.redirect()` status. Closes #1336 - * add support for `res.json()` to retain previously defined Content-Types. Closes #1349 - * update connect - * change `res.redirect()` to utilize a pathname-relative Location again. Closes #1382 - * remove non-primitive string support for `res.send()` - * fix view-locals example. Closes #1370 - * fix route-separation example - -3.0.0rc5 / 2012-09-18 -================== - - * update connect - * add redis search example - * add static-files example - * add "x-powered-by" setting (`app.disable('x-powered-by')`) - * add "application/octet-stream" redirect Accept test case. Closes #1317 - -3.0.0rc4 / 2012-08-30 -================== - - * add `res.jsonp()`. Closes #1307 - * add "verbose errors" option to error-pages example - * add another route example to express(1) so people are not so confused - * add redis online user activity tracking example - * update connect dep - * fix etag quoting. Closes #1310 - * fix error-pages 404 status - * fix jsonp callback char restrictions - * remove old OPTIONS default response - -3.0.0rc3 / 2012-08-13 -================== - - * update connect dep - * fix signed cookies to work with `connect.cookieParser()` ("s:" prefix was missing) [tnydwrds] - * fix `res.render()` clobbering of "locals" - -3.0.0rc2 / 2012-08-03 -================== - - * add CORS example - * update connect dep - * deprecate `.createServer()` & remove old stale examples - * fix: escape `res.redirect()` link - * fix vhost example - -3.0.0rc1 / 2012-07-24 -================== - - * add more examples to view-locals - * add scheme-relative redirects (`res.redirect("//foo.com")`) support - * update cookie dep - * update connect dep - * update send dep - * fix `express(1)` -h flag, use -H for hogan. Closes #1245 - * fix `res.sendfile()` socket error handling regression - -3.0.0beta7 / 2012-07-16 -================== - - * update connect dep for `send()` root normalization regression - -3.0.0beta6 / 2012-07-13 -================== - - * add `err.view` property for view errors. Closes #1226 - * add "jsonp callback name" setting - * add support for "/foo/:bar*" non-greedy matches - * change `res.sendfile()` to use `send()` module - * change `res.send` to use "response-send" module - * remove `app.locals.use` and `res.locals.use`, use regular middleware - -3.0.0beta5 / 2012-07-03 -================== - - * add "make check" support - * add route-map example - * add `res.json(obj, status)` support back for BC - * add "methods" dep, remove internal methods module - * update connect dep - * update auth example to utilize cores pbkdf2 - * updated tests to use "supertest" - -3.0.0beta4 / 2012-06-25 -================== - - * Added `req.auth` - * Added `req.range(size)` - * Added `res.links(obj)` - * Added `res.send(body, status)` support back for backwards compat - * Added `.default()` support to `res.format()` - * Added 2xx / 304 check to `req.fresh` - * Revert "Added + support to the router" - * Fixed `res.send()` freshness check, respect res.statusCode - -3.0.0beta3 / 2012-06-15 -================== - - * Added hogan `--hjs` to express(1) [nullfirm] - * Added another example to content-negotiation - * Added `fresh` dep - * Changed: `res.send()` always checks freshness - * Fixed: expose connects mime module. Cloases #1165 - -3.0.0beta2 / 2012-06-06 -================== - - * Added `+` support to the router - * Added `req.host` - * Changed `req.param()` to check route first - * Update connect dep - -3.0.0beta1 / 2012-06-01 -================== - - * Added `res.format()` callback to override default 406 behaviour - * Fixed `res.redirect()` 406. Closes #1154 - -3.0.0alpha5 / 2012-05-30 -================== - - * Added `req.ip` - * Added `{ signed: true }` option to `res.cookie()` - * Removed `res.signedCookie()` - * Changed: dont reverse `req.ips` - * Fixed "trust proxy" setting check for `req.ips` - -3.0.0alpha4 / 2012-05-09 -================== - - * Added: allow `[]` in jsonp callback. Closes #1128 - * Added `PORT` env var support in generated template. Closes #1118 [benatkin] - * Updated: connect 2.2.2 - -3.0.0alpha3 / 2012-05-04 -================== - - * Added public `app.routes`. Closes #887 - * Added _view-locals_ example - * Added _mvc_ example - * Added `res.locals.use()`. Closes #1120 - * Added conditional-GET support to `res.send()` - * Added: coerce `res.set()` values to strings - * Changed: moved `static()` in generated apps below router - * Changed: `res.send()` only set ETag when not previously set - * Changed connect 2.2.1 dep - * Changed: `make test` now runs unit / acceptance tests - * Fixed req/res proto inheritance - -3.0.0alpha2 / 2012-04-26 -================== - - * Added `make benchmark` back - * Added `res.send()` support for `String` objects - * Added client-side data exposing example - * Added `res.header()` and `req.header()` aliases for BC - * Added `express.createServer()` for BC - * Perf: memoize parsed urls - * Perf: connect 2.2.0 dep - * Changed: make `expressInit()` middleware self-aware - * Fixed: use app.get() for all core settings - * Fixed redis session example - * Fixed session example. Closes #1105 - * Fixed generated express dep. Closes #1078 - -3.0.0alpha1 / 2012-04-15 -================== - - * Added `app.locals.use(callback)` - * Added `app.locals` object - * Added `app.locals(obj)` - * Added `res.locals` object - * Added `res.locals(obj)` - * Added `res.format()` for content-negotiation - * Added `app.engine()` - * Added `res.cookie()` JSON cookie support - * Added "trust proxy" setting - * Added `req.subdomains` - * Added `req.protocol` - * Added `req.secure` - * Added `req.path` - * Added `req.ips` - * Added `req.fresh` - * Added `req.stale` - * Added comma-delmited / array support for `req.accepts()` - * Added debug instrumentation - * Added `res.set(obj)` - * Added `res.set(field, value)` - * Added `res.get(field)` - * Added `app.get(setting)`. Closes #842 - * Added `req.acceptsLanguage()` - * Added `req.acceptsCharset()` - * Added `req.accepted` - * Added `req.acceptedLanguages` - * Added `req.acceptedCharsets` - * Added "json replacer" setting - * Added "json spaces" setting - * Added X-Forwarded-Proto support to `res.redirect()`. Closes #92 - * Added `--less` support to express(1) - * Added `express.response` prototype - * Added `express.request` prototype - * Added `express.application` prototype - * Added `app.path()` - * Added `app.render()` - * Added `res.type()` to replace `res.contentType()` - * Changed: `res.redirect()` to add relative support - * Changed: enable "jsonp callback" by default - * Changed: renamed "case sensitive routes" to "case sensitive routing" - * Rewrite of all tests with mocha - * Removed "root" setting - * Removed `res.redirect('home')` support - * Removed `req.notify()` - * Removed `app.register()` - * Removed `app.redirect()` - * Removed `app.is()` - * Removed `app.helpers()` - * Removed `app.dynamicHelpers()` - * Fixed `res.sendfile()` with non-GET. Closes #723 - * Fixed express(1) public dir for windows. Closes #866 - -2.5.9/ 2012-04-02 -================== - - * Added support for PURGE request method [pbuyle] - * Fixed `express(1)` generated app `app.address()` before `listening` [mmalecki] - -2.5.8 / 2012-02-08 -================== - - * Update mkdirp dep. Closes #991 - -2.5.7 / 2012-02-06 -================== - - * Fixed `app.all` duplicate DELETE requests [mscdex] - -2.5.6 / 2012-01-13 -================== - - * Updated hamljs dev dep. Closes #953 - -2.5.5 / 2012-01-08 -================== - - * Fixed: set `filename` on cached templates [matthewleon] - -2.5.4 / 2012-01-02 -================== - - * Fixed `express(1)` eol on 0.4.x. Closes #947 - -2.5.3 / 2011-12-30 -================== - - * Fixed `req.is()` when a charset is present - -2.5.2 / 2011-12-10 -================== - - * Fixed: express(1) LF -> CRLF for windows - -2.5.1 / 2011-11-17 -================== - - * Changed: updated connect to 1.8.x - * Removed sass.js support from express(1) - -2.5.0 / 2011-10-24 -================== - - * Added ./routes dir for generated app by default - * Added npm install reminder to express(1) app gen - * Added 0.5.x support - * Removed `make test-cov` since it wont work with node 0.5.x - * Fixed express(1) public dir for windows. Closes #866 - -2.4.7 / 2011-10-05 -================== - - * Added mkdirp to express(1). Closes #795 - * Added simple _json-config_ example - * Added shorthand for the parsed request's pathname via `req.path` - * Changed connect dep to 1.7.x to fix npm issue... - * Fixed `res.redirect()` __HEAD__ support. [reported by xerox] - * Fixed `req.flash()`, only escape args - * Fixed absolute path checking on windows. Closes #829 [reported by andrewpmckenzie] - -2.4.6 / 2011-08-22 -================== - - * Fixed multiple param callback regression. Closes #824 [reported by TroyGoode] - -2.4.5 / 2011-08-19 -================== - - * Added support for routes to handle errors. Closes #809 - * Added `app.routes.all()`. Closes #803 - * Added "basepath" setting to work in conjunction with reverse proxies etc. - * Refactored `Route` to use a single array of callbacks - * Added support for multiple callbacks for `app.param()`. Closes #801 -Closes #805 - * Changed: removed .call(self) for route callbacks - * Dependency: `qs >= 0.3.1` - * Fixed `res.redirect()` on windows due to `join()` usage. Closes #808 - -2.4.4 / 2011-08-05 -================== - - * Fixed `res.header()` intention of a set, even when `undefined` - * Fixed `*`, value no longer required - * Fixed `res.send(204)` support. Closes #771 - -2.4.3 / 2011-07-14 -================== - - * Added docs for `status` option special-case. Closes #739 - * Fixed `options.filename`, exposing the view path to template engines - -2.4.2. / 2011-07-06 -================== - - * Revert "removed jsonp stripping" for XSS - -2.4.1 / 2011-07-06 -================== - - * Added `res.json()` JSONP support. Closes #737 - * Added _extending-templates_ example. Closes #730 - * Added "strict routing" setting for trailing slashes - * Added support for multiple envs in `app.configure()` calls. Closes #735 - * Changed: `res.send()` using `res.json()` - * Changed: when cookie `path === null` don't default it - * Changed; default cookie path to "home" setting. Closes #731 - * Removed _pids/logs_ creation from express(1) - -2.4.0 / 2011-06-28 -================== - - * Added chainable `res.status(code)` - * Added `res.json()`, an explicit version of `res.send(obj)` - * Added simple web-service example - -2.3.12 / 2011-06-22 -================== - - * \#express is now on freenode! come join! - * Added `req.get(field, param)` - * Added links to Japanese documentation, thanks @hideyukisaito! - * Added; the `express(1)` generated app outputs the env - * Added `content-negotiation` example - * Dependency: connect >= 1.5.1 < 2.0.0 - * Fixed view layout bug. Closes #720 - * Fixed; ignore body on 304. Closes #701 - -2.3.11 / 2011-06-04 -================== - - * Added `npm test` - * Removed generation of dummy test file from `express(1)` - * Fixed; `express(1)` adds express as a dep - * Fixed; prune on `prepublish` - -2.3.10 / 2011-05-27 -================== - - * Added `req.route`, exposing the current route - * Added _package.json_ generation support to `express(1)` - * Fixed call to `app.param()` function for optional params. Closes #682 - -2.3.9 / 2011-05-25 -================== - - * Fixed bug-ish with `../' in `res.partial()` calls - -2.3.8 / 2011-05-24 -================== - - * Fixed `app.options()` - -2.3.7 / 2011-05-23 -================== - - * Added route `Collection`, ex: `app.get('/user/:id').remove();` - * Added support for `app.param(fn)` to define param logic - * Removed `app.param()` support for callback with return value - * Removed module.parent check from express(1) generated app. Closes #670 - * Refactored router. Closes #639 - -2.3.6 / 2011-05-20 -================== - - * Changed; using devDependencies instead of git submodules - * Fixed redis session example - * Fixed markdown example - * Fixed view caching, should not be enabled in development - -2.3.5 / 2011-05-20 -================== - - * Added export `.view` as alias for `.View` - -2.3.4 / 2011-05-08 -================== - - * Added `./examples/say` - * Fixed `res.sendfile()` bug preventing the transfer of files with spaces - -2.3.3 / 2011-05-03 -================== - - * Added "case sensitive routes" option. - * Changed; split methods supported per rfc [slaskis] - * Fixed route-specific middleware when using the same callback function several times - -2.3.2 / 2011-04-27 -================== - - * Fixed view hints - -2.3.1 / 2011-04-26 -================== - - * Added `app.match()` as `app.match.all()` - * Added `app.lookup()` as `app.lookup.all()` - * Added `app.remove()` for `app.remove.all()` - * Added `app.remove.VERB()` - * Fixed template caching collision issue. Closes #644 - * Moved router over from connect and started refactor - -2.3.0 / 2011-04-25 -================== - - * Added options support to `res.clearCookie()` - * Added `res.helpers()` as alias of `res.locals()` - * Added; json defaults to UTF-8 with `res.send()`. Closes #632. [Daniel * Dependency `connect >= 1.4.0` - * Changed; auto set Content-Type in res.attachement [Aaron Heckmann] - * Renamed "cache views" to "view cache". Closes #628 - * Fixed caching of views when using several apps. Closes #637 - * Fixed gotcha invoking `app.param()` callbacks once per route middleware. -Closes #638 - * Fixed partial lookup precedence. Closes #631 -Shaw] - -2.2.2 / 2011-04-12 -================== - - * Added second callback support for `res.download()` connection errors - * Fixed `filename` option passing to template engine - -2.2.1 / 2011-04-04 -================== - - * Added `layout(path)` helper to change the layout within a view. Closes #610 - * Fixed `partial()` collection object support. - Previously only anything with `.length` would work. - When `.length` is present one must still be aware of holes, - however now `{ collection: {foo: 'bar'}}` is valid, exposes - `keyInCollection` and `keysInCollection`. - - * Performance improved with better view caching - * Removed `request` and `response` locals - * Changed; errorHandler page title is now `Express` instead of `Connect` - -2.2.0 / 2011-03-30 -================== - - * Added `app.lookup.VERB()`, ex `app.lookup.put('/user/:id')`. Closes #606 - * Added `app.match.VERB()`, ex `app.match.put('/user/12')`. Closes #606 - * Added `app.VERB(path)` as alias of `app.lookup.VERB()`. - * Dependency `connect >= 1.2.0` - -2.1.1 / 2011-03-29 -================== - - * Added; expose `err.view` object when failing to locate a view - * Fixed `res.partial()` call `next(err)` when no callback is given [reported by aheckmann] - * Fixed; `res.send(undefined)` responds with 204 [aheckmann] - -2.1.0 / 2011-03-24 -================== - - * Added `/_?` partial lookup support. Closes #447 - * Added `request`, `response`, and `app` local variables - * Added `settings` local variable, containing the app's settings - * Added `req.flash()` exception if `req.session` is not available - * Added `res.send(bool)` support (json response) - * Fixed stylus example for latest version - * Fixed; wrap try/catch around `res.render()` - -2.0.0 / 2011-03-17 -================== - - * Fixed up index view path alternative. - * Changed; `res.locals()` without object returns the locals - -2.0.0rc3 / 2011-03-17 -================== - - * Added `res.locals(obj)` to compliment `res.local(key, val)` - * Added `res.partial()` callback support - * Fixed recursive error reporting issue in `res.render()` - -2.0.0rc2 / 2011-03-17 -================== - - * Changed; `partial()` "locals" are now optional - * Fixed `SlowBuffer` support. Closes #584 [reported by tyrda01] - * Fixed .filename view engine option [reported by drudge] - * Fixed blog example - * Fixed `{req,res}.app` reference when mounting [Ben Weaver] - -2.0.0rc / 2011-03-14 -================== - - * Fixed; expose `HTTPSServer` constructor - * Fixed express(1) default test charset. Closes #579 [reported by secoif] - * Fixed; default charset to utf-8 instead of utf8 for lame IE [reported by NickP] - -2.0.0beta3 / 2011-03-09 -================== - - * Added support for `res.contentType()` literal - The original `res.contentType('.json')`, - `res.contentType('application/json')`, and `res.contentType('json')` - will work now. - * Added `res.render()` status option support back - * Added charset option for `res.render()` - * Added `.charset` support (via connect 1.0.4) - * Added view resolution hints when in development and a lookup fails - * Added layout lookup support relative to the page view. - For example while rendering `./views/user/index.jade` if you create - `./views/user/layout.jade` it will be used in favour of the root layout. - * Fixed `res.redirect()`. RFC states absolute url [reported by unlink] - * Fixed; default `res.send()` string charset to utf8 - * Removed `Partial` constructor (not currently used) - -2.0.0beta2 / 2011-03-07 -================== - - * Added res.render() `.locals` support back to aid in migration process - * Fixed flash example - -2.0.0beta / 2011-03-03 -================== - - * Added HTTPS support - * Added `res.cookie()` maxAge support - * Added `req.header()` _Referrer_ / _Referer_ special-case, either works - * Added mount support for `res.redirect()`, now respects the mount-point - * Added `union()` util, taking place of `merge(clone())` combo - * Added stylus support to express(1) generated app - * Added secret to session middleware used in examples and generated app - * Added `res.local(name, val)` for progressive view locals - * Added default param support to `req.param(name, default)` - * Added `app.disabled()` and `app.enabled()` - * Added `app.register()` support for omitting leading ".", either works - * Added `res.partial()`, using the same interface as `partial()` within a view. Closes #539 - * Added `app.param()` to map route params to async/sync logic - * Added; aliased `app.helpers()` as `app.locals()`. Closes #481 - * Added extname with no leading "." support to `res.contentType()` - * Added `cache views` setting, defaulting to enabled in "production" env - * Added index file partial resolution, eg: partial('user') may try _views/user/index.jade_. - * Added `req.accepts()` support for extensions - * Changed; `res.download()` and `res.sendfile()` now utilize Connect's - static file server `connect.static.send()`. - * Changed; replaced `connect.utils.mime()` with npm _mime_ module - * Changed; allow `req.query` to be pre-defined (via middleware or other parent - * Changed view partial resolution, now relative to parent view - * Changed view engine signature. no longer `engine.render(str, options, callback)`, now `engine.compile(str, options) -> Function`, the returned function accepts `fn(locals)`. - * Fixed `req.param()` bug returning Array.prototype methods. Closes #552 - * Fixed; using `Stream#pipe()` instead of `sys.pump()` in `res.sendfile()` - * Fixed; using _qs_ module instead of _querystring_ - * Fixed; strip unsafe chars from jsonp callbacks - * Removed "stream threshold" setting - -1.0.8 / 2011-03-01 -================== - - * Allow `req.query` to be pre-defined (via middleware or other parent app) - * "connect": ">= 0.5.0 < 1.0.0". Closes #547 - * Removed the long deprecated __EXPRESS_ENV__ support - -1.0.7 / 2011-02-07 -================== - - * Fixed `render()` setting inheritance. - Mounted apps would not inherit "view engine" - -1.0.6 / 2011-02-07 -================== - - * Fixed `view engine` setting bug when period is in dirname - -1.0.5 / 2011-02-05 -================== - - * Added secret to generated app `session()` call - -1.0.4 / 2011-02-05 -================== - - * Added `qs` dependency to _package.json_ - * Fixed namespaced `require()`s for latest connect support - -1.0.3 / 2011-01-13 -================== - - * Remove unsafe characters from JSONP callback names [Ryan Grove] - -1.0.2 / 2011-01-10 -================== - - * Removed nested require, using `connect.router` - -1.0.1 / 2010-12-29 -================== - - * Fixed for middleware stacked via `createServer()` - previously the `foo` middleware passed to `createServer(foo)` - would not have access to Express methods such as `res.send()` - or props like `req.query` etc. - -1.0.0 / 2010-11-16 -================== - - * Added; deduce partial object names from the last segment. - For example by default `partial('forum/post', postObject)` will - give you the _post_ object, providing a meaningful default. - * Added http status code string representation to `res.redirect()` body - * Added; `res.redirect()` supporting _text/plain_ and _text/html_ via __Accept__. - * Added `req.is()` to aid in content negotiation - * Added partial local inheritance [suggested by masylum]. Closes #102 - providing access to parent template locals. - * Added _-s, --session[s]_ flag to express(1) to add session related middleware - * Added _--template_ flag to express(1) to specify the - template engine to use. - * Added _--css_ flag to express(1) to specify the - stylesheet engine to use (or just plain css by default). - * Added `app.all()` support [thanks aheckmann] - * Added partial direct object support. - You may now `partial('user', user)` providing the "user" local, - vs previously `partial('user', { object: user })`. - * Added _route-separation_ example since many people question ways - to do this with CommonJS modules. Also view the _blog_ example for - an alternative. - * Performance; caching view path derived partial object names - * Fixed partial local inheritance precedence. [reported by Nick Poulden] Closes #454 - * Fixed jsonp support; _text/javascript_ as per mailinglist discussion - -1.0.0rc4 / 2010-10-14 -================== - - * Added _NODE_ENV_ support, _EXPRESS_ENV_ is deprecated and will be removed in 1.0.0 - * Added route-middleware support (very helpful, see the [docs](http://expressjs.com/guide.html#Route-Middleware)) - * Added _jsonp callback_ setting to enable/disable jsonp autowrapping [Dav Glass] - * Added callback query check on response.send to autowrap JSON objects for simple webservice implementations [Dav Glass] - * Added `partial()` support for array-like collections. Closes #434 - * Added support for swappable querystring parsers - * Added session usage docs. Closes #443 - * Added dynamic helper caching. Closes #439 [suggested by maritz] - * Added authentication example - * Added basic Range support to `res.sendfile()` (and `res.download()` etc) - * Changed; `express(1)` generated app using 2 spaces instead of 4 - * Default env to "development" again [aheckmann] - * Removed _context_ option is no more, use "scope" - * Fixed; exposing _./support_ libs to examples so they can run without installs - * Fixed mvc example - -1.0.0rc3 / 2010-09-20 -================== - - * Added confirmation for `express(1)` app generation. Closes #391 - * Added extending of flash formatters via `app.flashFormatters` - * Added flash formatter support. Closes #411 - * Added streaming support to `res.sendfile()` using `sys.pump()` when >= "stream threshold" - * Added _stream threshold_ setting for `res.sendfile()` - * Added `res.send()` __HEAD__ support - * Added `res.clearCookie()` - * Added `res.cookie()` - * Added `res.render()` headers option - * Added `res.redirect()` response bodies - * Added `res.render()` status option support. Closes #425 [thanks aheckmann] - * Fixed `res.sendfile()` responding with 403 on malicious path - * Fixed `res.download()` bug; when an error occurs remove _Content-Disposition_ - * Fixed; mounted apps settings now inherit from parent app [aheckmann] - * Fixed; stripping Content-Length / Content-Type when 204 - * Fixed `res.send()` 204. Closes #419 - * Fixed multiple _Set-Cookie_ headers via `res.header()`. Closes #402 - * Fixed bug messing with error handlers when `listenFD()` is called instead of `listen()`. [thanks guillermo] - - -1.0.0rc2 / 2010-08-17 -================== - - * Added `app.register()` for template engine mapping. Closes #390 - * Added `res.render()` callback support as second argument (no options) - * Added callback support to `res.download()` - * Added callback support for `res.sendfile()` - * Added support for middleware access via `express.middlewareName()` vs `connect.middlewareName()` - * Added "partials" setting to docs - * Added default expresso tests to `express(1)` generated app. Closes #384 - * Fixed `res.sendfile()` error handling, defer via `next()` - * Fixed `res.render()` callback when a layout is used [thanks guillermo] - * Fixed; `make install` creating ~/.node_libraries when not present - * Fixed issue preventing error handlers from being defined anywhere. Closes #387 - -1.0.0rc / 2010-07-28 -================== - - * Added mounted hook. Closes #369 - * Added connect dependency to _package.json_ - - * Removed "reload views" setting and support code - development env never caches, production always caches. - - * Removed _param_ in route callbacks, signature is now - simply (req, res, next), previously (req, res, params, next). - Use _req.params_ for path captures, _req.query_ for GET params. - - * Fixed "home" setting - * Fixed middleware/router precedence issue. Closes #366 - * Fixed; _configure()_ callbacks called immediately. Closes #368 - -1.0.0beta2 / 2010-07-23 -================== - - * Added more examples - * Added; exporting `Server` constructor - * Added `Server#helpers()` for view locals - * Added `Server#dynamicHelpers()` for dynamic view locals. Closes #349 - * Added support for absolute view paths - * Added; _home_ setting defaults to `Server#route` for mounted apps. Closes #363 - * Added Guillermo Rauch to the contributor list - * Added support for "as" for non-collection partials. Closes #341 - * Fixed _install.sh_, ensuring _~/.node_libraries_ exists. Closes #362 [thanks jf] - * Fixed `res.render()` exceptions, now passed to `next()` when no callback is given [thanks guillermo] - * Fixed instanceof `Array` checks, now `Array.isArray()` - * Fixed express(1) expansion of public dirs. Closes #348 - * Fixed middleware precedence. Closes #345 - * Fixed view watcher, now async [thanks aheckmann] - -1.0.0beta / 2010-07-15 -================== - - * Re-write - - much faster - - much lighter - - Check [ExpressJS.com](http://expressjs.com) for migration guide and updated docs - -0.14.0 / 2010-06-15 -================== - - * Utilize relative requires - * Added Static bufferSize option [aheckmann] - * Fixed caching of view and partial subdirectories [aheckmann] - * Fixed mime.type() comments now that ".ext" is not supported - * Updated haml submodule - * Updated class submodule - * Removed bin/express - -0.13.0 / 2010-06-01 -================== - - * Added node v0.1.97 compatibility - * Added support for deleting cookies via Request#cookie('key', null) - * Updated haml submodule - * Fixed not-found page, now using using charset utf-8 - * Fixed show-exceptions page, now using using charset utf-8 - * Fixed view support due to fs.readFile Buffers - * Changed; mime.type() no longer accepts ".type" due to node extname() changes - -0.12.0 / 2010-05-22 -================== - - * Added node v0.1.96 compatibility - * Added view `helpers` export which act as additional local variables - * Updated haml submodule - * Changed ETag; removed inode, modified time only - * Fixed LF to CRLF for setting multiple cookies - * Fixed cookie complation; values are now urlencoded - * Fixed cookies parsing; accepts quoted values and url escaped cookies - -0.11.0 / 2010-05-06 -================== - - * Added support for layouts using different engines - - this.render('page.html.haml', { layout: 'super-cool-layout.html.ejs' }) - - this.render('page.html.haml', { layout: 'foo' }) // assumes 'foo.html.haml' - - this.render('page.html.haml', { layout: false }) // no layout - * Updated ext submodule - * Updated haml submodule - * Fixed EJS partial support by passing along the context. Issue #307 - -0.10.1 / 2010-05-03 -================== - - * Fixed binary uploads. - -0.10.0 / 2010-04-30 -================== - - * Added charset support via Request#charset (automatically assigned to 'UTF-8' when respond()'s - encoding is set to 'utf8' or 'utf-8'. - * Added "encoding" option to Request#render(). Closes #299 - * Added "dump exceptions" setting, which is enabled by default. - * Added simple ejs template engine support - * Added error reponse support for text/plain, application/json. Closes #297 - * Added callback function param to Request#error() - * Added Request#sendHead() - * Added Request#stream() - * Added support for Request#respond(304, null) for empty response bodies - * Added ETag support to Request#sendfile() - * Added options to Request#sendfile(), passed to fs.createReadStream() - * Added filename arg to Request#download() - * Performance enhanced due to pre-reversing plugins so that plugins.reverse() is not called on each request - * Performance enhanced by preventing several calls to toLowerCase() in Router#match() - * Changed; Request#sendfile() now streams - * Changed; Renamed Request#halt() to Request#respond(). Closes #289 - * Changed; Using sys.inspect() instead of JSON.encode() for error output - * Changed; run() returns the http.Server instance. Closes #298 - * Changed; Defaulting Server#host to null (INADDR_ANY) - * Changed; Logger "common" format scale of 0.4f - * Removed Logger "request" format - * Fixed; Catching ENOENT in view caching, preventing error when "views/partials" is not found - * Fixed several issues with http client - * Fixed Logger Content-Length output - * Fixed bug preventing Opera from retaining the generated session id. Closes #292 - -0.9.0 / 2010-04-14 -================== - - * Added DSL level error() route support - * Added DSL level notFound() route support - * Added Request#error() - * Added Request#notFound() - * Added Request#render() callback function. Closes #258 - * Added "max upload size" setting - * Added "magic" variables to collection partials (\_\_index\_\_, \_\_length\_\_, \_\_isFirst\_\_, \_\_isLast\_\_). Closes #254 - * Added [haml.js](http://github.com/visionmedia/haml.js) submodule; removed haml-js - * Added callback function support to Request#halt() as 3rd/4th arg - * Added preprocessing of route param wildcards using param(). Closes #251 - * Added view partial support (with collections etc) - * Fixed bug preventing falsey params (such as ?page=0). Closes #286 - * Fixed setting of multiple cookies. Closes #199 - * Changed; view naming convention is now NAME.TYPE.ENGINE (for example page.html.haml) - * Changed; session cookie is now httpOnly - * Changed; Request is no longer global - * Changed; Event is no longer global - * Changed; "sys" module is no longer global - * Changed; moved Request#download to Static plugin where it belongs - * Changed; Request instance created before body parsing. Closes #262 - * Changed; Pre-caching views in memory when "cache view contents" is enabled. Closes #253 - * Changed; Pre-caching view partials in memory when "cache view partials" is enabled - * Updated support to node --version 0.1.90 - * Updated dependencies - * Removed set("session cookie") in favour of use(Session, { cookie: { ... }}) - * Removed utils.mixin(); use Object#mergeDeep() - -0.8.0 / 2010-03-19 -================== - - * Added coffeescript example app. Closes #242 - * Changed; cache api now async friendly. Closes #240 - * Removed deprecated 'express/static' support. Use 'express/plugins/static' - -0.7.6 / 2010-03-19 -================== - - * Added Request#isXHR. Closes #229 - * Added `make install` (for the executable) - * Added `express` executable for setting up simple app templates - * Added "GET /public/*" to Static plugin, defaulting to /public - * Added Static plugin - * Fixed; Request#render() only calls cache.get() once - * Fixed; Namespacing View caches with "view:" - * Fixed; Namespacing Static caches with "static:" - * Fixed; Both example apps now use the Static plugin - * Fixed set("views"). Closes #239 - * Fixed missing space for combined log format - * Deprecated Request#sendfile() and 'express/static' - * Removed Server#running - -0.7.5 / 2010-03-16 -================== - - * Added Request#flash() support without args, now returns all flashes - * Updated ext submodule - -0.7.4 / 2010-03-16 -================== - - * Fixed session reaper - * Changed; class.js replacing js-oo Class implementation (quite a bit faster, no browser cruft) - -0.7.3 / 2010-03-16 -================== - - * Added package.json - * Fixed requiring of haml / sass due to kiwi removal - -0.7.2 / 2010-03-16 -================== - - * Fixed GIT submodules (HAH!) - -0.7.1 / 2010-03-16 -================== - - * Changed; Express now using submodules again until a PM is adopted - * Changed; chat example using millisecond conversions from ext - -0.7.0 / 2010-03-15 -================== - - * Added Request#pass() support (finds the next matching route, or the given path) - * Added Logger plugin (default "common" format replaces CommonLogger) - * Removed Profiler plugin - * Removed CommonLogger plugin - -0.6.0 / 2010-03-11 -================== - - * Added seed.yml for kiwi package management support - * Added HTTP client query string support when method is GET. Closes #205 - - * Added support for arbitrary view engines. - For example "foo.engine.html" will now require('engine'), - the exports from this module are cached after the first require(). - - * Added async plugin support - - * Removed usage of RESTful route funcs as http client - get() etc, use http.get() and friends - - * Removed custom exceptions - -0.5.0 / 2010-03-10 -================== - - * Added ext dependency (library of js extensions) - * Removed extname() / basename() utils. Use path module - * Removed toArray() util. Use arguments.values - * Removed escapeRegexp() util. Use RegExp.escape() - * Removed process.mixin() dependency. Use utils.mixin() - * Removed Collection - * Removed ElementCollection - * Shameless self promotion of ebook "Advanced JavaScript" (http://dev-mag.com) ;) - -0.4.0 / 2010-02-11 -================== - - * Added flash() example to sample upload app - * Added high level restful http client module (express/http) - * Changed; RESTful route functions double as HTTP clients. Closes #69 - * Changed; throwing error when routes are added at runtime - * Changed; defaulting render() context to the current Request. Closes #197 - * Updated haml submodule - -0.3.0 / 2010-02-11 -================== - - * Updated haml / sass submodules. Closes #200 - * Added flash message support. Closes #64 - * Added accepts() now allows multiple args. fixes #117 - * Added support for plugins to halt. Closes #189 - * Added alternate layout support. Closes #119 - * Removed Route#run(). Closes #188 - * Fixed broken specs due to use(Cookie) missing - -0.2.1 / 2010-02-05 -================== - - * Added "plot" format option for Profiler (for gnuplot processing) - * Added request number to Profiler plugin - * Fixed binary encoding for multi-part file uploads, was previously defaulting to UTF8 - * Fixed issue with routes not firing when not files are present. Closes #184 - * Fixed process.Promise -> events.Promise - -0.2.0 / 2010-02-03 -================== - - * Added parseParam() support for name[] etc. (allows for file inputs with "multiple" attr) Closes #180 - * Added Both Cache and Session option "reapInterval" may be "reapEvery". Closes #174 - * Added expiration support to cache api with reaper. Closes #133 - * Added cache Store.Memory#reap() - * Added Cache; cache api now uses first class Cache instances - * Added abstract session Store. Closes #172 - * Changed; cache Memory.Store#get() utilizing Collection - * Renamed MemoryStore -> Store.Memory - * Fixed use() of the same plugin several time will always use latest options. Closes #176 - -0.1.0 / 2010-02-03 -================== - - * Changed; Hooks (before / after) pass request as arg as well as evaluated in their context - * Updated node support to 0.1.27 Closes #169 - * Updated dirname(__filename) -> __dirname - * Updated libxmljs support to v0.2.0 - * Added session support with memory store / reaping - * Added quick uid() helper - * Added multi-part upload support - * Added Sass.js support / submodule - * Added production env caching view contents and static files - * Added static file caching. Closes #136 - * Added cache plugin with memory stores - * Added support to StaticFile so that it works with non-textual files. - * Removed dirname() helper - * Removed several globals (now their modules must be required) - -0.0.2 / 2010-01-10 -================== - - * Added view benchmarks; currently haml vs ejs - * Added Request#attachment() specs. Closes #116 - * Added use of node's parseQuery() util. Closes #123 - * Added `make init` for submodules - * Updated Haml - * Updated sample chat app to show messages on load - * Updated libxmljs parseString -> parseHtmlString - * Fixed `make init` to work with older versions of git - * Fixed specs can now run independant specs for those who cant build deps. Closes #127 - * Fixed issues introduced by the node url module changes. Closes 126. - * Fixed two assertions failing due to Collection#keys() returning strings - * Fixed faulty Collection#toArray() spec due to keys() returning strings - * Fixed `make test` now builds libxmljs.node before testing - -0.0.1 / 2010-01-03 -================== - - * Initial release diff --git a/node_modules/express/LICENSE b/node_modules/express/LICENSE deleted file mode 100644 index 36075a3..0000000 --- a/node_modules/express/LICENSE +++ /dev/null @@ -1,22 +0,0 @@ -(The MIT License) - -Copyright (c) 2009-2011 TJ Holowaychuk - -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. \ No newline at end of file diff --git a/node_modules/express/Makefile b/node_modules/express/Makefile deleted file mode 100644 index 228f299..0000000 --- a/node_modules/express/Makefile +++ /dev/null @@ -1,33 +0,0 @@ - -MOCHA_OPTS= --check-leaks -REPORTER = dot - -check: test - -test: test-unit test-acceptance - -test-unit: - @NODE_ENV=test ./node_modules/.bin/mocha \ - --reporter $(REPORTER) \ - $(MOCHA_OPTS) - -test-acceptance: - @NODE_ENV=test ./node_modules/.bin/mocha \ - --reporter $(REPORTER) \ - --bail \ - test/acceptance/*.js - -test-cov: lib-cov - @EXPRESS_COV=1 $(MAKE) test REPORTER=html-cov > coverage.html - -lib-cov: - @jscoverage lib lib-cov - -benchmark: - @./support/bench - -clean: - rm -f coverage.html - rm -fr lib-cov - -.PHONY: test test-unit test-acceptance benchmark clean diff --git a/node_modules/express/Readme.md b/node_modules/express/Readme.md deleted file mode 100644 index 61ee6ea..0000000 --- a/node_modules/express/Readme.md +++ /dev/null @@ -1,180 +0,0 @@ -![express logo](http://f.cl.ly/items/0V2S1n0K1i3y1c122g04/Screen%20Shot%202012-04-11%20at%209.59.42%20AM.png) - - Fast, unopinionated, minimalist web framework for [node](http://nodejs.org). [![Build Status](https://secure.travis-ci.org/visionmedia/express.png)](http://travis-ci.org/visionmedia/express) [![Dependency Status](https://gemnasium.com/visionmedia/express.png)](https://gemnasium.com/visionmedia/express) - -```js -var express = require('express'); -var app = express(); - -app.get('/', function(req, res){ - res.send('Hello World'); -}); - -app.listen(3000); -``` - -## Installation - - $ npm install -g express - -## Quick Start - - The quickest way to get started with express is to utilize the executable `express(1)` to generate an application as shown below: - - Create the app: - - $ npm install -g express - $ express /tmp/foo && cd /tmp/foo - - Install dependencies: - - $ npm install - - Start the server: - - $ node app - -## Features - - * Built on [Connect](http://github.com/senchalabs/connect) - * Robust routing - * HTTP helpers (redirection, caching, etc) - * View system supporting 14+ template engines - * Content negotiation - * Focus on high performance - * Environment based configuration - * Executable for generating applications quickly - * High test coverage - -## Philosophy - - The Express philosophy is to provide small, robust tooling for HTTP servers. Making - it a great solution for single page applications, web sites, hybrids, or public - HTTP APIs. - - Built on Connect you can use _only_ what you need, and nothing more, applications - can be as big or as small as you like, even a single file. Express does - not force you to use any specific ORM or template engine. With support for over - 14 template engines via [Consolidate.js](http://github.com/visionmedia/consolidate.js) - you can quickly craft your perfect framework. - -## More Information - - * Join #express on freenode - * [Google Group](http://groups.google.com/group/express-js) for discussion - * Follow [tjholowaychuk](http://twitter.com/tjholowaychuk) on twitter for updates - * Visit the [Wiki](http://github.com/visionmedia/express/wiki) - * [日本語ドキュメンテーション](http://hideyukisaito.com/doc/expressjs/) by [hideyukisaito](https://github.com/hideyukisaito) - * [Русскоязычная документация](http://jsman.ru/express/) - * Run express examples [online](https://runnable.com/express) - -## Viewing Examples - -Clone the Express repo, then install the dev dependencies to install all the example / test suite deps: - - $ git clone git://github.com/visionmedia/express.git --depth 1 - $ cd express - $ npm install - -then run whichever tests you want: - - $ node examples/content-negotiation - -## Running Tests - -To run the test suite first invoke the following command within the repo, installing the development dependencies: - - $ npm install - -then run the tests: - - $ make test - -## Contributors - -``` -project: express -commits: 3559 -active : 468 days -files : 237 -authors: - 1891 Tj Holowaychuk 53.1% - 1285 visionmedia 36.1% - 182 TJ Holowaychuk 5.1% - 54 Aaron Heckmann 1.5% - 34 csausdev 1.0% - 26 ciaranj 0.7% - 21 Robert Sköld 0.6% - 6 Guillermo Rauch 0.2% - 3 Dav Glass 0.1% - 3 Nick Poulden 0.1% - 2 Randy Merrill 0.1% - 2 Benny Wong 0.1% - 2 Hunter Loftis 0.1% - 2 Jake Gordon 0.1% - 2 Brian McKinney 0.1% - 2 Roman Shtylman 0.1% - 2 Ben Weaver 0.1% - 2 Dave Hoover 0.1% - 2 Eivind Fjeldstad 0.1% - 2 Daniel Shaw 0.1% - 1 Matt Colyer 0.0% - 1 Pau Ramon 0.0% - 1 Pero Pejovic 0.0% - 1 Peter Rekdal Sunde 0.0% - 1 Raynos 0.0% - 1 Teng Siong Ong 0.0% - 1 Viktor Kelemen 0.0% - 1 ctide 0.0% - 1 8bitDesigner 0.0% - 1 isaacs 0.0% - 1 mgutz 0.0% - 1 pikeas 0.0% - 1 shuwatto 0.0% - 1 tstrimple 0.0% - 1 ewoudj 0.0% - 1 Adam Sanderson 0.0% - 1 Andrii Kostenko 0.0% - 1 Andy Hiew 0.0% - 1 Arpad Borsos 0.0% - 1 Ashwin Purohit 0.0% - 1 Benjen 0.0% - 1 Darren Torpey 0.0% - 1 Greg Ritter 0.0% - 1 Gregory Ritter 0.0% - 1 James Herdman 0.0% - 1 Jim Snodgrass 0.0% - 1 Joe McCann 0.0% - 1 Jonathan Dumaine 0.0% - 1 Jonathan Palardy 0.0% - 1 Jonathan Zacsh 0.0% - 1 Justin Lilly 0.0% - 1 Ken Sato 0.0% - 1 Maciej Małecki 0.0% - 1 Masahiro Hayashi 0.0% -``` - -## License - -(The MIT License) - -Copyright (c) 2009-2012 TJ Holowaychuk <tj@vision-media.ca> - -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/node_modules/express/bin/express b/node_modules/express/bin/express deleted file mode 100755 index 337c74e..0000000 --- a/node_modules/express/bin/express +++ /dev/null @@ -1,422 +0,0 @@ -#!/usr/bin/env node - -/** - * Module dependencies. - */ - -var exec = require('child_process').exec - , program = require('commander') - , mkdirp = require('mkdirp') - , pkg = require('../package.json') - , version = pkg.version - , os = require('os') - , fs = require('fs'); - -// CLI - -program - .version(version) - .option('-s, --sessions', 'add session support') - .option('-e, --ejs', 'add ejs engine support (defaults to jade)') - .option('-J, --jshtml', 'add jshtml engine support (defaults to jade)') - .option('-H, --hogan', 'add hogan.js engine support') - .option('-c, --css ', 'add stylesheet support (less|stylus) (defaults to plain css)') - .option('-f, --force', 'force on non-empty directory') - .parse(process.argv); - -// Path - -var path = program.args.shift() || '.'; - -// end-of-line code - -var eol = os.EOL - -// Template engine - -program.template = 'jade'; -if (program.ejs) program.template = 'ejs'; -if (program.jshtml) program.template = 'jshtml'; -if (program.hogan) program.template = 'hjs'; - -/** - * Routes index template. - */ - -var index = [ - '' - , '/*' - , ' * GET home page.' - , ' */' - , '' - , 'exports.index = function(req, res){' - , ' res.render(\'index\', { title: \'Express\' });' - , '};' -].join(eol); - -/** - * Routes users template. - */ - -var users = [ - '' - , '/*' - , ' * GET users listing.' - , ' */' - , '' - , 'exports.list = function(req, res){' - , ' res.send("respond with a resource");' - , '};' -].join(eol); - -/** - * Jade layout template. - */ - -var jadeLayout = [ - 'doctype 5' - , 'html' - , ' head' - , ' title= title' - , ' link(rel=\'stylesheet\', href=\'/stylesheets/style.css\')' - , ' body' - , ' block content' -].join(eol); - -/** - * Jade index template. - */ - -var jadeIndex = [ - 'extends layout' - , '' - , 'block content' - , ' h1= title' - , ' p Welcome to #{title}' -].join(eol); - -/** - * EJS index template. - */ - -var ejsIndex = [ - '' - , '' - , ' ' - , ' <%= title %>' - , ' ' - , ' ' - , ' ' - , '

<%= title %>

' - , '

Welcome to <%= title %>

' - , ' ' - , '' -].join(eol); - -/** - * JSHTML layout template. - */ - -var jshtmlLayout = [ - '' - , '' - , ' ' - , ' @write(title) ' - , ' ' - , ' ' - , ' ' - , ' @write(body)' - , ' ' - , '' -].join(eol); - -/** - * JSHTML index template. - */ - -var jshtmlIndex = [ - '

@write(title)

' - , '

Welcome to @write(title)

' -].join(eol); - -/** - * Hogan.js index template. - */ -var hoganIndex = [ - '' - , '' - , ' ' - , ' {{ title }}' - , ' ' - , ' ' - , ' ' - , '

{{ title }}

' - , '

Welcome to {{ title }}

' - , ' ' - , '' -].join(eol); - -/** - * Default css template. - */ - -var css = [ - 'body {' - , ' padding: 50px;' - , ' font: 14px "Lucida Grande", Helvetica, Arial, sans-serif;' - , '}' - , '' - , 'a {' - , ' color: #00B7FF;' - , '}' -].join(eol); - -/** - * Default less template. - */ - -var less = [ - 'body {' - , ' padding: 50px;' - , ' font: 14px "Lucida Grande", Helvetica, Arial, sans-serif;' - , '}' - , '' - , 'a {' - , ' color: #00B7FF;' - , '}' -].join(eol); - -/** - * Default stylus template. - */ - -var stylus = [ - 'body' - , ' padding: 50px' - , ' font: 14px "Lucida Grande", Helvetica, Arial, sans-serif' - , 'a' - , ' color: #00B7FF' -].join(eol); - -/** - * App template. - */ - -var app = [ - '' - , '/**' - , ' * Module dependencies.' - , ' */' - , '' - , 'var express = require(\'express\')' - , ' , routes = require(\'./routes\')' - , ' , user = require(\'./routes/user\')' - , ' , http = require(\'http\')' - , ' , path = require(\'path\');' - , '' - , 'var app = express();' - , '' - , '// all environments' - , 'app.set(\'port\', process.env.PORT || 3000);' - , 'app.set(\'views\', __dirname + \'/views\');' - , 'app.set(\'view engine\', \':TEMPLATE\');' - , 'app.use(express.favicon());' - , 'app.use(express.logger(\'dev\'));' - , 'app.use(express.bodyParser());' - , 'app.use(express.methodOverride());{sess}' - , 'app.use(app.router);{css}' - , 'app.use(express.static(path.join(__dirname, \'public\')));' - , '' - , '// development only' - , 'if (\'development\' == app.get(\'env\')) {' - , ' app.use(express.errorHandler());' - , '}' - , '' - , 'app.get(\'/\', routes.index);' - , 'app.get(\'/users\', user.list);' - , '' - , 'http.createServer(app).listen(app.get(\'port\'), function(){' - , ' console.log(\'Express server listening on port \' + app.get(\'port\'));' - , '});' - , '' -].join(eol); - -// Generate application - -(function createApplication(path) { - emptyDirectory(path, function(empty){ - if (empty || program.force) { - createApplicationAt(path); - } else { - program.confirm('destination is not empty, continue? ', function(ok){ - if (ok) { - process.stdin.destroy(); - createApplicationAt(path); - } else { - abort('aborting'); - } - }); - } - }); -})(path); - -/** - * Create application at the given directory `path`. - * - * @param {String} path - */ - -function createApplicationAt(path) { - console.log(); - process.on('exit', function(){ - console.log(); - console.log(' install dependencies:'); - console.log(' $ cd %s && npm install', path); - console.log(); - console.log(' run the app:'); - console.log(' $ node app'); - console.log(); - }); - - mkdir(path, function(){ - mkdir(path + '/public'); - mkdir(path + '/public/javascripts'); - mkdir(path + '/public/images'); - mkdir(path + '/public/stylesheets', function(){ - switch (program.css) { - case 'less': - write(path + '/public/stylesheets/style.less', less); - break; - case 'stylus': - write(path + '/public/stylesheets/style.styl', stylus); - break; - default: - write(path + '/public/stylesheets/style.css', css); - } - }); - - mkdir(path + '/routes', function(){ - write(path + '/routes/index.js', index); - write(path + '/routes/user.js', users); - }); - - mkdir(path + '/views', function(){ - switch (program.template) { - case 'ejs': - write(path + '/views/index.ejs', ejsIndex); - break; - case 'jade': - write(path + '/views/layout.jade', jadeLayout); - write(path + '/views/index.jade', jadeIndex); - break; - case 'jshtml': - write(path + '/views/layout.jshtml', jshtmlLayout); - write(path + '/views/index.jshtml', jshtmlIndex); - break; - case 'hjs': - write(path + '/views/index.hjs', hoganIndex); - break; - - } - }); - - // CSS Engine support - switch (program.css) { - case 'less': - app = app.replace('{css}', eol + ' app.use(require(\'less-middleware\')({ src: __dirname + \'/public\' }));'); - break; - case 'stylus': - app = app.replace('{css}', eol + ' app.use(require(\'stylus\').middleware(__dirname + \'/public\'));'); - break; - default: - app = app.replace('{css}', ''); - } - - // Session support - app = app.replace('{sess}', program.sessions - ? eol + ' app.use(express.cookieParser(\'your secret here\'));' + eol + ' app.use(express.session());' - : ''); - - // Template support - app = app.replace(':TEMPLATE', program.template); - - // package.json - var pkg = { - name: 'application-name' - , version: '0.0.1' - , private: true - , scripts: { start: 'node app.js' } - , dependencies: { - express: version - } - } - - if (program.template) pkg.dependencies[program.template] = '*'; - - // CSS Engine support - switch (program.css) { - case 'less': - pkg.dependencies['less-middleware'] = '*'; - break; - default: - if (program.css) { - pkg.dependencies[program.css] = '*'; - } - } - - write(path + '/package.json', JSON.stringify(pkg, null, 2)); - write(path + '/app.js', app); - }); -} - -/** - * Check if the given directory `path` is empty. - * - * @param {String} path - * @param {Function} fn - */ - -function emptyDirectory(path, fn) { - fs.readdir(path, function(err, files){ - if (err && 'ENOENT' != err.code) throw err; - fn(!files || !files.length); - }); -} - -/** - * echo str > path. - * - * @param {String} path - * @param {String} str - */ - -function write(path, str) { - fs.writeFile(path, str); - console.log(' \x1b[36mcreate\x1b[0m : ' + path); -} - -/** - * Mkdir -p. - * - * @param {String} path - * @param {Function} fn - */ - -function mkdir(path, fn) { - mkdirp(path, 0755, function(err){ - if (err) throw err; - console.log(' \033[36mcreate\033[0m : ' + path); - fn && fn(); - }); -} - -/** - * Exit with the given `str`. - * - * @param {String} str - */ - -function abort(str) { - console.error(str); - process.exit(1); -} diff --git a/node_modules/express/client.js b/node_modules/express/client.js deleted file mode 100644 index 8984c44..0000000 --- a/node_modules/express/client.js +++ /dev/null @@ -1,25 +0,0 @@ - -var http = require('http'); - -var times = 50; - -while (times--) { - var req = http.request({ - port: 3000 - , method: 'POST' - , headers: { 'Content-Type': 'application/x-www-form-urlencoded' } - }); - - req.on('response', function(res){ - console.log(res.statusCode); - }); - - var n = 500000; - while (n--) { - req.write('foo=bar&bar=baz&'); - } - - req.write('foo=bar&bar=baz'); - - req.end(); -} \ No newline at end of file diff --git a/node_modules/express/index.js b/node_modules/express/index.js deleted file mode 100644 index bfe9934..0000000 --- a/node_modules/express/index.js +++ /dev/null @@ -1,4 +0,0 @@ - -module.exports = process.env.EXPRESS_COV - ? require('./lib-cov/express') - : require('./lib/express'); \ No newline at end of file diff --git a/node_modules/express/lib/application.js b/node_modules/express/lib/application.js deleted file mode 100644 index 557f734..0000000 --- a/node_modules/express/lib/application.js +++ /dev/null @@ -1,530 +0,0 @@ -/** - * Module dependencies. - */ - -var connect = require('connect') - , Router = require('./router') - , methods = require('methods') - , middleware = require('./middleware') - , debug = require('debug')('express:application') - , locals = require('./utils').locals - , View = require('./view') - , utils = connect.utils - , path = require('path') - , http = require('http') - , join = path.join; - -/** - * Application prototype. - */ - -var app = exports = module.exports = {}; - -/** - * Initialize the server. - * - * - setup default configuration - * - setup default middleware - * - setup route reflection methods - * - * @api private - */ - -app.init = function(){ - this.cache = {}; - this.settings = {}; - this.engines = {}; - this.defaultConfiguration(); -}; - -/** - * Initialize application configuration. - * - * @api private - */ - -app.defaultConfiguration = function(){ - // default settings - this.enable('x-powered-by'); - this.set('env', process.env.NODE_ENV || 'development'); - this.set('subdomain offset', 2); - debug('booting in %s mode', this.get('env')); - - // implicit middleware - this.use(connect.query()); - this.use(middleware.init(this)); - - // inherit protos - this.on('mount', function(parent){ - this.request.__proto__ = parent.request; - this.response.__proto__ = parent.response; - this.engines.__proto__ = parent.engines; - this.settings.__proto__ = parent.settings; - }); - - // router - this._router = new Router(this); - this.routes = this._router.map; - this.__defineGetter__('router', function(){ - this._usedRouter = true; - this._router.caseSensitive = this.enabled('case sensitive routing'); - this._router.strict = this.enabled('strict routing'); - return this._router.middleware; - }); - - // setup locals - this.locals = locals(this); - - // default locals - this.locals.settings = this.settings; - - // default configuration - this.set('views', process.cwd() + '/views'); - this.set('jsonp callback name', 'callback'); - - this.configure('development', function(){ - this.set('json spaces', 2); - }); - - this.configure('production', function(){ - this.enable('view cache'); - }); -}; - -/** - * Proxy `connect#use()` to apply settings to - * mounted applications. - * - * @param {String|Function|Server} route - * @param {Function|Server} fn - * @return {app} for chaining - * @api public - */ - -app.use = function(route, fn){ - var app; - - // default route to '/' - if ('string' != typeof route) fn = route, route = '/'; - - // express app - if (fn.handle && fn.set) app = fn; - - // restore .app property on req and res - if (app) { - app.route = route; - fn = function(req, res, next) { - var orig = req.app; - app.handle(req, res, function(err){ - req.app = res.app = orig; - req.__proto__ = orig.request; - res.__proto__ = orig.response; - next(err); - }); - }; - } - - connect.proto.use.call(this, route, fn); - - // mounted an app - if (app) { - app.parent = this; - app.emit('mount', this); - } - - return this; -}; - -/** - * Register the given template engine callback `fn` - * as `ext`. - * - * By default will `require()` the engine based on the - * file extension. For example if you try to render - * a "foo.jade" file Express will invoke the following internally: - * - * app.engine('jade', require('jade').__express); - * - * For engines that do not provide `.__express` out of the box, - * or if you wish to "map" a different extension to the template engine - * you may use this method. For example mapping the EJS template engine to - * ".html" files: - * - * app.engine('html', require('ejs').renderFile); - * - * In this case EJS provides a `.renderFile()` method with - * the same signature that Express expects: `(path, options, callback)`, - * though note that it aliases this method as `ejs.__express` internally - * so if you're using ".ejs" extensions you dont need to do anything. - * - * Some template engines do not follow this convention, the - * [Consolidate.js](https://github.com/visionmedia/consolidate.js) - * library was created to map all of node's popular template - * engines to follow this convention, thus allowing them to - * work seamlessly within Express. - * - * @param {String} ext - * @param {Function} fn - * @return {app} for chaining - * @api public - */ - -app.engine = function(ext, fn){ - if ('function' != typeof fn) throw new Error('callback function required'); - if ('.' != ext[0]) ext = '.' + ext; - this.engines[ext] = fn; - return this; -}; - -/** - * Map the given param placeholder `name`(s) to the given callback(s). - * - * Parameter mapping is used to provide pre-conditions to routes - * which use normalized placeholders. For example a _:user_id_ parameter - * could automatically load a user's information from the database without - * any additional code, - * - * The callback uses the samesignature as middleware, the only differencing - * being that the value of the placeholder is passed, in this case the _id_ - * of the user. Once the `next()` function is invoked, just like middleware - * it will continue on to execute the route, or subsequent parameter functions. - * - * app.param('user_id', function(req, res, next, id){ - * User.find(id, function(err, user){ - * if (err) { - * next(err); - * } else if (user) { - * req.user = user; - * next(); - * } else { - * next(new Error('failed to load user')); - * } - * }); - * }); - * - * @param {String|Array} name - * @param {Function} fn - * @return {app} for chaining - * @api public - */ - -app.param = function(name, fn){ - var self = this - , fns = [].slice.call(arguments, 1); - - // array - if (Array.isArray(name)) { - name.forEach(function(name){ - fns.forEach(function(fn){ - self.param(name, fn); - }); - }); - // param logic - } else if ('function' == typeof name) { - this._router.param(name); - // single - } else { - if (':' == name[0]) name = name.substr(1); - fns.forEach(function(fn){ - self._router.param(name, fn); - }); - } - - return this; -}; - -/** - * Assign `setting` to `val`, or return `setting`'s value. - * - * app.set('foo', 'bar'); - * app.get('foo'); - * // => "bar" - * - * Mounted servers inherit their parent server's settings. - * - * @param {String} setting - * @param {String} val - * @return {Server} for chaining - * @api public - */ - -app.set = function(setting, val){ - if (1 == arguments.length) { - return this.settings[setting]; - } else { - this.settings[setting] = val; - return this; - } -}; - -/** - * Return the app's absolute pathname - * based on the parent(s) that have - * mounted it. - * - * For example if the application was - * mounted as "/admin", which itself - * was mounted as "/blog" then the - * return value would be "/blog/admin". - * - * @return {String} - * @api private - */ - -app.path = function(){ - return this.parent - ? this.parent.path() + this.route - : ''; -}; - -/** - * Check if `setting` is enabled (truthy). - * - * app.enabled('foo') - * // => false - * - * app.enable('foo') - * app.enabled('foo') - * // => true - * - * @param {String} setting - * @return {Boolean} - * @api public - */ - -app.enabled = function(setting){ - return !!this.set(setting); -}; - -/** - * Check if `setting` is disabled. - * - * app.disabled('foo') - * // => true - * - * app.enable('foo') - * app.disabled('foo') - * // => false - * - * @param {String} setting - * @return {Boolean} - * @api public - */ - -app.disabled = function(setting){ - return !this.set(setting); -}; - -/** - * Enable `setting`. - * - * @param {String} setting - * @return {app} for chaining - * @api public - */ - -app.enable = function(setting){ - return this.set(setting, true); -}; - -/** - * Disable `setting`. - * - * @param {String} setting - * @return {app} for chaining - * @api public - */ - -app.disable = function(setting){ - return this.set(setting, false); -}; - -/** - * Configure callback for zero or more envs, - * when no `env` is specified that callback will - * be invoked for all environments. Any combination - * can be used multiple times, in any order desired. - * - * Examples: - * - * app.configure(function(){ - * // executed for all envs - * }); - * - * app.configure('stage', function(){ - * // executed staging env - * }); - * - * app.configure('stage', 'production', function(){ - * // executed for stage and production - * }); - * - * Note: - * - * These callbacks are invoked immediately, and - * are effectively sugar for the following: - * - * var env = process.env.NODE_ENV || 'development'; - * - * switch (env) { - * case 'development': - * ... - * break; - * case 'stage': - * ... - * break; - * case 'production': - * ... - * break; - * } - * - * @param {String} env... - * @param {Function} fn - * @return {app} for chaining - * @api public - */ - -app.configure = function(env, fn){ - var envs = 'all' - , args = [].slice.call(arguments); - fn = args.pop(); - if (args.length) envs = args; - if ('all' == envs || ~envs.indexOf(this.settings.env)) fn.call(this); - return this; -}; - -/** - * Delegate `.VERB(...)` calls to `router.VERB(...)`. - */ - -methods.forEach(function(method){ - app[method] = function(path){ - if ('get' == method && 1 == arguments.length) return this.set(path); - - // if no router attached yet, attach the router - if (!this._usedRouter) this.use(this.router); - - // setup route - this._router[method].apply(this._router, arguments); - return this; - }; -}); - -/** - * Special-cased "all" method, applying the given route `path`, - * middleware, and callback to _every_ HTTP method. - * - * @param {String} path - * @param {Function} ... - * @return {app} for chaining - * @api public - */ - -app.all = function(path){ - var args = arguments; - methods.forEach(function(method){ - app[method].apply(this, args); - }, this); - return this; -}; - -// del -> delete alias - -app.del = app.delete; - -/** - * Render the given view `name` name with `options` - * and a callback accepting an error and the - * rendered template string. - * - * Example: - * - * app.render('email', { name: 'Tobi' }, function(err, html){ - * // ... - * }) - * - * @param {String} name - * @param {String|Function} options or fn - * @param {Function} fn - * @api public - */ - -app.render = function(name, options, fn){ - var opts = {} - , cache = this.cache - , engines = this.engines - , view; - - // support callback function as second arg - if ('function' == typeof options) { - fn = options, options = {}; - } - - // merge app.locals - utils.merge(opts, this.locals); - - // merge options._locals - if (options._locals) utils.merge(opts, options._locals); - - // merge options - utils.merge(opts, options); - - // set .cache unless explicitly provided - opts.cache = null == opts.cache - ? this.enabled('view cache') - : opts.cache; - - // primed cache - if (opts.cache) view = cache[name]; - - // view - if (!view) { - view = new View(name, { - defaultEngine: this.get('view engine'), - root: this.get('views'), - engines: engines - }); - - if (!view.path) { - var err = new Error('Failed to lookup view "' + name + '"'); - err.view = view; - return fn(err); - } - - // prime the cache - if (opts.cache) cache[name] = view; - } - - // render - try { - view.render(opts, fn); - } catch (err) { - fn(err); - } -}; - -/** - * Listen for connections. - * - * A node `http.Server` is returned, with this - * application (which is a `Function`) as its - * callback. If you wish to create both an HTTP - * and HTTPS server you may do so with the "http" - * and "https" modules as shown here: - * - * var http = require('http') - * , https = require('https') - * , express = require('express') - * , app = express(); - * - * http.createServer(app).listen(80); - * https.createServer({ ... }, app).listen(443); - * - * @return {http.Server} - * @api public - */ - -app.listen = function(){ - var server = http.createServer(this); - return server.listen.apply(server, arguments); -}; diff --git a/node_modules/express/lib/express.js b/node_modules/express/lib/express.js deleted file mode 100644 index d0b7a12..0000000 --- a/node_modules/express/lib/express.js +++ /dev/null @@ -1,92 +0,0 @@ -/** - * Module dependencies. - */ - -var connect = require('connect') - , proto = require('./application') - , Route = require('./router/route') - , Router = require('./router') - , req = require('./request') - , res = require('./response') - , utils = connect.utils; - -/** - * Expose `createApplication()`. - */ - -exports = module.exports = createApplication; - -/** - * Framework version. - */ - -exports.version = '3.1.2'; - -/** - * Expose mime. - */ - -exports.mime = connect.mime; - -/** - * Create an express application. - * - * @return {Function} - * @api public - */ - -function createApplication() { - var app = connect(); - utils.merge(app, proto); - app.request = { __proto__: req }; - app.response = { __proto__: res }; - app.init(); - return app; -} - -/** - * Expose connect.middleware as express.* - * for example `express.logger` etc. - */ - -for (var key in connect.middleware) { - Object.defineProperty( - exports - , key - , Object.getOwnPropertyDescriptor(connect.middleware, key)); -} - -/** - * Error on createServer(). - */ - -exports.createServer = function(){ - console.warn('Warning: express.createServer() is deprecated, express'); - console.warn('applications no longer inherit from http.Server,'); - console.warn('please use:'); - console.warn(''); - console.warn(' var express = require("express");'); - console.warn(' var app = express();'); - console.warn(''); - return createApplication(); -}; - -/** - * Expose the prototypes. - */ - -exports.application = proto; -exports.request = req; -exports.response = res; - -/** - * Expose constructors. - */ - -exports.Route = Route; -exports.Router = Router; - -// Error handler title - -exports.errorHandler.title = 'Express'; - diff --git a/node_modules/express/lib/middleware.js b/node_modules/express/lib/middleware.js deleted file mode 100644 index 308c5bb..0000000 --- a/node_modules/express/lib/middleware.js +++ /dev/null @@ -1,33 +0,0 @@ - -/** - * Module dependencies. - */ - -var utils = require('./utils'); - -/** - * Initialization middleware, exposing the - * request and response to eachother, as well - * as defaulting the X-Powered-By header field. - * - * @param {Function} app - * @return {Function} - * @api private - */ - -exports.init = function(app){ - return function expressInit(req, res, next){ - req.app = res.app = app; - if (app.enabled('x-powered-by')) res.setHeader('X-Powered-By', 'Express'); - req.res = res; - res.req = req; - req.next = next; - - req.__proto__ = app.request; - res.__proto__ = app.response; - - res.locals = res.locals || utils.locals(res); - - next(); - } -}; diff --git a/node_modules/express/lib/request.js b/node_modules/express/lib/request.js deleted file mode 100644 index 616b0af..0000000 --- a/node_modules/express/lib/request.js +++ /dev/null @@ -1,496 +0,0 @@ - -/** - * Module dependencies. - */ - -var http = require('http') - , utils = require('./utils') - , connect = require('connect') - , fresh = require('fresh') - , parseRange = require('range-parser') - , parse = connect.utils.parseUrl - , mime = connect.mime; - -/** - * Request prototype. - */ - -var req = exports = module.exports = { - __proto__: http.IncomingMessage.prototype -}; - -/** - * Return request header. - * - * The `Referrer` header field is special-cased, - * both `Referrer` and `Referer` are interchangeable. - * - * Examples: - * - * req.get('Content-Type'); - * // => "text/plain" - * - * req.get('content-type'); - * // => "text/plain" - * - * req.get('Something'); - * // => undefined - * - * Aliased as `req.header()`. - * - * @param {String} name - * @return {String} - * @api public - */ - -req.get = -req.header = function(name){ - switch (name = name.toLowerCase()) { - case 'referer': - case 'referrer': - return this.headers.referrer - || this.headers.referer; - default: - return this.headers[name]; - } -}; - -/** - * Check if the given `type(s)` is acceptable, returning - * the best match when true, otherwise `undefined`, in which - * case you should respond with 406 "Not Acceptable". - * - * The `type` value may be a single mime type string - * such as "application/json", the extension name - * such as "json", a comma-delimted list such as "json, html, text/plain", - * or an array `["json", "html", "text/plain"]`. When a list - * or array is given the _best_ match, if any is returned. - * - * Examples: - * - * // Accept: text/html - * req.accepts('html'); - * // => "html" - * - * // Accept: text/*, application/json - * req.accepts('html'); - * // => "html" - * req.accepts('text/html'); - * // => "text/html" - * req.accepts('json, text'); - * // => "json" - * req.accepts('application/json'); - * // => "application/json" - * - * // Accept: text/*, application/json - * req.accepts('image/png'); - * req.accepts('png'); - * // => undefined - * - * // Accept: text/*;q=.5, application/json - * req.accepts(['html', 'json']); - * req.accepts('html, json'); - * // => "json" - * - * @param {String|Array} type(s) - * @return {String} - * @api public - */ - -req.accepts = function(type){ - return utils.accepts(type, this.get('Accept')); -}; - -/** - * Check if the given `charset` is acceptable, - * otherwise you should respond with 406 "Not Acceptable". - * - * @param {String} charset - * @return {Boolean} - * @api public - */ - -req.acceptsCharset = function(charset){ - var accepted = this.acceptedCharsets; - return accepted.length - ? ~accepted.indexOf(charset) - : true; -}; - -/** - * Check if the given `lang` is acceptable, - * otherwise you should respond with 406 "Not Acceptable". - * - * @param {String} lang - * @return {Boolean} - * @api public - */ - -req.acceptsLanguage = function(lang){ - var accepted = this.acceptedLanguages; - return accepted.length - ? ~accepted.indexOf(lang) - : true; -}; - -/** - * Parse Range header field, - * capping to the given `size`. - * - * Unspecified ranges such as "0-" require - * knowledge of your resource length. In - * the case of a byte range this is of course - * the total number of bytes. If the Range - * header field is not given `null` is returned, - * `-1` when unsatisfiable, `-2` when syntactically invalid. - * - * NOTE: remember that ranges are inclusive, so - * for example "Range: users=0-3" should respond - * with 4 users when available, not 3. - * - * @param {Number} size - * @return {Array} - * @api public - */ - -req.range = function(size){ - var range = this.get('Range'); - if (!range) return; - return parseRange(size, range); -}; - -/** - * Return an array of Accepted media types - * ordered from highest quality to lowest. - * - * Examples: - * - * [ { value: 'application/json', - * quality: 1, - * type: 'application', - * subtype: 'json' }, - * { value: 'text/html', - * quality: 0.5, - * type: 'text', - * subtype: 'html' } ] - * - * @return {Array} - * @api public - */ - -req.__defineGetter__('accepted', function(){ - var accept = this.get('Accept'); - return accept - ? utils.parseAccept(accept) - : []; -}); - -/** - * Return an array of Accepted languages - * ordered from highest quality to lowest. - * - * Examples: - * - * Accept-Language: en;q=.5, en-us - * ['en-us', 'en'] - * - * @return {Array} - * @api public - */ - -req.__defineGetter__('acceptedLanguages', function(){ - var accept = this.get('Accept-Language'); - return accept - ? utils - .parseParams(accept) - .map(function(obj){ - return obj.value; - }) - : []; -}); - -/** - * Return an array of Accepted charsets - * ordered from highest quality to lowest. - * - * Examples: - * - * Accept-Charset: iso-8859-5;q=.2, unicode-1-1;q=0.8 - * ['unicode-1-1', 'iso-8859-5'] - * - * @return {Array} - * @api public - */ - -req.__defineGetter__('acceptedCharsets', function(){ - var accept = this.get('Accept-Charset'); - return accept - ? utils - .parseParams(accept) - .map(function(obj){ - return obj.value; - }) - : []; -}); - -/** - * Return the value of param `name` when present or `defaultValue`. - * - * - Checks route placeholders, ex: _/user/:id_ - * - Checks body params, ex: id=12, {"id":12} - * - Checks query string params, ex: ?id=12 - * - * To utilize request bodies, `req.body` - * should be an object. This can be done by using - * the `connect.bodyParser()` middleware. - * - * @param {String} name - * @param {Mixed} [defaultValue] - * @return {String} - * @api public - */ - -req.param = function(name, defaultValue){ - var params = this.params || {}; - var body = this.body || {}; - var query = this.query || {}; - if (null != params[name] && params.hasOwnProperty(name)) return params[name]; - if (null != body[name]) return body[name]; - if (null != query[name]) return query[name]; - return defaultValue; -}; - -/** - * Check if the incoming request contains the "Content-Type" - * header field, and it contains the give mime `type`. - * - * Examples: - * - * // With Content-Type: text/html; charset=utf-8 - * req.is('html'); - * req.is('text/html'); - * req.is('text/*'); - * // => true - * - * // When Content-Type is application/json - * req.is('json'); - * req.is('application/json'); - * req.is('application/*'); - * // => true - * - * req.is('html'); - * // => false - * - * @param {String} type - * @return {Boolean} - * @api public - */ - -req.is = function(type){ - var ct = this.get('Content-Type'); - if (!ct) return false; - ct = ct.split(';')[0]; - if (!~type.indexOf('/')) type = mime.lookup(type); - if (~type.indexOf('*')) { - type = type.split('/'); - ct = ct.split('/'); - if ('*' == type[0] && type[1] == ct[1]) return true; - if ('*' == type[1] && type[0] == ct[0]) return true; - return false; - } - return !! ~ct.indexOf(type); -}; - -/** - * Return the protocol string "http" or "https" - * when requested with TLS. When the "trust proxy" - * setting is enabled the "X-Forwarded-Proto" header - * field will be trusted. If you're running behind - * a reverse proxy that supplies https for you this - * may be enabled. - * - * @return {String} - * @api public - */ - -req.__defineGetter__('protocol', function(){ - var trustProxy = this.app.get('trust proxy'); - return this.connection.encrypted - ? 'https' - : trustProxy - ? (this.get('X-Forwarded-Proto') || 'http') - : 'http'; -}); - -/** - * Short-hand for: - * - * req.protocol == 'https' - * - * @return {Boolean} - * @api public - */ - -req.__defineGetter__('secure', function(){ - return 'https' == this.protocol; -}); - -/** - * Return the remote address, or when - * "trust proxy" is `true` return - * the upstream addr. - * - * @return {String} - * @api public - */ - -req.__defineGetter__('ip', function(){ - return this.ips[0] || this.connection.remoteAddress; -}); - -/** - * When "trust proxy" is `true`, parse - * the "X-Forwarded-For" ip address list. - * - * For example if the value were "client, proxy1, proxy2" - * you would receive the array `["client", "proxy1", "proxy2"]` - * where "proxy2" is the furthest down-stream. - * - * @return {Array} - * @api public - */ - -req.__defineGetter__('ips', function(){ - var trustProxy = this.app.get('trust proxy'); - var val = this.get('X-Forwarded-For'); - return trustProxy && val - ? val.split(/ *, */) - : []; -}); - -/** - * Return basic auth credentials. - * - * Examples: - * - * // http://tobi:hello@example.com - * req.auth - * // => { username: 'tobi', password: 'hello' } - * - * @return {Object} or undefined - * @api public - */ - -req.__defineGetter__('auth', function(){ - // missing - var auth = this.get('Authorization'); - if (!auth) return; - - // malformed - var parts = auth.split(' '); - if ('basic' != parts[0].toLowerCase()) return; - if (!parts[1]) return; - auth = parts[1]; - - // credentials - auth = new Buffer(auth, 'base64').toString().match(/^([^:]*):(.*)$/); - if (!auth) return; - return { username: auth[1], password: auth[2] }; -}); - -/** - * Return subdomains as an array. - * - * Subdomains are the dot-separated parts of the host before the main domain of - * the app. By default, the domain of the app is assumed to be the last two - * parts of the host. This can be changed by setting "subdomain offset". - * - * For example, if the domain is "tobi.ferrets.example.com": - * If "subdomain offset" is not set, req.subdomains is `["ferrets", "tobi"]`. - * If "subdomain offset" is 3, req.subdomains is `["tobi"]`. - * - * @return {Array} - * @api public - */ - -req.__defineGetter__('subdomains', function(){ - var offset = this.app.get('subdomain offset'); - return this.get('Host') - .split('.') - .reverse() - .slice(offset); -}); - -/** - * Short-hand for `url.parse(req.url).pathname`. - * - * @return {String} - * @api public - */ - -req.__defineGetter__('path', function(){ - return parse(this).pathname; -}); - -/** - * Parse the "Host" header field hostname. - * - * @return {String} - * @api public - */ - -req.__defineGetter__('host', function(){ - var trustProxy = this.app.get('trust proxy'); - var host = trustProxy && this.get('X-Forwarded-Host'); - host = host || this.get('Host'); - return host.split(':')[0]; -}); - -/** - * Check if the request is fresh, aka - * Last-Modified and/or the ETag - * still match. - * - * @return {Boolean} - * @api public - */ - -req.__defineGetter__('fresh', function(){ - var method = this.method; - var s = this.res.statusCode; - - // GET or HEAD for weak freshness validation only - if ('GET' != method && 'HEAD' != method) return false; - - // 2xx or 304 as per rfc2616 14.26 - if ((s >= 200 && s < 300) || 304 == s) { - return fresh(this.headers, this.res._headers); - } - - return false; -}); - -/** - * Check if the request is stale, aka - * "Last-Modified" and / or the "ETag" for the - * resource has changed. - * - * @return {Boolean} - * @api public - */ - -req.__defineGetter__('stale', function(){ - return !this.fresh; -}); - -/** - * Check if the request was an _XMLHttpRequest_. - * - * @return {Boolean} - * @api public - */ - -req.__defineGetter__('xhr', function(){ - var val = this.get('X-Requested-With') || ''; - return 'xmlhttprequest' == val.toLowerCase(); -}); diff --git a/node_modules/express/lib/response.js b/node_modules/express/lib/response.js deleted file mode 100644 index 50fcd05..0000000 --- a/node_modules/express/lib/response.js +++ /dev/null @@ -1,756 +0,0 @@ -/** - * Module dependencies. - */ - -var http = require('http') - , path = require('path') - , connect = require('connect') - , utils = connect.utils - , sign = require('cookie-signature').sign - , normalizeType = require('./utils').normalizeType - , normalizeTypes = require('./utils').normalizeTypes - , etag = require('./utils').etag - , statusCodes = http.STATUS_CODES - , cookie = require('cookie') - , send = require('send') - , mime = connect.mime - , basename = path.basename - , extname = path.extname - , join = path.join; - -/** - * Response prototype. - */ - -var res = module.exports = { - __proto__: http.ServerResponse.prototype -}; - -/** - * Set status `code`. - * - * @param {Number} code - * @return {ServerResponse} - * @api public - */ - -res.status = function(code){ - this.statusCode = code; - return this; -}; - -/** - * Set Link header field with the given `links`. - * - * Examples: - * - * res.links({ - * next: 'http://api.example.com/users?page=2', - * last: 'http://api.example.com/users?page=5' - * }); - * - * @param {Object} links - * @return {ServerResponse} - * @api public - */ - -res.links = function(links){ - return this.set('Link', Object.keys(links).map(function(rel){ - return '<' + links[rel] + '>; rel="' + rel + '"'; - }).join(', ')); -}; - -/** - * Send a response. - * - * Examples: - * - * res.send(new Buffer('wahoo')); - * res.send({ some: 'json' }); - * res.send('

some html

'); - * res.send(404, 'Sorry, cant find that'); - * res.send(404); - * - * @param {Mixed} body or status - * @param {Mixed} body - * @return {ServerResponse} - * @api public - */ - -res.send = function(body){ - var req = this.req - , head = 'HEAD' == req.method - , len; - - // allow status / body - if (2 == arguments.length) { - // res.send(body, status) backwards compat - if ('number' != typeof body && 'number' == typeof arguments[1]) { - this.statusCode = arguments[1]; - } else { - this.statusCode = body; - body = arguments[1]; - } - } - - switch (typeof body) { - // response status - case 'number': - this.get('Content-Type') || this.type('txt'); - this.statusCode = body; - body = http.STATUS_CODES[body]; - break; - // string defaulting to html - case 'string': - if (!this.get('Content-Type')) { - this.charset = this.charset || 'utf-8'; - this.type('html'); - } - break; - case 'boolean': - case 'object': - if (null == body) { - body = ''; - } else if (Buffer.isBuffer(body)) { - this.get('Content-Type') || this.type('bin'); - } else { - return this.json(body); - } - break; - } - - // populate Content-Length - if (undefined !== body && !this.get('Content-Length')) { - this.set('Content-Length', len = Buffer.isBuffer(body) - ? body.length - : Buffer.byteLength(body)); - } - - // ETag support - // TODO: W/ support - if (len > 1024) { - if (!this.get('ETag')) { - this.set('ETag', etag(body)); - } - } - - // freshness - if (req.fresh) this.statusCode = 304; - - // strip irrelevant headers - if (204 == this.statusCode || 304 == this.statusCode) { - this.removeHeader('Content-Type'); - this.removeHeader('Content-Length'); - this.removeHeader('Transfer-Encoding'); - body = ''; - } - - // respond - this.end(head ? null : body); - return this; -}; - -/** - * Send JSON response. - * - * Examples: - * - * res.json(null); - * res.json({ user: 'tj' }); - * res.json(500, 'oh noes!'); - * res.json(404, 'I dont have that'); - * - * @param {Mixed} obj or status - * @param {Mixed} obj - * @return {ServerResponse} - * @api public - */ - -res.json = function(obj){ - // allow status / body - if (2 == arguments.length) { - // res.json(body, status) backwards compat - if ('number' == typeof arguments[1]) { - this.statusCode = arguments[1]; - } else { - this.statusCode = obj; - obj = arguments[1]; - } - } - - // settings - var app = this.app; - var replacer = app.get('json replacer'); - var spaces = app.get('json spaces'); - var body = JSON.stringify(obj, replacer, spaces); - - // content-type - this.charset = this.charset || 'utf-8'; - this.get('Content-Type') || this.set('Content-Type', 'application/json'); - - return this.send(body); -}; - -/** - * Send JSON response with JSONP callback support. - * - * Examples: - * - * res.jsonp(null); - * res.jsonp({ user: 'tj' }); - * res.jsonp(500, 'oh noes!'); - * res.jsonp(404, 'I dont have that'); - * - * @param {Mixed} obj or status - * @param {Mixed} obj - * @return {ServerResponse} - * @api public - */ - -res.jsonp = function(obj){ - // allow status / body - if (2 == arguments.length) { - // res.json(body, status) backwards compat - if ('number' == typeof arguments[1]) { - this.statusCode = arguments[1]; - } else { - this.statusCode = obj; - obj = arguments[1]; - } - } - - // settings - var app = this.app; - var replacer = app.get('json replacer'); - var spaces = app.get('json spaces'); - var body = JSON.stringify(obj, replacer, spaces) - .replace(/\u2028/g, '\\u2028') - .replace(/\u2029/g, '\\u2029'); - var callback = this.req.query[app.get('jsonp callback name')]; - - // content-type - this.charset = this.charset || 'utf-8'; - this.set('Content-Type', 'application/json'); - - // jsonp - if (callback) { - this.set('Content-Type', 'text/javascript'); - var cb = callback.replace(/[^\[\]\w$.]/g, ''); - body = cb + ' && ' + cb + '(' + body + ');'; - } - - return this.send(body); -}; - -/** - * Transfer the file at the given `path`. - * - * Automatically sets the _Content-Type_ response header field. - * The callback `fn(err)` is invoked when the transfer is complete - * or when an error occurs. Be sure to check `res.sentHeader` - * if you wish to attempt responding, as the header and some data - * may have already been transferred. - * - * Options: - * - * - `maxAge` defaulting to 0 - * - `root` root directory for relative filenames - * - * Examples: - * - * The following example illustrates how `res.sendfile()` may - * be used as an alternative for the `static()` middleware for - * dynamic situations. The code backing `res.sendfile()` is actually - * the same code, so HTTP cache support etc is identical. - * - * app.get('/user/:uid/photos/:file', function(req, res){ - * var uid = req.params.uid - * , file = req.params.file; - * - * req.user.mayViewFilesFrom(uid, function(yes){ - * if (yes) { - * res.sendfile('/uploads/' + uid + '/' + file); - * } else { - * res.send(403, 'Sorry! you cant see that.'); - * } - * }); - * }); - * - * @param {String} path - * @param {Object|Function} options or fn - * @param {Function} fn - * @api public - */ - -res.sendfile = function(path, options, fn){ - var self = this - , req = self.req - , next = this.req.next - , options = options || {} - , done; - - // support function as second arg - if ('function' == typeof options) { - fn = options; - options = {}; - } - - // socket errors - req.socket.on('error', error); - - // errors - function error(err) { - if (done) return; - done = true; - - // clean up - cleanup(); - if (!self.headerSent) self.removeHeader('Content-Disposition'); - - // callback available - if (fn) return fn(err); - - // list in limbo if there's no callback - if (self.headerSent) return; - - // delegate - next(err); - } - - // streaming - function stream() { - if (done) return; - cleanup(); - if (fn) self.on('finish', fn); - } - - // cleanup - function cleanup() { - req.socket.removeListener('error', error); - } - - // transfer - var file = send(req, path); - if (options.root) file.root(options.root); - file.maxage(options.maxAge || 0); - file.on('error', error); - file.on('directory', next); - file.on('stream', stream); - file.pipe(this); - this.on('finish', cleanup); -}; - -/** - * Transfer the file at the given `path` as an attachment. - * - * Optionally providing an alternate attachment `filename`, - * and optional callback `fn(err)`. The callback is invoked - * when the data transfer is complete, or when an error has - * ocurred. Be sure to check `res.headerSent` if you plan to respond. - * - * This method uses `res.sendfile()`. - * - * @param {String} path - * @param {String|Function} filename or fn - * @param {Function} fn - * @api public - */ - -res.download = function(path, filename, fn){ - // support function as second arg - if ('function' == typeof filename) { - fn = filename; - filename = null; - } - - filename = filename || path; - this.set('Content-Disposition', 'attachment; filename="' + basename(filename) + '"'); - return this.sendfile(path, fn); -}; - -/** - * Set _Content-Type_ response header with `type` through `mime.lookup()` - * when it does not contain "/", or set the Content-Type to `type` otherwise. - * - * Examples: - * - * res.type('.html'); - * res.type('html'); - * res.type('json'); - * res.type('application/json'); - * res.type('png'); - * - * @param {String} type - * @return {ServerResponse} for chaining - * @api public - */ - -res.contentType = -res.type = function(type){ - return this.set('Content-Type', ~type.indexOf('/') - ? type - : mime.lookup(type)); -}; - -/** - * Respond to the Acceptable formats using an `obj` - * of mime-type callbacks. - * - * This method uses `req.accepted`, an array of - * acceptable types ordered by their quality values. - * When "Accept" is not present the _first_ callback - * is invoked, otherwise the first match is used. When - * no match is performed the server responds with - * 406 "Not Acceptable". - * - * Content-Type is set for you, however if you choose - * you may alter this within the callback using `res.type()` - * or `res.set('Content-Type', ...)`. - * - * res.format({ - * 'text/plain': function(){ - * res.send('hey'); - * }, - * - * 'text/html': function(){ - * res.send('

hey

'); - * }, - * - * 'appliation/json': function(){ - * res.send({ message: 'hey' }); - * } - * }); - * - * In addition to canonicalized MIME types you may - * also use extnames mapped to these types: - * - * res.format({ - * text: function(){ - * res.send('hey'); - * }, - * - * html: function(){ - * res.send('

hey

'); - * }, - * - * json: function(){ - * res.send({ message: 'hey' }); - * } - * }); - * - * By default Express passes an `Error` - * with a `.status` of 406 to `next(err)` - * if a match is not made. If you provide - * a `.default` callback it will be invoked - * instead. - * - * @param {Object} obj - * @return {ServerResponse} for chaining - * @api public - */ - -res.format = function(obj){ - var req = this.req - , next = req.next; - - var fn = obj.default; - if (fn) delete obj.default; - var keys = Object.keys(obj); - - var key = req.accepts(keys); - - this.set('Vary', 'Accept'); - - if (key) { - this.set('Content-Type', normalizeType(key).value); - obj[key](req, this, next); - } else if (fn) { - fn(); - } else { - var err = new Error('Not Acceptable'); - err.status = 406; - err.types = normalizeTypes(keys).map(function(o){ return o.value }); - next(err); - } - - return this; -}; - -/** - * Set _Content-Disposition_ header to _attachment_ with optional `filename`. - * - * @param {String} filename - * @return {ServerResponse} - * @api public - */ - -res.attachment = function(filename){ - if (filename) this.type(extname(filename)); - this.set('Content-Disposition', filename - ? 'attachment; filename="' + basename(filename) + '"' - : 'attachment'); - return this; -}; - -/** - * Set header `field` to `val`, or pass - * an object of header fields. - * - * Examples: - * - * res.set('Foo', ['bar', 'baz']); - * res.set('Accept', 'application/json'); - * res.set({ Accept: 'text/plain', 'X-API-Key': 'tobi' }); - * - * Aliased as `res.header()`. - * - * @param {String|Object|Array} field - * @param {String} val - * @return {ServerResponse} for chaining - * @api public - */ - -res.set = -res.header = function(field, val){ - if (2 == arguments.length) { - if (Array.isArray(val)) val = val.map(String); - else val = String(val); - this.setHeader(field, val); - } else { - for (var key in field) { - this.set(key, field[key]); - } - } - return this; -}; - -/** - * Get value for header `field`. - * - * @param {String} field - * @return {String} - * @api public - */ - -res.get = function(field){ - return this.getHeader(field); -}; - -/** - * Clear cookie `name`. - * - * @param {String} name - * @param {Object} options - * @param {ServerResponse} for chaining - * @api public - */ - -res.clearCookie = function(name, options){ - var opts = { expires: new Date(1), path: '/' }; - return this.cookie(name, '', options - ? utils.merge(opts, options) - : opts); -}; - -/** - * Set cookie `name` to `val`, with the given `options`. - * - * Options: - * - * - `maxAge` max-age in milliseconds, converted to `expires` - * - `signed` sign the cookie - * - `path` defaults to "/" - * - * Examples: - * - * // "Remember Me" for 15 minutes - * res.cookie('rememberme', '1', { expires: new Date(Date.now() + 900000), httpOnly: true }); - * - * // save as above - * res.cookie('rememberme', '1', { maxAge: 900000, httpOnly: true }) - * - * @param {String} name - * @param {String|Object} val - * @param {Options} options - * @api public - */ - -res.cookie = function(name, val, options){ - options = utils.merge({}, options); - var secret = this.req.secret; - var signed = options.signed; - if (signed && !secret) throw new Error('connect.cookieParser("secret") required for signed cookies'); - if ('object' == typeof val) val = 'j:' + JSON.stringify(val); - if (signed) val = 's:' + sign(val, secret); - if ('maxAge' in options) { - options.expires = new Date(Date.now() + options.maxAge); - options.maxAge /= 1000; - } - if (null == options.path) options.path = '/'; - this.set('Set-Cookie', cookie.serialize(name, String(val), options)); - return this; -}; - - -/** - * Set the location header to `url`. - * - * The given `url` can also be the name of a mapped url, for - * example by default express supports "back" which redirects - * to the _Referrer_ or _Referer_ headers or "/". - * - * Examples: - * - * res.location('/foo/bar').; - * res.location('http://example.com'); - * res.location('../login'); // /blog/post/1 -> /blog/login - * - * Mounting: - * - * When an application is mounted and `res.location()` - * is given a path that does _not_ lead with "/" it becomes - * relative to the mount-point. For example if the application - * is mounted at "/blog", the following would become "/blog/login". - * - * res.location('login'); - * - * While the leading slash would result in a location of "/login": - * - * res.location('/login'); - * - * @param {String} url - * @api public - */ - -res.location = function(url){ - var app = this.app - , req = this.req; - - // setup redirect map - var map = { back: req.get('Referrer') || '/' }; - - // perform redirect - url = map[url] || url; - - // relative - if (!~url.indexOf('://') && 0 != url.indexOf('//')) { - var path - - // relative to path - if ('.' == url[0]) { - path = req.originalUrl.split('?')[0] - url = path + ('/' == path[path.length - 1] ? '' : '/') + url; - // relative to mount-point - } else if ('/' != url[0]) { - path = app.path(); - url = path + '/' + url; - } - } - - // Respond - this.set('Location', url); - return this; -}; - -/** - * Redirect to the given `url` with optional response `status` - * defaulting to 302. - * - * The resulting `url` is determined by `res.location()`, so - * it will play nicely with mounted apps, relative paths, - * `"back"` etc. - * - * Examples: - * - * res.redirect('/foo/bar'); - * res.redirect('http://example.com'); - * res.redirect(301, 'http://example.com'); - * res.redirect('http://example.com', 301); - * res.redirect('../login'); // /blog/post/1 -> /blog/login - * - * @param {String} url - * @param {Number} code - * @api public - */ - -res.redirect = function(url){ - var app = this.app - , head = 'HEAD' == this.req.method - , status = 302 - , body; - - // allow status / url - if (2 == arguments.length) { - if ('number' == typeof url) { - status = url; - url = arguments[1]; - } else { - status = arguments[1]; - } - } - - // Set location header - this.location(url); - url = this.get('Location'); - - // Support text/{plain,html} by default - this.format({ - text: function(){ - body = statusCodes[status] + '. Redirecting to ' + encodeURI(url); - }, - - html: function(){ - var u = utils.escape(url); - body = '

' + statusCodes[status] + '. Redirecting to ' + u + '

'; - }, - - default: function(){ - body = ''; - } - }); - - // Respond - this.statusCode = status; - this.set('Content-Length', Buffer.byteLength(body)); - this.end(head ? null : body); -}; - -/** - * Render `view` with the given `options` and optional callback `fn`. - * When a callback function is given a response will _not_ be made - * automatically, otherwise a response of _200_ and _text/html_ is given. - * - * Options: - * - * - `cache` boolean hinting to the engine it should cache - * - `filename` filename of the view being rendered - * - * @param {String} view - * @param {Object|Function} options or callback function - * @param {Function} fn - * @api public - */ - -res.render = function(view, options, fn){ - var self = this - , options = options || {} - , req = this.req - , app = req.app; - - // support callback function as second arg - if ('function' == typeof options) { - fn = options, options = {}; - } - - // merge res.locals - options._locals = self.locals; - - // default callback to respond - fn = fn || function(err, str){ - if (err) return req.next(err); - self.send(str); - }; - - // render - app.render(view, options, fn); -}; diff --git a/node_modules/express/lib/router/index.js b/node_modules/express/lib/router/index.js deleted file mode 100644 index 662dc29..0000000 --- a/node_modules/express/lib/router/index.js +++ /dev/null @@ -1,273 +0,0 @@ -/** - * Module dependencies. - */ - -var Route = require('./route') - , utils = require('../utils') - , methods = require('methods') - , debug = require('debug')('express:router') - , parse = require('connect').utils.parseUrl; - -/** - * Expose `Router` constructor. - */ - -exports = module.exports = Router; - -/** - * Initialize a new `Router` with the given `options`. - * - * @param {Object} options - * @api private - */ - -function Router(options) { - options = options || {}; - var self = this; - this.map = {}; - this.params = {}; - this._params = []; - this.caseSensitive = options.caseSensitive; - this.strict = options.strict; - this.middleware = function router(req, res, next){ - self._dispatch(req, res, next); - }; -} - -/** - * Register a param callback `fn` for the given `name`. - * - * @param {String|Function} name - * @param {Function} fn - * @return {Router} for chaining - * @api public - */ - -Router.prototype.param = function(name, fn){ - // param logic - if ('function' == typeof name) { - this._params.push(name); - return; - } - - // apply param functions - var params = this._params - , len = params.length - , ret; - - for (var i = 0; i < len; ++i) { - if (ret = params[i](name, fn)) { - fn = ret; - } - } - - // ensure we end up with a - // middleware function - if ('function' != typeof fn) { - throw new Error('invalid param() call for ' + name + ', got ' + fn); - } - - (this.params[name] = this.params[name] || []).push(fn); - return this; -}; - -/** - * Route dispatcher aka the route "middleware". - * - * @param {IncomingMessage} req - * @param {ServerResponse} res - * @param {Function} next - * @api private - */ - -Router.prototype._dispatch = function(req, res, next){ - var params = this.params - , self = this; - - debug('dispatching %s %s (%s)', req.method, req.url, req.originalUrl); - - // route dispatch - (function pass(i, err){ - var paramCallbacks - , paramIndex = 0 - , paramVal - , route - , keys - , key; - - // match next route - function nextRoute(err) { - pass(req._route_index + 1, err); - } - - // match route - req.route = route = self.matchRequest(req, i); - - // no route - if (!route) return next(err); - debug('matched %s %s', route.method, route.path); - - // we have a route - // start at param 0 - req.params = route.params; - keys = route.keys; - i = 0; - - // param callbacks - function param(err) { - paramIndex = 0; - key = keys[i++]; - paramVal = key && req.params[key.name]; - paramCallbacks = key && params[key.name]; - - try { - if ('route' == err) { - nextRoute(); - } else if (err) { - i = 0; - callbacks(err); - } else if (paramCallbacks && undefined !== paramVal) { - paramCallback(); - } else if (key) { - param(); - } else { - i = 0; - callbacks(); - } - } catch (err) { - param(err); - } - }; - - param(err); - - // single param callbacks - function paramCallback(err) { - var fn = paramCallbacks[paramIndex++]; - if (err || !fn) return param(err); - fn(req, res, paramCallback, paramVal, key.name); - } - - // invoke route callbacks - function callbacks(err) { - var fn = route.callbacks[i++]; - try { - if ('route' == err) { - nextRoute(); - } else if (err && fn) { - if (fn.length < 4) return callbacks(err); - fn(err, req, res, callbacks); - } else if (fn) { - if (fn.length < 4) return fn(req, res, callbacks); - callbacks(); - } else { - nextRoute(err); - } - } catch (err) { - callbacks(err); - } - } - })(0); -}; - -/** - * Attempt to match a route for `req` - * with optional starting index of `i` - * defaulting to 0. - * - * @param {IncomingMessage} req - * @param {Number} i - * @return {Route} - * @api private - */ - -Router.prototype.matchRequest = function(req, i, head){ - var method = req.method.toLowerCase() - , url = parse(req) - , path = url.pathname - , routes = this.map - , i = i || 0 - , route; - - // HEAD support - if (!head && 'head' == method) { - route = this.matchRequest(req, i, true); - if (route) return route; - method = 'get'; - } - - // routes for this method - if (routes = routes[method]) { - - // matching routes - for (var len = routes.length; i < len; ++i) { - route = routes[i]; - if (route.match(path)) { - req._route_index = i; - return route; - } - } - } -}; - -/** - * Attempt to match a route for `method` - * and `url` with optional starting - * index of `i` defaulting to 0. - * - * @param {String} method - * @param {String} url - * @param {Number} i - * @return {Route} - * @api private - */ - -Router.prototype.match = function(method, url, i, head){ - var req = { method: method, url: url }; - return this.matchRequest(req, i, head); -}; - -/** - * Route `method`, `path`, and one or more callbacks. - * - * @param {String} method - * @param {String} path - * @param {Function} callback... - * @return {Router} for chaining - * @api private - */ - -Router.prototype.route = function(method, path, callbacks){ - var method = method.toLowerCase() - , callbacks = utils.flatten([].slice.call(arguments, 2)); - - // ensure path was given - if (!path) throw new Error('Router#' + method + '() requires a path'); - - // ensure all callbacks are functions - callbacks.forEach(function(fn, i){ - if ('function' == typeof fn) return; - var type = {}.toString.call(fn); - var msg = '.' + method + '() requires callback functions but got a ' + type; - throw new Error(msg); - }); - - // create the route - debug('defined %s %s', method, path); - var route = new Route(method, path, callbacks, { - sensitive: this.caseSensitive, - strict: this.strict - }); - - // add it - (this.map[method] = this.map[method] || []).push(route); - return this; -}; - -methods.forEach(function(method){ - Router.prototype[method] = function(path){ - var args = [method].concat([].slice.call(arguments)); - this.route.apply(this, args); - return this; - }; -}); diff --git a/node_modules/express/lib/router/route.js b/node_modules/express/lib/router/route.js deleted file mode 100644 index c1a0b5e..0000000 --- a/node_modules/express/lib/router/route.js +++ /dev/null @@ -1,72 +0,0 @@ - -/** - * Module dependencies. - */ - -var utils = require('../utils'); - -/** - * Expose `Route`. - */ - -module.exports = Route; - -/** - * Initialize `Route` with the given HTTP `method`, `path`, - * and an array of `callbacks` and `options`. - * - * Options: - * - * - `sensitive` enable case-sensitive routes - * - `strict` enable strict matching for trailing slashes - * - * @param {String} method - * @param {String} path - * @param {Array} callbacks - * @param {Object} options. - * @api private - */ - -function Route(method, path, callbacks, options) { - options = options || {}; - this.path = path; - this.method = method; - this.callbacks = callbacks; - this.regexp = utils.pathRegexp(path - , this.keys = [] - , options.sensitive - , options.strict); -} - -/** - * Check if this route matches `path`, if so - * populate `.params`. - * - * @param {String} path - * @return {Boolean} - * @api private - */ - -Route.prototype.match = function(path){ - var keys = this.keys - , params = this.params = [] - , m = this.regexp.exec(path); - - if (!m) return false; - - for (var i = 1, len = m.length; i < len; ++i) { - var key = keys[i - 1]; - - var val = 'string' == typeof m[i] - ? decodeURIComponent(m[i]) - : m[i]; - - if (key) { - params[key.name] = val; - } else { - params.push(val); - } - } - - return true; -}; diff --git a/node_modules/express/lib/utils.js b/node_modules/express/lib/utils.js deleted file mode 100644 index 5e8fd1f..0000000 --- a/node_modules/express/lib/utils.js +++ /dev/null @@ -1,302 +0,0 @@ - -/** - * Module dependencies. - */ - -var mime = require('connect').mime - , crc32 = require('buffer-crc32'); - -/** - * Return ETag for `body`. - * - * @param {String|Buffer} body - * @return {String} - * @api private - */ - -exports.etag = function(body){ - return '"' + crc32.signed(body) + '"'; -}; - -/** - * Make `locals()` bound to the given `obj`. - * - * This is used for `app.locals` and `res.locals`. - * - * @param {Object} obj - * @return {Function} - * @api private - */ - -exports.locals = function(obj){ - function locals(obj){ - for (var key in obj) locals[key] = obj[key]; - return obj; - }; - - return locals; -}; - -/** - * Check if `path` looks absolute. - * - * @param {String} path - * @return {Boolean} - * @api private - */ - -exports.isAbsolute = function(path){ - if ('/' == path[0]) return true; - if (':' == path[1] && '\\' == path[2]) return true; -}; - -/** - * Flatten the given `arr`. - * - * @param {Array} arr - * @return {Array} - * @api private - */ - -exports.flatten = function(arr, ret){ - var ret = ret || [] - , len = arr.length; - for (var i = 0; i < len; ++i) { - if (Array.isArray(arr[i])) { - exports.flatten(arr[i], ret); - } else { - ret.push(arr[i]); - } - } - return ret; -}; - -/** - * Normalize the given `type`, for example "html" becomes "text/html". - * - * @param {String} type - * @return {Object} - * @api private - */ - -exports.normalizeType = function(type){ - return ~type.indexOf('/') - ? acceptParams(type) - : { value: mime.lookup(type), params: {} }; -}; - -/** - * Normalize `types`, for example "html" becomes "text/html". - * - * @param {Array} types - * @return {Array} - * @api private - */ - -exports.normalizeTypes = function(types){ - var ret = []; - - for (var i = 0; i < types.length; ++i) { - ret.push(exports.normalizeType(types[i])); - } - - return ret; -}; - -/** - * Return the acceptable type in `types`, if any. - * - * @param {Array} types - * @param {String} str - * @return {String} - * @api private - */ - -exports.acceptsArray = function(types, str){ - // accept anything when Accept is not present - if (!str) return types[0]; - - // parse - var accepted = exports.parseAccept(str) - , normalized = exports.normalizeTypes(types) - , len = accepted.length; - - for (var i = 0; i < len; ++i) { - for (var j = 0, jlen = types.length; j < jlen; ++j) { - if (exports.accept(normalized[j], accepted[i])) { - return types[j]; - } - } - } -}; - -/** - * Check if `type(s)` are acceptable based on - * the given `str`. - * - * @param {String|Array} type(s) - * @param {String} str - * @return {Boolean|String} - * @api private - */ - -exports.accepts = function(type, str){ - if ('string' == typeof type) type = type.split(/ *, */); - return exports.acceptsArray(type, str); -}; - -/** - * Check if `type` array is acceptable for `other`. - * - * @param {Object} type - * @param {Object} other - * @return {Boolean} - * @api private - */ - -exports.accept = function(type, other){ - var t = type.value.split('/'); - return (t[0] == other.type || '*' == other.type) - && (t[1] == other.subtype || '*' == other.subtype) - && paramsEqual(type.params, other.params); -}; - -/** - * Check if accept params are equal. - * - * @param {Object} a - * @param {Object} b - * @return {Boolean} - * @api private - */ - -function paramsEqual(a, b){ - return !Object.keys(a).some(function(k) { - return a[k] != b[k]; - }); -} - -/** - * Parse accept `str`, returning - * an array objects containing - * `.type` and `.subtype` along - * with the values provided by - * `parseQuality()`. - * - * @param {Type} name - * @return {Type} - * @api private - */ - -exports.parseAccept = function(str){ - return exports - .parseParams(str) - .map(function(obj){ - var parts = obj.value.split('/'); - obj.type = parts[0]; - obj.subtype = parts[1]; - return obj; - }); -}; - -/** - * Parse quality `str`, returning an - * array of objects with `.value`, - * `.quality` and optional `.params` - * - * @param {String} str - * @return {Array} - * @api private - */ - -exports.parseParams = function(str){ - return str - .split(/ *, */) - .map(acceptParams) - .filter(function(obj){ - return obj.quality; - }) - .sort(function(a, b){ - return b.quality - a.quality; - }); -}; - -/** - * Parse accept params `str` returning an - * object with `.value`, `.quality` and `.params`. - * - * @param {String} str - * @return {Object} - * @api private - */ - -function acceptParams(str) { - var parts = str.split(/ *; */); - var ret = { value: parts[0], quality: 1, params: {} }; - - for (var i = 1; i < parts.length; ++i) { - var pms = parts[i].split(/ *= */); - if ('q' == pms[0]) { - ret.quality = parseFloat(pms[1]); - } else { - ret.params[pms[0]] = pms[1]; - } - } - - return ret; -} - -/** - * Escape special characters in the given string of html. - * - * @param {String} html - * @return {String} - * @api private - */ - -exports.escape = function(html) { - return String(html) - .replace(/&/g, '&') - .replace(/"/g, '"') - .replace(//g, '>'); -}; - -/** - * Normalize the given path string, - * returning a regular expression. - * - * An empty array should be passed, - * which will contain the placeholder - * key names. For example "/user/:id" will - * then contain ["id"]. - * - * @param {String|RegExp|Array} path - * @param {Array} keys - * @param {Boolean} sensitive - * @param {Boolean} strict - * @return {RegExp} - * @api private - */ - -exports.pathRegexp = function(path, keys, sensitive, strict) { - if (path instanceof RegExp) return path; - if (Array.isArray(path)) path = '(' + path.join('|') + ')'; - path = path - .concat(strict ? '' : '/?') - .replace(/\/\(/g, '(?:/') - .replace(/(\/)?(\.)?:(\w+)(?:(\(.*?\)))?(\?)?(\*)?/g, function(_, slash, format, key, capture, optional, star){ - keys.push({ name: key, optional: !! optional }); - slash = slash || ''; - return '' - + (optional ? '' : slash) - + '(?:' - + (optional ? slash : '') - + (format || '') + (capture || (format && '([^/.]+?)' || '([^/]+?)')) + ')' - + (optional || '') - + (star ? '(/*)?' : ''); - }) - .replace(/([\/.])/g, '\\$1') - .replace(/\*/g, '(.*)'); - return new RegExp('^' + path + '$', sensitive ? '' : 'i'); -} diff --git a/node_modules/express/lib/view.js b/node_modules/express/lib/view.js deleted file mode 100644 index ae20b17..0000000 --- a/node_modules/express/lib/view.js +++ /dev/null @@ -1,76 +0,0 @@ -/** - * Module dependencies. - */ - -var path = require('path') - , fs = require('fs') - , utils = require('./utils') - , dirname = path.dirname - , basename = path.basename - , extname = path.extname - , exists = fs.existsSync || path.existsSync - , join = path.join; - -/** - * Expose `View`. - */ - -module.exports = View; - -/** - * Initialize a new `View` with the given `name`. - * - * Options: - * - * - `defaultEngine` the default template engine name - * - `engines` template engine require() cache - * - `root` root path for view lookup - * - * @param {String} name - * @param {Object} options - * @api private - */ - -function View(name, options) { - options = options || {}; - this.name = name; - this.root = options.root; - var engines = options.engines; - this.defaultEngine = options.defaultEngine; - var ext = this.ext = extname(name); - if (!ext) name += (ext = this.ext = ('.' != this.defaultEngine[0] ? '.' : '') + this.defaultEngine); - this.engine = engines[ext] || (engines[ext] = require(ext.slice(1)).__express); - this.path = this.lookup(name); -} - -/** - * Lookup view by the given `path` - * - * @param {String} path - * @return {String} - * @api private - */ - -View.prototype.lookup = function(path){ - var ext = this.ext; - - // . - if (!utils.isAbsolute(path)) path = join(this.root, path); - if (exists(path)) return path; - - // /index. - path = join(dirname(path), basename(path, ext), 'index' + ext); - if (exists(path)) return path; -}; - -/** - * Render with the given `options` and callback `fn(err, str)`. - * - * @param {Object} options - * @param {Function} fn - * @api private - */ - -View.prototype.render = function(options, fn){ - this.engine(this.path, options, fn); -}; diff --git a/node_modules/express/node_modules/buffer-crc32/.npmignore b/node_modules/express/node_modules/buffer-crc32/.npmignore deleted file mode 100644 index b512c09..0000000 --- a/node_modules/express/node_modules/buffer-crc32/.npmignore +++ /dev/null @@ -1 +0,0 @@ -node_modules \ No newline at end of file diff --git a/node_modules/express/node_modules/buffer-crc32/.travis.yml b/node_modules/express/node_modules/buffer-crc32/.travis.yml deleted file mode 100644 index 7a902e8..0000000 --- a/node_modules/express/node_modules/buffer-crc32/.travis.yml +++ /dev/null @@ -1,8 +0,0 @@ -language: node_js -node_js: - - 0.6 - - 0.8 -notifications: - email: - recipients: - - brianloveswords@gmail.com \ No newline at end of file diff --git a/node_modules/express/node_modules/buffer-crc32/README.md b/node_modules/express/node_modules/buffer-crc32/README.md deleted file mode 100644 index 0d9d8b8..0000000 --- a/node_modules/express/node_modules/buffer-crc32/README.md +++ /dev/null @@ -1,47 +0,0 @@ -# buffer-crc32 - -[![Build Status](https://secure.travis-ci.org/brianloveswords/buffer-crc32.png?branch=master)](http://travis-ci.org/brianloveswords/buffer-crc32) - -crc32 that works with binary data and fancy character sets, outputs -buffer, signed or unsigned data and has tests. - -Derived from the sample CRC implementation in the PNG specification: http://www.w3.org/TR/PNG/#D-CRCAppendix - -# install -``` -npm install buffer-crc32 -``` - -# example -```js -var crc32 = require('buffer-crc32'); -// works with buffers -var buf = Buffer([0x00, 0x73, 0x75, 0x70, 0x20, 0x62, 0x72, 0x6f, 0x00]) -crc32(buf) // -> - -// has convenience methods for getting signed or unsigned ints -crc32.signed(buf) // -> -1805997238 -crc32.unsigned(buf) // -> 2488970058 - -// will cast to buffer if given a string, so you can -// directly use foreign characters safely -crc32('自動販売機') // -> - -// and works in append mode too -var partialCrc = crc32('hey'); -var partialCrc = crc32(' ', partialCrc); -var partialCrc = crc32('sup', partialCrc); -var partialCrc = crc32(' ', partialCrc); -var finalCrc = crc32('bros', partialCrc); // -> -``` - -# tests -This was tested against the output of zlib's crc32 method. You can run -the tests with`npm test` (requires tap) - -# see also -https://github.com/alexgorbatchev/node-crc, `crc.buffer.crc32` also -supports buffer inputs and return unsigned ints (thanks @tjholowaychuk). - -# license -MIT/X11 diff --git a/node_modules/express/node_modules/buffer-crc32/index.js b/node_modules/express/node_modules/buffer-crc32/index.js deleted file mode 100644 index e29ce3e..0000000 --- a/node_modules/express/node_modules/buffer-crc32/index.js +++ /dev/null @@ -1,88 +0,0 @@ -var Buffer = require('buffer').Buffer; - -var CRC_TABLE = [ - 0x00000000, 0x77073096, 0xee0e612c, 0x990951ba, 0x076dc419, - 0x706af48f, 0xe963a535, 0x9e6495a3, 0x0edb8832, 0x79dcb8a4, - 0xe0d5e91e, 0x97d2d988, 0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, - 0x90bf1d91, 0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de, - 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7, 0x136c9856, - 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec, 0x14015c4f, 0x63066cd9, - 0xfa0f3d63, 0x8d080df5, 0x3b6e20c8, 0x4c69105e, 0xd56041e4, - 0xa2677172, 0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b, - 0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940, 0x32d86ce3, - 0x45df5c75, 0xdcd60dcf, 0xabd13d59, 0x26d930ac, 0x51de003a, - 0xc8d75180, 0xbfd06116, 0x21b4f4b5, 0x56b3c423, 0xcfba9599, - 0xb8bda50f, 0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924, - 0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d, 0x76dc4190, - 0x01db7106, 0x98d220bc, 0xefd5102a, 0x71b18589, 0x06b6b51f, - 0x9fbfe4a5, 0xe8b8d433, 0x7807c9a2, 0x0f00f934, 0x9609a88e, - 0xe10e9818, 0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01, - 0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e, 0x6c0695ed, - 0x1b01a57b, 0x8208f4c1, 0xf50fc457, 0x65b0d9c6, 0x12b7e950, - 0x8bbeb8ea, 0xfcb9887c, 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, - 0xfbd44c65, 0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2, - 0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb, 0x4369e96a, - 0x346ed9fc, 0xad678846, 0xda60b8d0, 0x44042d73, 0x33031de5, - 0xaa0a4c5f, 0xdd0d7cc9, 0x5005713c, 0x270241aa, 0xbe0b1010, - 0xc90c2086, 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f, - 0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4, 0x59b33d17, - 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad, 0xedb88320, 0x9abfb3b6, - 0x03b6e20c, 0x74b1d29a, 0xead54739, 0x9dd277af, 0x04db2615, - 0x73dc1683, 0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8, - 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1, 0xf00f9344, - 0x8708a3d2, 0x1e01f268, 0x6906c2fe, 0xf762575d, 0x806567cb, - 0x196c3671, 0x6e6b06e7, 0xfed41b76, 0x89d32be0, 0x10da7a5a, - 0x67dd4acc, 0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5, - 0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252, 0xd1bb67f1, - 0xa6bc5767, 0x3fb506dd, 0x48b2364b, 0xd80d2bda, 0xaf0a1b4c, - 0x36034af6, 0x41047a60, 0xdf60efc3, 0xa867df55, 0x316e8eef, - 0x4669be79, 0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236, - 0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f, 0xc5ba3bbe, - 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04, 0xc2d7ffa7, 0xb5d0cf31, - 0x2cd99e8b, 0x5bdeae1d, 0x9b64c2b0, 0xec63f226, 0x756aa39c, - 0x026d930a, 0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713, - 0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38, 0x92d28e9b, - 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21, 0x86d3d2d4, 0xf1d4e242, - 0x68ddb3f8, 0x1fda836e, 0x81be16cd, 0xf6b9265b, 0x6fb077e1, - 0x18b74777, 0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c, - 0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45, 0xa00ae278, - 0xd70dd2ee, 0x4e048354, 0x3903b3c2, 0xa7672661, 0xd06016f7, - 0x4969474d, 0x3e6e77db, 0xaed16a4a, 0xd9d65adc, 0x40df0b66, - 0x37d83bf0, 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9, - 0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6, 0xbad03605, - 0xcdd70693, 0x54de5729, 0x23d967bf, 0xb3667a2e, 0xc4614ab8, - 0x5d681b02, 0x2a6f2b94, 0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, - 0x2d02ef8d -]; - -function bufferizeInt(num) { - var tmp = Buffer(4); - tmp.writeInt32BE(num, 0); - return tmp; -} - -function _crc32(buf, previous) { - if (!Buffer.isBuffer(buf)) { - buf = Buffer(buf); - } - if (Buffer.isBuffer(previous)) { - previous = previous.readUInt32BE(0); - } - var crc = ~~previous ^ -1; - for (var n = 0; n < buf.length; n++) { - crc = CRC_TABLE[(crc ^ buf[n]) & 0xff] ^ (crc >>> 8); - } - return (crc ^ -1); -} - -function crc32() { - return bufferizeInt(_crc32.apply(null, arguments)); -} -crc32.signed = function () { - return _crc32.apply(null, arguments); -}; -crc32.unsigned = function () { - return _crc32.apply(null, arguments) >>> 0; -}; - -module.exports = crc32; diff --git a/node_modules/express/node_modules/buffer-crc32/package.json b/node_modules/express/node_modules/buffer-crc32/package.json deleted file mode 100644 index 401b2a0..0000000 --- a/node_modules/express/node_modules/buffer-crc32/package.json +++ /dev/null @@ -1,39 +0,0 @@ -{ - "author": { - "name": "Brian J. Brennan", - "email": "brianloveswords@gmail.com", - "url": "http://bjb.io" - }, - "name": "buffer-crc32", - "description": "A pure javascript CRC32 algorithm that plays nice with binary data", - "version": "0.2.1", - "contributors": [ - { - "name": "Vladimir Kuznetsov" - } - ], - "homepage": "https://github.com/brianloveswords/buffer-crc32", - "repository": { - "type": "git", - "url": "git://github.com/brianloveswords/buffer-crc32.git" - }, - "main": "index.js", - "scripts": { - "test": "./node_modules/.bin/tap tests/*.test.js" - }, - "dependencies": {}, - "devDependencies": { - "tap": "~0.2.5" - }, - "optionalDependencies": {}, - "engines": { - "node": "*" - }, - "readme": "# buffer-crc32\n\n[![Build Status](https://secure.travis-ci.org/brianloveswords/buffer-crc32.png?branch=master)](http://travis-ci.org/brianloveswords/buffer-crc32)\n\ncrc32 that works with binary data and fancy character sets, outputs\nbuffer, signed or unsigned data and has tests.\n\nDerived from the sample CRC implementation in the PNG specification: http://www.w3.org/TR/PNG/#D-CRCAppendix\n\n# install\n```\nnpm install buffer-crc32\n```\n\n# example\n```js\nvar crc32 = require('buffer-crc32');\n// works with buffers\nvar buf = Buffer([0x00, 0x73, 0x75, 0x70, 0x20, 0x62, 0x72, 0x6f, 0x00])\ncrc32(buf) // -> \n\n// has convenience methods for getting signed or unsigned ints\ncrc32.signed(buf) // -> -1805997238\ncrc32.unsigned(buf) // -> 2488970058\n\n// will cast to buffer if given a string, so you can\n// directly use foreign characters safely\ncrc32('自動販売機') // -> \n\n// and works in append mode too\nvar partialCrc = crc32('hey');\nvar partialCrc = crc32(' ', partialCrc);\nvar partialCrc = crc32('sup', partialCrc);\nvar partialCrc = crc32(' ', partialCrc);\nvar finalCrc = crc32('bros', partialCrc); // -> \n```\n\n# tests\nThis was tested against the output of zlib's crc32 method. You can run\nthe tests with`npm test` (requires tap)\n\n# see also\nhttps://github.com/alexgorbatchev/node-crc, `crc.buffer.crc32` also\nsupports buffer inputs and return unsigned ints (thanks @tjholowaychuk).\n\n# license\nMIT/X11\n", - "readmeFilename": "README.md", - "bugs": { - "url": "https://github.com/brianloveswords/buffer-crc32/issues" - }, - "_id": "buffer-crc32@0.2.1", - "_from": "buffer-crc32@~0.2.1" -} diff --git a/node_modules/express/node_modules/buffer-crc32/tests/crc.test.js b/node_modules/express/node_modules/buffer-crc32/tests/crc.test.js deleted file mode 100644 index bb0f9ef..0000000 --- a/node_modules/express/node_modules/buffer-crc32/tests/crc.test.js +++ /dev/null @@ -1,89 +0,0 @@ -var crc32 = require('..'); -var test = require('tap').test; - -test('simple crc32 is no problem', function (t) { - var input = Buffer('hey sup bros'); - var expected = Buffer([0x47, 0xfa, 0x55, 0x70]); - t.same(crc32(input), expected); - t.end(); -}); - -test('another simple one', function (t) { - var input = Buffer('IEND'); - var expected = Buffer([0xae, 0x42, 0x60, 0x82]); - t.same(crc32(input), expected); - t.end(); -}); - -test('slightly more complex', function (t) { - var input = Buffer([0x00, 0x00, 0x00]); - var expected = Buffer([0xff, 0x41, 0xd9, 0x12]); - t.same(crc32(input), expected); - t.end(); -}); - -test('complex crc32 gets calculated like a champ', function (t) { - var input = Buffer('शीर्षक'); - var expected = Buffer([0x17, 0xb8, 0xaf, 0xf1]); - t.same(crc32(input), expected); - t.end(); -}); - -test('casts to buffer if necessary', function (t) { - var input = 'शीर्षक'; - var expected = Buffer([0x17, 0xb8, 0xaf, 0xf1]); - t.same(crc32(input), expected); - t.end(); -}); - -test('can do signed', function (t) { - var input = 'ham sandwich'; - var expected = -1891873021; - t.same(crc32.signed(input), expected); - t.end(); -}); - -test('can do unsigned', function (t) { - var input = 'bear sandwich'; - var expected = 3711466352; - t.same(crc32.unsigned(input), expected); - t.end(); -}); - - -test('simple crc32 in append mode', function (t) { - var input = [Buffer('hey'), Buffer(' '), Buffer('sup'), Buffer(' '), Buffer('bros')]; - var expected = Buffer([0x47, 0xfa, 0x55, 0x70]); - for (var crc = 0, i = 0; i < input.length; i++) { - crc = crc32(input[i], crc); - } - t.same(crc, expected); - t.end(); -}); - - -test('can do signed in append mode', function (t) { - var input1 = 'ham'; - var input2 = ' '; - var input3 = 'sandwich'; - var expected = -1891873021; - - var crc = crc32.signed(input1); - crc = crc32.signed(input2, crc); - crc = crc32.signed(input3, crc); - - t.same(crc, expected); - t.end(); -}); - -test('can do unsigned in append mode', function (t) { - var input1 = 'bear san'; - var input2 = 'dwich'; - var expected = 3711466352; - - var crc = crc32.unsigned(input1); - crc = crc32.unsigned(input2, crc); - t.same(crc, expected); - t.end(); -}); - diff --git a/node_modules/express/node_modules/commander/.npmignore b/node_modules/express/node_modules/commander/.npmignore deleted file mode 100644 index f1250e5..0000000 --- a/node_modules/express/node_modules/commander/.npmignore +++ /dev/null @@ -1,4 +0,0 @@ -support -test -examples -*.sock diff --git a/node_modules/express/node_modules/commander/.travis.yml b/node_modules/express/node_modules/commander/.travis.yml deleted file mode 100644 index f1d0f13..0000000 --- a/node_modules/express/node_modules/commander/.travis.yml +++ /dev/null @@ -1,4 +0,0 @@ -language: node_js -node_js: - - 0.4 - - 0.6 diff --git a/node_modules/express/node_modules/commander/History.md b/node_modules/express/node_modules/commander/History.md deleted file mode 100644 index 4961d2e..0000000 --- a/node_modules/express/node_modules/commander/History.md +++ /dev/null @@ -1,107 +0,0 @@ - -0.6.1 / 2012-06-01 -================== - - * Added: append (yes or no) on confirmation - * Added: allow node.js v0.7.x - -0.6.0 / 2012-04-10 -================== - - * Added `.prompt(obj, callback)` support. Closes #49 - * Added default support to .choose(). Closes #41 - * Fixed the choice example - -0.5.1 / 2011-12-20 -================== - - * Fixed `password()` for recent nodes. Closes #36 - -0.5.0 / 2011-12-04 -================== - - * Added sub-command option support [itay] - -0.4.3 / 2011-12-04 -================== - - * Fixed custom help ordering. Closes #32 - -0.4.2 / 2011-11-24 -================== - - * Added travis support - * Fixed: line-buffered input automatically trimmed. Closes #31 - -0.4.1 / 2011-11-18 -================== - - * Removed listening for "close" on --help - -0.4.0 / 2011-11-15 -================== - - * Added support for `--`. Closes #24 - -0.3.3 / 2011-11-14 -================== - - * Fixed: wait for close event when writing help info [Jerry Hamlet] - -0.3.2 / 2011-11-01 -================== - - * Fixed long flag definitions with values [felixge] - -0.3.1 / 2011-10-31 -================== - - * Changed `--version` short flag to `-V` from `-v` - * Changed `.version()` so it's configurable [felixge] - -0.3.0 / 2011-10-31 -================== - - * Added support for long flags only. Closes #18 - -0.2.1 / 2011-10-24 -================== - - * "node": ">= 0.4.x < 0.7.0". Closes #20 - -0.2.0 / 2011-09-26 -================== - - * Allow for defaults that are not just boolean. Default peassignment only occurs for --no-*, optional, and required arguments. [Jim Isaacs] - -0.1.0 / 2011-08-24 -================== - - * Added support for custom `--help` output - -0.0.5 / 2011-08-18 -================== - - * Changed: when the user enters nothing prompt for password again - * Fixed issue with passwords beginning with numbers [NuckChorris] - -0.0.4 / 2011-08-15 -================== - - * Fixed `Commander#args` - -0.0.3 / 2011-08-15 -================== - - * Added default option value support - -0.0.2 / 2011-08-15 -================== - - * Added mask support to `Command#password(str[, mask], fn)` - * Added `Command#password(str, fn)` - -0.0.1 / 2010-01-03 -================== - - * Initial release diff --git a/node_modules/express/node_modules/commander/Makefile b/node_modules/express/node_modules/commander/Makefile deleted file mode 100644 index 0074625..0000000 --- a/node_modules/express/node_modules/commander/Makefile +++ /dev/null @@ -1,7 +0,0 @@ - -TESTS = $(shell find test/test.*.js) - -test: - @./test/run $(TESTS) - -.PHONY: test \ No newline at end of file diff --git a/node_modules/express/node_modules/commander/Readme.md b/node_modules/express/node_modules/commander/Readme.md deleted file mode 100644 index b8328c3..0000000 --- a/node_modules/express/node_modules/commander/Readme.md +++ /dev/null @@ -1,262 +0,0 @@ -# Commander.js - - The complete solution for [node.js](http://nodejs.org) command-line interfaces, inspired by Ruby's [commander](https://github.com/visionmedia/commander). - - [![Build Status](https://secure.travis-ci.org/visionmedia/commander.js.png)](http://travis-ci.org/visionmedia/commander.js) - -## Installation - - $ npm install commander - -## Option parsing - - Options with commander are defined with the `.option()` method, also serving as documentation for the options. The example below parses args and options from `process.argv`, leaving remaining args as the `program.args` array which were not consumed by options. - -```js -#!/usr/bin/env node - -/** - * Module dependencies. - */ - -var program = require('commander'); - -program - .version('0.0.1') - .option('-p, --peppers', 'Add peppers') - .option('-P, --pineapple', 'Add pineapple') - .option('-b, --bbq', 'Add bbq sauce') - .option('-c, --cheese [type]', 'Add the specified type of cheese [marble]', 'marble') - .parse(process.argv); - -console.log('you ordered a pizza with:'); -if (program.peppers) console.log(' - peppers'); -if (program.pineapple) console.log(' - pineappe'); -if (program.bbq) console.log(' - bbq'); -console.log(' - %s cheese', program.cheese); -``` - - Short flags may be passed as a single arg, for example `-abc` is equivalent to `-a -b -c`. Multi-word options such as "--template-engine" are camel-cased, becoming `program.templateEngine` etc. - -## Automated --help - - The help information is auto-generated based on the information commander already knows about your program, so the following `--help` info is for free: - -``` - $ ./examples/pizza --help - - Usage: pizza [options] - - Options: - - -V, --version output the version number - -p, --peppers Add peppers - -P, --pineapple Add pineappe - -b, --bbq Add bbq sauce - -c, --cheese Add the specified type of cheese [marble] - -h, --help output usage information - -``` - -## Coercion - -```js -function range(val) { - return val.split('..').map(Number); -} - -function list(val) { - return val.split(','); -} - -program - .version('0.0.1') - .usage('[options] ') - .option('-i, --integer ', 'An integer argument', parseInt) - .option('-f, --float ', 'A float argument', parseFloat) - .option('-r, --range ..', 'A range', range) - .option('-l, --list ', 'A list', list) - .option('-o, --optional [value]', 'An optional value') - .parse(process.argv); - -console.log(' int: %j', program.integer); -console.log(' float: %j', program.float); -console.log(' optional: %j', program.optional); -program.range = program.range || []; -console.log(' range: %j..%j', program.range[0], program.range[1]); -console.log(' list: %j', program.list); -console.log(' args: %j', program.args); -``` - -## Custom help - - You can display arbitrary `-h, --help` information - by listening for "--help". Commander will automatically - exit once you are done so that the remainder of your program - does not execute causing undesired behaviours, for example - in the following executable "stuff" will not output when - `--help` is used. - -```js -#!/usr/bin/env node - -/** - * Module dependencies. - */ - -var program = require('../'); - -function list(val) { - return val.split(',').map(Number); -} - -program - .version('0.0.1') - .option('-f, --foo', 'enable some foo') - .option('-b, --bar', 'enable some bar') - .option('-B, --baz', 'enable some baz'); - -// must be before .parse() since -// node's emit() is immediate - -program.on('--help', function(){ - console.log(' Examples:'); - console.log(''); - console.log(' $ custom-help --help'); - console.log(' $ custom-help -h'); - console.log(''); -}); - -program.parse(process.argv); - -console.log('stuff'); -``` - -yielding the following help output: - -``` - -Usage: custom-help [options] - -Options: - - -h, --help output usage information - -V, --version output the version number - -f, --foo enable some foo - -b, --bar enable some bar - -B, --baz enable some baz - -Examples: - - $ custom-help --help - $ custom-help -h - -``` - -## .prompt(msg, fn) - - Single-line prompt: - -```js -program.prompt('name: ', function(name){ - console.log('hi %s', name); -}); -``` - - Multi-line prompt: - -```js -program.prompt('description:', function(name){ - console.log('hi %s', name); -}); -``` - - Coercion: - -```js -program.prompt('Age: ', Number, function(age){ - console.log('age: %j', age); -}); -``` - -```js -program.prompt('Birthdate: ', Date, function(date){ - console.log('date: %s', date); -}); -``` - -## .password(msg[, mask], fn) - -Prompt for password without echoing: - -```js -program.password('Password: ', function(pass){ - console.log('got "%s"', pass); - process.stdin.destroy(); -}); -``` - -Prompt for password with mask char "*": - -```js -program.password('Password: ', '*', function(pass){ - console.log('got "%s"', pass); - process.stdin.destroy(); -}); -``` - -## .confirm(msg, fn) - - Confirm with the given `msg`: - -```js -program.confirm('continue? ', function(ok){ - console.log(' got %j', ok); -}); -``` - -## .choose(list, fn) - - Let the user choose from a `list`: - -```js -var list = ['tobi', 'loki', 'jane', 'manny', 'luna']; - -console.log('Choose the coolest pet:'); -program.choose(list, function(i){ - console.log('you chose %d "%s"', i, list[i]); -}); -``` - -## Links - - - [API documentation](http://visionmedia.github.com/commander.js/) - - [ascii tables](https://github.com/LearnBoost/cli-table) - - [progress bars](https://github.com/visionmedia/node-progress) - - [more progress bars](https://github.com/substack/node-multimeter) - - [examples](https://github.com/visionmedia/commander.js/tree/master/examples) - -## License - -(The MIT License) - -Copyright (c) 2011 TJ Holowaychuk <tj@vision-media.ca> - -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. \ No newline at end of file diff --git a/node_modules/express/node_modules/commander/index.js b/node_modules/express/node_modules/commander/index.js deleted file mode 100644 index 06ec1e4..0000000 --- a/node_modules/express/node_modules/commander/index.js +++ /dev/null @@ -1,2 +0,0 @@ - -module.exports = require('./lib/commander'); \ No newline at end of file diff --git a/node_modules/express/node_modules/commander/lib/commander.js b/node_modules/express/node_modules/commander/lib/commander.js deleted file mode 100644 index 5ba87eb..0000000 --- a/node_modules/express/node_modules/commander/lib/commander.js +++ /dev/null @@ -1,1026 +0,0 @@ - -/*! - * commander - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var EventEmitter = require('events').EventEmitter - , path = require('path') - , tty = require('tty') - , basename = path.basename; - -/** - * Expose the root command. - */ - -exports = module.exports = new Command; - -/** - * Expose `Command`. - */ - -exports.Command = Command; - -/** - * Expose `Option`. - */ - -exports.Option = Option; - -/** - * Initialize a new `Option` with the given `flags` and `description`. - * - * @param {String} flags - * @param {String} description - * @api public - */ - -function Option(flags, description) { - this.flags = flags; - this.required = ~flags.indexOf('<'); - this.optional = ~flags.indexOf('['); - this.bool = !~flags.indexOf('-no-'); - flags = flags.split(/[ ,|]+/); - if (flags.length > 1 && !/^[[<]/.test(flags[1])) this.short = flags.shift(); - this.long = flags.shift(); - this.description = description; -} - -/** - * Return option name. - * - * @return {String} - * @api private - */ - -Option.prototype.name = function(){ - return this.long - .replace('--', '') - .replace('no-', ''); -}; - -/** - * Check if `arg` matches the short or long flag. - * - * @param {String} arg - * @return {Boolean} - * @api private - */ - -Option.prototype.is = function(arg){ - return arg == this.short - || arg == this.long; -}; - -/** - * Initialize a new `Command`. - * - * @param {String} name - * @api public - */ - -function Command(name) { - this.commands = []; - this.options = []; - this.args = []; - this.name = name; -} - -/** - * Inherit from `EventEmitter.prototype`. - */ - -Command.prototype.__proto__ = EventEmitter.prototype; - -/** - * Add command `name`. - * - * The `.action()` callback is invoked when the - * command `name` is specified via __ARGV__, - * and the remaining arguments are applied to the - * function for access. - * - * When the `name` is "*" an un-matched command - * will be passed as the first arg, followed by - * the rest of __ARGV__ remaining. - * - * Examples: - * - * program - * .version('0.0.1') - * .option('-C, --chdir ', 'change the working directory') - * .option('-c, --config ', 'set config path. defaults to ./deploy.conf') - * .option('-T, --no-tests', 'ignore test hook') - * - * program - * .command('setup') - * .description('run remote setup commands') - * .action(function(){ - * console.log('setup'); - * }); - * - * program - * .command('exec ') - * .description('run the given remote command') - * .action(function(cmd){ - * console.log('exec "%s"', cmd); - * }); - * - * program - * .command('*') - * .description('deploy the given env') - * .action(function(env){ - * console.log('deploying "%s"', env); - * }); - * - * program.parse(process.argv); - * - * @param {String} name - * @return {Command} the new command - * @api public - */ - -Command.prototype.command = function(name){ - var args = name.split(/ +/); - var cmd = new Command(args.shift()); - this.commands.push(cmd); - cmd.parseExpectedArgs(args); - cmd.parent = this; - return cmd; -}; - -/** - * Parse expected `args`. - * - * For example `["[type]"]` becomes `[{ required: false, name: 'type' }]`. - * - * @param {Array} args - * @return {Command} for chaining - * @api public - */ - -Command.prototype.parseExpectedArgs = function(args){ - if (!args.length) return; - var self = this; - args.forEach(function(arg){ - switch (arg[0]) { - case '<': - self.args.push({ required: true, name: arg.slice(1, -1) }); - break; - case '[': - self.args.push({ required: false, name: arg.slice(1, -1) }); - break; - } - }); - return this; -}; - -/** - * Register callback `fn` for the command. - * - * Examples: - * - * program - * .command('help') - * .description('display verbose help') - * .action(function(){ - * // output help here - * }); - * - * @param {Function} fn - * @return {Command} for chaining - * @api public - */ - -Command.prototype.action = function(fn){ - var self = this; - this.parent.on(this.name, function(args, unknown){ - // Parse any so-far unknown options - unknown = unknown || []; - var parsed = self.parseOptions(unknown); - - // Output help if necessary - outputHelpIfNecessary(self, parsed.unknown); - - // If there are still any unknown options, then we simply - // die, unless someone asked for help, in which case we give it - // to them, and then we die. - if (parsed.unknown.length > 0) { - self.unknownOption(parsed.unknown[0]); - } - - self.args.forEach(function(arg, i){ - if (arg.required && null == args[i]) { - self.missingArgument(arg.name); - } - }); - - // Always append ourselves to the end of the arguments, - // to make sure we match the number of arguments the user - // expects - if (self.args.length) { - args[self.args.length] = self; - } else { - args.push(self); - } - - fn.apply(this, args); - }); - return this; -}; - -/** - * Define option with `flags`, `description` and optional - * coercion `fn`. - * - * The `flags` string should contain both the short and long flags, - * separated by comma, a pipe or space. The following are all valid - * all will output this way when `--help` is used. - * - * "-p, --pepper" - * "-p|--pepper" - * "-p --pepper" - * - * Examples: - * - * // simple boolean defaulting to false - * program.option('-p, --pepper', 'add pepper'); - * - * --pepper - * program.pepper - * // => Boolean - * - * // simple boolean defaulting to false - * program.option('-C, --no-cheese', 'remove cheese'); - * - * program.cheese - * // => true - * - * --no-cheese - * program.cheese - * // => true - * - * // required argument - * program.option('-C, --chdir ', 'change the working directory'); - * - * --chdir /tmp - * program.chdir - * // => "/tmp" - * - * // optional argument - * program.option('-c, --cheese [type]', 'add cheese [marble]'); - * - * @param {String} flags - * @param {String} description - * @param {Function|Mixed} fn or default - * @param {Mixed} defaultValue - * @return {Command} for chaining - * @api public - */ - -Command.prototype.option = function(flags, description, fn, defaultValue){ - var self = this - , option = new Option(flags, description) - , oname = option.name() - , name = camelcase(oname); - - // default as 3rd arg - if ('function' != typeof fn) defaultValue = fn, fn = null; - - // preassign default value only for --no-*, [optional], or - if (false == option.bool || option.optional || option.required) { - // when --no-* we make sure default is true - if (false == option.bool) defaultValue = true; - // preassign only if we have a default - if (undefined !== defaultValue) self[name] = defaultValue; - } - - // register the option - this.options.push(option); - - // when it's passed assign the value - // and conditionally invoke the callback - this.on(oname, function(val){ - // coercion - if (null != val && fn) val = fn(val); - - // unassigned or bool - if ('boolean' == typeof self[name] || 'undefined' == typeof self[name]) { - // if no value, bool true, and we have a default, then use it! - if (null == val) { - self[name] = option.bool - ? defaultValue || true - : false; - } else { - self[name] = val; - } - } else if (null !== val) { - // reassign - self[name] = val; - } - }); - - return this; -}; - -/** - * Parse `argv`, settings options and invoking commands when defined. - * - * @param {Array} argv - * @return {Command} for chaining - * @api public - */ - -Command.prototype.parse = function(argv){ - // store raw args - this.rawArgs = argv; - - // guess name - if (!this.name) this.name = basename(argv[1]); - - // process argv - var parsed = this.parseOptions(this.normalize(argv.slice(2))); - this.args = parsed.args; - return this.parseArgs(this.args, parsed.unknown); -}; - -/** - * Normalize `args`, splitting joined short flags. For example - * the arg "-abc" is equivalent to "-a -b -c". - * - * @param {Array} args - * @return {Array} - * @api private - */ - -Command.prototype.normalize = function(args){ - var ret = [] - , arg; - - for (var i = 0, len = args.length; i < len; ++i) { - arg = args[i]; - if (arg.length > 1 && '-' == arg[0] && '-' != arg[1]) { - arg.slice(1).split('').forEach(function(c){ - ret.push('-' + c); - }); - } else { - ret.push(arg); - } - } - - return ret; -}; - -/** - * Parse command `args`. - * - * When listener(s) are available those - * callbacks are invoked, otherwise the "*" - * event is emitted and those actions are invoked. - * - * @param {Array} args - * @return {Command} for chaining - * @api private - */ - -Command.prototype.parseArgs = function(args, unknown){ - var cmds = this.commands - , len = cmds.length - , name; - - if (args.length) { - name = args[0]; - if (this.listeners(name).length) { - this.emit(args.shift(), args, unknown); - } else { - this.emit('*', args); - } - } else { - outputHelpIfNecessary(this, unknown); - - // If there were no args and we have unknown options, - // then they are extraneous and we need to error. - if (unknown.length > 0) { - this.unknownOption(unknown[0]); - } - } - - return this; -}; - -/** - * Return an option matching `arg` if any. - * - * @param {String} arg - * @return {Option} - * @api private - */ - -Command.prototype.optionFor = function(arg){ - for (var i = 0, len = this.options.length; i < len; ++i) { - if (this.options[i].is(arg)) { - return this.options[i]; - } - } -}; - -/** - * Parse options from `argv` returning `argv` - * void of these options. - * - * @param {Array} argv - * @return {Array} - * @api public - */ - -Command.prototype.parseOptions = function(argv){ - var args = [] - , len = argv.length - , literal - , option - , arg; - - var unknownOptions = []; - - // parse options - for (var i = 0; i < len; ++i) { - arg = argv[i]; - - // literal args after -- - if ('--' == arg) { - literal = true; - continue; - } - - if (literal) { - args.push(arg); - continue; - } - - // find matching Option - option = this.optionFor(arg); - - // option is defined - if (option) { - // requires arg - if (option.required) { - arg = argv[++i]; - if (null == arg) return this.optionMissingArgument(option); - if ('-' == arg[0]) return this.optionMissingArgument(option, arg); - this.emit(option.name(), arg); - // optional arg - } else if (option.optional) { - arg = argv[i+1]; - if (null == arg || '-' == arg[0]) { - arg = null; - } else { - ++i; - } - this.emit(option.name(), arg); - // bool - } else { - this.emit(option.name()); - } - continue; - } - - // looks like an option - if (arg.length > 1 && '-' == arg[0]) { - unknownOptions.push(arg); - - // If the next argument looks like it might be - // an argument for this option, we pass it on. - // If it isn't, then it'll simply be ignored - if (argv[i+1] && '-' != argv[i+1][0]) { - unknownOptions.push(argv[++i]); - } - continue; - } - - // arg - args.push(arg); - } - - return { args: args, unknown: unknownOptions }; -}; - -/** - * Argument `name` is missing. - * - * @param {String} name - * @api private - */ - -Command.prototype.missingArgument = function(name){ - console.error(); - console.error(" error: missing required argument `%s'", name); - console.error(); - process.exit(1); -}; - -/** - * `Option` is missing an argument, but received `flag` or nothing. - * - * @param {String} option - * @param {String} flag - * @api private - */ - -Command.prototype.optionMissingArgument = function(option, flag){ - console.error(); - if (flag) { - console.error(" error: option `%s' argument missing, got `%s'", option.flags, flag); - } else { - console.error(" error: option `%s' argument missing", option.flags); - } - console.error(); - process.exit(1); -}; - -/** - * Unknown option `flag`. - * - * @param {String} flag - * @api private - */ - -Command.prototype.unknownOption = function(flag){ - console.error(); - console.error(" error: unknown option `%s'", flag); - console.error(); - process.exit(1); -}; - -/** - * Set the program version to `str`. - * - * This method auto-registers the "-V, --version" flag - * which will print the version number when passed. - * - * @param {String} str - * @param {String} flags - * @return {Command} for chaining - * @api public - */ - -Command.prototype.version = function(str, flags){ - if (0 == arguments.length) return this._version; - this._version = str; - flags = flags || '-V, --version'; - this.option(flags, 'output the version number'); - this.on('version', function(){ - console.log(str); - process.exit(0); - }); - return this; -}; - -/** - * Set the description `str`. - * - * @param {String} str - * @return {String|Command} - * @api public - */ - -Command.prototype.description = function(str){ - if (0 == arguments.length) return this._description; - this._description = str; - return this; -}; - -/** - * Set / get the command usage `str`. - * - * @param {String} str - * @return {String|Command} - * @api public - */ - -Command.prototype.usage = function(str){ - var args = this.args.map(function(arg){ - return arg.required - ? '<' + arg.name + '>' - : '[' + arg.name + ']'; - }); - - var usage = '[options' - + (this.commands.length ? '] [command' : '') - + ']' - + (this.args.length ? ' ' + args : ''); - if (0 == arguments.length) return this._usage || usage; - this._usage = str; - - return this; -}; - -/** - * Return the largest option length. - * - * @return {Number} - * @api private - */ - -Command.prototype.largestOptionLength = function(){ - return this.options.reduce(function(max, option){ - return Math.max(max, option.flags.length); - }, 0); -}; - -/** - * Return help for options. - * - * @return {String} - * @api private - */ - -Command.prototype.optionHelp = function(){ - var width = this.largestOptionLength(); - - // Prepend the help information - return [pad('-h, --help', width) + ' ' + 'output usage information'] - .concat(this.options.map(function(option){ - return pad(option.flags, width) - + ' ' + option.description; - })) - .join('\n'); -}; - -/** - * Return command help documentation. - * - * @return {String} - * @api private - */ - -Command.prototype.commandHelp = function(){ - if (!this.commands.length) return ''; - return [ - '' - , ' Commands:' - , '' - , this.commands.map(function(cmd){ - var args = cmd.args.map(function(arg){ - return arg.required - ? '<' + arg.name + '>' - : '[' + arg.name + ']'; - }).join(' '); - - return cmd.name - + (cmd.options.length - ? ' [options]' - : '') + ' ' + args - + (cmd.description() - ? '\n' + cmd.description() - : ''); - }).join('\n\n').replace(/^/gm, ' ') - , '' - ].join('\n'); -}; - -/** - * Return program help documentation. - * - * @return {String} - * @api private - */ - -Command.prototype.helpInformation = function(){ - return [ - '' - , ' Usage: ' + this.name + ' ' + this.usage() - , '' + this.commandHelp() - , ' Options:' - , '' - , '' + this.optionHelp().replace(/^/gm, ' ') - , '' - , '' - ].join('\n'); -}; - -/** - * Prompt for a `Number`. - * - * @param {String} str - * @param {Function} fn - * @api private - */ - -Command.prototype.promptForNumber = function(str, fn){ - var self = this; - this.promptSingleLine(str, function parseNumber(val){ - val = Number(val); - if (isNaN(val)) return self.promptSingleLine(str + '(must be a number) ', parseNumber); - fn(val); - }); -}; - -/** - * Prompt for a `Date`. - * - * @param {String} str - * @param {Function} fn - * @api private - */ - -Command.prototype.promptForDate = function(str, fn){ - var self = this; - this.promptSingleLine(str, function parseDate(val){ - val = new Date(val); - if (isNaN(val.getTime())) return self.promptSingleLine(str + '(must be a date) ', parseDate); - fn(val); - }); -}; - -/** - * Single-line prompt. - * - * @param {String} str - * @param {Function} fn - * @api private - */ - -Command.prototype.promptSingleLine = function(str, fn){ - if ('function' == typeof arguments[2]) { - return this['promptFor' + (fn.name || fn)](str, arguments[2]); - } - - process.stdout.write(str); - process.stdin.setEncoding('utf8'); - process.stdin.once('data', function(val){ - fn(val.trim()); - }).resume(); -}; - -/** - * Multi-line prompt. - * - * @param {String} str - * @param {Function} fn - * @api private - */ - -Command.prototype.promptMultiLine = function(str, fn){ - var buf = []; - console.log(str); - process.stdin.setEncoding('utf8'); - process.stdin.on('data', function(val){ - if ('\n' == val || '\r\n' == val) { - process.stdin.removeAllListeners('data'); - fn(buf.join('\n')); - } else { - buf.push(val.trimRight()); - } - }).resume(); -}; - -/** - * Prompt `str` and callback `fn(val)` - * - * Commander supports single-line and multi-line prompts. - * To issue a single-line prompt simply add white-space - * to the end of `str`, something like "name: ", whereas - * for a multi-line prompt omit this "description:". - * - * - * Examples: - * - * program.prompt('Username: ', function(name){ - * console.log('hi %s', name); - * }); - * - * program.prompt('Description:', function(desc){ - * console.log('description was "%s"', desc.trim()); - * }); - * - * @param {String|Object} str - * @param {Function} fn - * @api public - */ - -Command.prototype.prompt = function(str, fn){ - var self = this; - - if ('string' == typeof str) { - if (/ $/.test(str)) return this.promptSingleLine.apply(this, arguments); - this.promptMultiLine(str, fn); - } else { - var keys = Object.keys(str) - , obj = {}; - - function next() { - var key = keys.shift() - , label = str[key]; - - if (!key) return fn(obj); - self.prompt(label, function(val){ - obj[key] = val; - next(); - }); - } - - next(); - } -}; - -/** - * Prompt for password with `str`, `mask` char and callback `fn(val)`. - * - * The mask string defaults to '', aka no output is - * written while typing, you may want to use "*" etc. - * - * Examples: - * - * program.password('Password: ', function(pass){ - * console.log('got "%s"', pass); - * process.stdin.destroy(); - * }); - * - * program.password('Password: ', '*', function(pass){ - * console.log('got "%s"', pass); - * process.stdin.destroy(); - * }); - * - * @param {String} str - * @param {String} mask - * @param {Function} fn - * @api public - */ - -Command.prototype.password = function(str, mask, fn){ - var self = this - , buf = ''; - - // default mask - if ('function' == typeof mask) { - fn = mask; - mask = ''; - } - - process.stdin.resume(); - tty.setRawMode(true); - process.stdout.write(str); - - // keypress - process.stdin.on('keypress', function(c, key){ - if (key && 'enter' == key.name) { - console.log(); - process.stdin.removeAllListeners('keypress'); - tty.setRawMode(false); - if (!buf.trim().length) return self.password(str, mask, fn); - fn(buf); - return; - } - - if (key && key.ctrl && 'c' == key.name) { - console.log('%s', buf); - process.exit(); - } - - process.stdout.write(mask); - buf += c; - }).resume(); -}; - -/** - * Confirmation prompt with `str` and callback `fn(bool)` - * - * Examples: - * - * program.confirm('continue? ', function(ok){ - * console.log(' got %j', ok); - * process.stdin.destroy(); - * }); - * - * @param {String} str - * @param {Function} fn - * @api public - */ - - -Command.prototype.confirm = function(str, fn, verbose){ - var self = this; - this.prompt(str, function(ok){ - if (!ok.trim()) { - if (!verbose) str += '(yes or no) '; - return self.confirm(str, fn, true); - } - fn(parseBool(ok)); - }); -}; - -/** - * Choice prompt with `list` of items and callback `fn(index, item)` - * - * Examples: - * - * var list = ['tobi', 'loki', 'jane', 'manny', 'luna']; - * - * console.log('Choose the coolest pet:'); - * program.choose(list, function(i){ - * console.log('you chose %d "%s"', i, list[i]); - * process.stdin.destroy(); - * }); - * - * @param {Array} list - * @param {Number|Function} index or fn - * @param {Function} fn - * @api public - */ - -Command.prototype.choose = function(list, index, fn){ - var self = this - , hasDefault = 'number' == typeof index; - - if (!hasDefault) { - fn = index; - index = null; - } - - list.forEach(function(item, i){ - if (hasDefault && i == index) { - console.log('* %d) %s', i + 1, item); - } else { - console.log(' %d) %s', i + 1, item); - } - }); - - function again() { - self.prompt(' : ', function(val){ - val = parseInt(val, 10) - 1; - if (hasDefault && isNaN(val)) val = index; - - if (null == list[val]) { - again(); - } else { - fn(val, list[val]); - } - }); - } - - again(); -}; - -/** - * Camel-case the given `flag` - * - * @param {String} flag - * @return {String} - * @api private - */ - -function camelcase(flag) { - return flag.split('-').reduce(function(str, word){ - return str + word[0].toUpperCase() + word.slice(1); - }); -} - -/** - * Parse a boolean `str`. - * - * @param {String} str - * @return {Boolean} - * @api private - */ - -function parseBool(str) { - return /^y|yes|ok|true$/i.test(str); -} - -/** - * Pad `str` to `width`. - * - * @param {String} str - * @param {Number} width - * @return {String} - * @api private - */ - -function pad(str, width) { - var len = Math.max(0, width - str.length); - return str + Array(len + 1).join(' '); -} - -/** - * Output help information if necessary - * - * @param {Command} command to output help for - * @param {Array} array of options to search for -h or --help - * @api private - */ - -function outputHelpIfNecessary(cmd, options) { - options = options || []; - for (var i = 0; i < options.length; i++) { - if (options[i] == '--help' || options[i] == '-h') { - process.stdout.write(cmd.helpInformation()); - cmd.emit('--help'); - process.exit(0); - } - } -} diff --git a/node_modules/express/node_modules/commander/package.json b/node_modules/express/node_modules/commander/package.json deleted file mode 100644 index 6f9d567..0000000 --- a/node_modules/express/node_modules/commander/package.json +++ /dev/null @@ -1,38 +0,0 @@ -{ - "name": "commander", - "version": "0.6.1", - "description": "the complete solution for node.js command-line programs", - "keywords": [ - "command", - "option", - "parser", - "prompt", - "stdin" - ], - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca" - }, - "repository": { - "type": "git", - "url": "https://github.com/visionmedia/commander.js.git" - }, - "dependencies": {}, - "devDependencies": { - "should": ">= 0.0.1" - }, - "scripts": { - "test": "make test" - }, - "main": "index", - "engines": { - "node": ">= 0.4.x" - }, - "readme": "# Commander.js\n\n The complete solution for [node.js](http://nodejs.org) command-line interfaces, inspired by Ruby's [commander](https://github.com/visionmedia/commander).\n\n [![Build Status](https://secure.travis-ci.org/visionmedia/commander.js.png)](http://travis-ci.org/visionmedia/commander.js)\n\n## Installation\n\n $ npm install commander\n\n## Option parsing\n\n Options with commander are defined with the `.option()` method, also serving as documentation for the options. The example below parses args and options from `process.argv`, leaving remaining args as the `program.args` array which were not consumed by options.\n\n```js\n#!/usr/bin/env node\n\n/**\n * Module dependencies.\n */\n\nvar program = require('commander');\n\nprogram\n .version('0.0.1')\n .option('-p, --peppers', 'Add peppers')\n .option('-P, --pineapple', 'Add pineapple')\n .option('-b, --bbq', 'Add bbq sauce')\n .option('-c, --cheese [type]', 'Add the specified type of cheese [marble]', 'marble')\n .parse(process.argv);\n\nconsole.log('you ordered a pizza with:');\nif (program.peppers) console.log(' - peppers');\nif (program.pineapple) console.log(' - pineappe');\nif (program.bbq) console.log(' - bbq');\nconsole.log(' - %s cheese', program.cheese);\n```\n\n Short flags may be passed as a single arg, for example `-abc` is equivalent to `-a -b -c`. Multi-word options such as \"--template-engine\" are camel-cased, becoming `program.templateEngine` etc.\n\n## Automated --help\n\n The help information is auto-generated based on the information commander already knows about your program, so the following `--help` info is for free:\n\n``` \n $ ./examples/pizza --help\n\n Usage: pizza [options]\n\n Options:\n\n -V, --version output the version number\n -p, --peppers Add peppers\n -P, --pineapple Add pineappe\n -b, --bbq Add bbq sauce\n -c, --cheese Add the specified type of cheese [marble]\n -h, --help output usage information\n\n```\n\n## Coercion\n\n```js\nfunction range(val) {\n return val.split('..').map(Number);\n}\n\nfunction list(val) {\n return val.split(',');\n}\n\nprogram\n .version('0.0.1')\n .usage('[options] ')\n .option('-i, --integer ', 'An integer argument', parseInt)\n .option('-f, --float ', 'A float argument', parseFloat)\n .option('-r, --range ..', 'A range', range)\n .option('-l, --list ', 'A list', list)\n .option('-o, --optional [value]', 'An optional value')\n .parse(process.argv);\n\nconsole.log(' int: %j', program.integer);\nconsole.log(' float: %j', program.float);\nconsole.log(' optional: %j', program.optional);\nprogram.range = program.range || [];\nconsole.log(' range: %j..%j', program.range[0], program.range[1]);\nconsole.log(' list: %j', program.list);\nconsole.log(' args: %j', program.args);\n```\n\n## Custom help\n\n You can display arbitrary `-h, --help` information\n by listening for \"--help\". Commander will automatically\n exit once you are done so that the remainder of your program\n does not execute causing undesired behaviours, for example\n in the following executable \"stuff\" will not output when\n `--help` is used.\n\n```js\n#!/usr/bin/env node\n\n/**\n * Module dependencies.\n */\n\nvar program = require('../');\n\nfunction list(val) {\n return val.split(',').map(Number);\n}\n\nprogram\n .version('0.0.1')\n .option('-f, --foo', 'enable some foo')\n .option('-b, --bar', 'enable some bar')\n .option('-B, --baz', 'enable some baz');\n\n// must be before .parse() since\n// node's emit() is immediate\n\nprogram.on('--help', function(){\n console.log(' Examples:');\n console.log('');\n console.log(' $ custom-help --help');\n console.log(' $ custom-help -h');\n console.log('');\n});\n\nprogram.parse(process.argv);\n\nconsole.log('stuff');\n```\n\nyielding the following help output:\n\n```\n\nUsage: custom-help [options]\n\nOptions:\n\n -h, --help output usage information\n -V, --version output the version number\n -f, --foo enable some foo\n -b, --bar enable some bar\n -B, --baz enable some baz\n\nExamples:\n\n $ custom-help --help\n $ custom-help -h\n\n```\n\n## .prompt(msg, fn)\n\n Single-line prompt:\n\n```js\nprogram.prompt('name: ', function(name){\n console.log('hi %s', name);\n});\n```\n\n Multi-line prompt:\n\n```js\nprogram.prompt('description:', function(name){\n console.log('hi %s', name);\n});\n```\n\n Coercion:\n\n```js\nprogram.prompt('Age: ', Number, function(age){\n console.log('age: %j', age);\n});\n```\n\n```js\nprogram.prompt('Birthdate: ', Date, function(date){\n console.log('date: %s', date);\n});\n```\n\n## .password(msg[, mask], fn)\n\nPrompt for password without echoing:\n\n```js\nprogram.password('Password: ', function(pass){\n console.log('got \"%s\"', pass);\n process.stdin.destroy();\n});\n```\n\nPrompt for password with mask char \"*\":\n\n```js\nprogram.password('Password: ', '*', function(pass){\n console.log('got \"%s\"', pass);\n process.stdin.destroy();\n});\n```\n\n## .confirm(msg, fn)\n\n Confirm with the given `msg`:\n\n```js\nprogram.confirm('continue? ', function(ok){\n console.log(' got %j', ok);\n});\n```\n\n## .choose(list, fn)\n\n Let the user choose from a `list`:\n\n```js\nvar list = ['tobi', 'loki', 'jane', 'manny', 'luna'];\n\nconsole.log('Choose the coolest pet:');\nprogram.choose(list, function(i){\n console.log('you chose %d \"%s\"', i, list[i]);\n});\n```\n\n## Links\n\n - [API documentation](http://visionmedia.github.com/commander.js/)\n - [ascii tables](https://github.com/LearnBoost/cli-table)\n - [progress bars](https://github.com/visionmedia/node-progress)\n - [more progress bars](https://github.com/substack/node-multimeter)\n - [examples](https://github.com/visionmedia/commander.js/tree/master/examples)\n\n## License \n\n(The MIT License)\n\nCopyright (c) 2011 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.", - "readmeFilename": "Readme.md", - "bugs": { - "url": "https://github.com/visionmedia/commander.js/issues" - }, - "_id": "commander@0.6.1", - "_from": "commander@0.6.1" -} diff --git a/node_modules/express/node_modules/connect/.npmignore b/node_modules/express/node_modules/connect/.npmignore deleted file mode 100644 index 9046dde..0000000 --- a/node_modules/express/node_modules/connect/.npmignore +++ /dev/null @@ -1,12 +0,0 @@ -*.markdown -*.md -.git* -Makefile -benchmarks/ -docs/ -examples/ -install.sh -support/ -test/ -.DS_Store -coverage.html diff --git a/node_modules/express/node_modules/connect/.travis.yml b/node_modules/express/node_modules/connect/.travis.yml deleted file mode 100644 index c46b989..0000000 --- a/node_modules/express/node_modules/connect/.travis.yml +++ /dev/null @@ -1,5 +0,0 @@ -language: node_js -node_js: - - "0.6" - - "0.8" - - "0.10" \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/LICENSE b/node_modules/express/node_modules/connect/LICENSE deleted file mode 100644 index 0c5d22d..0000000 --- a/node_modules/express/node_modules/connect/LICENSE +++ /dev/null @@ -1,24 +0,0 @@ -(The MIT License) - -Copyright (c) 2010 Sencha Inc. -Copyright (c) 2011 LearnBoost -Copyright (c) 2011 TJ Holowaychuk - -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. \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/Readme.md b/node_modules/express/node_modules/connect/Readme.md deleted file mode 100644 index 7d65f9c..0000000 --- a/node_modules/express/node_modules/connect/Readme.md +++ /dev/null @@ -1,133 +0,0 @@ -[![build status](https://secure.travis-ci.org/senchalabs/connect.png)](http://travis-ci.org/senchalabs/connect) -# Connect - - Connect is an extensible HTTP server framework for [node](http://nodejs.org), providing high performance "plugins" known as _middleware_. - - Connect is bundled with over _20_ commonly used middleware, including - a logger, session support, cookie parser, and [more](http://senchalabs.github.com/connect). Be sure to view the 2.x [documentation](http://senchalabs.github.com/connect/). - -```js -var connect = require('connect') - , http = require('http'); - -var app = connect() - .use(connect.favicon()) - .use(connect.logger('dev')) - .use(connect.static('public')) - .use(connect.directory('public')) - .use(connect.cookieParser()) - .use(connect.session({ secret: 'my secret here' })) - .use(function(req, res){ - res.end('Hello from Connect!\n'); - }); - -http.createServer(app).listen(3000); -``` - -## Middleware - - - [csrf](http://www.senchalabs.org/connect/csrf.html) - - [basicAuth](http://www.senchalabs.org/connect/basicAuth.html) - - [bodyParser](http://www.senchalabs.org/connect/bodyParser.html) - - [json](http://www.senchalabs.org/connect/json.html) - - [multipart](http://www.senchalabs.org/connect/multipart.html) - - [urlencoded](http://www.senchalabs.org/connect/urlencoded.html) - - [cookieParser](http://www.senchalabs.org/connect/cookieParser.html) - - [directory](http://www.senchalabs.org/connect/directory.html) - - [compress](http://www.senchalabs.org/connect/compress.html) - - [errorHandler](http://www.senchalabs.org/connect/errorHandler.html) - - [favicon](http://www.senchalabs.org/connect/favicon.html) - - [limit](http://www.senchalabs.org/connect/limit.html) - - [logger](http://www.senchalabs.org/connect/logger.html) - - [methodOverride](http://www.senchalabs.org/connect/methodOverride.html) - - [query](http://www.senchalabs.org/connect/query.html) - - [responseTime](http://www.senchalabs.org/connect/responseTime.html) - - [session](http://www.senchalabs.org/connect/session.html) - - [static](http://www.senchalabs.org/connect/static.html) - - [staticCache](http://www.senchalabs.org/connect/staticCache.html) - - [vhost](http://www.senchalabs.org/connect/vhost.html) - - [subdomains](http://www.senchalabs.org/connect/subdomains.html) - - [cookieSession](http://www.senchalabs.org/connect/cookieSession.html) - -## Running Tests - -first: - - $ npm install -d - -then: - - $ make test - -## Authors - - Below is the output from [git-summary](http://github.com/visionmedia/git-extras). - - - project: connect - commits: 2033 - active : 301 days - files : 171 - authors: - 1414 Tj Holowaychuk 69.6% - 298 visionmedia 14.7% - 191 Tim Caswell 9.4% - 51 TJ Holowaychuk 2.5% - 10 Ryan Olds 0.5% - 8 Astro 0.4% - 5 Nathan Rajlich 0.2% - 5 Jakub Nešetřil 0.2% - 3 Daniel Dickison 0.1% - 3 David Rio Deiros 0.1% - 3 Alexander Simmerl 0.1% - 3 Andreas Lind Petersen 0.1% - 2 Aaron Heckmann 0.1% - 2 Jacques Crocker 0.1% - 2 Fabian Jakobs 0.1% - 2 Brian J Brennan 0.1% - 2 Adam Malcontenti-Wilson 0.1% - 2 Glen Mailer 0.1% - 2 James Campos 0.1% - 1 Trent Mick 0.0% - 1 Troy Kruthoff 0.0% - 1 Wei Zhu 0.0% - 1 comerc 0.0% - 1 darobin 0.0% - 1 nateps 0.0% - 1 Marco Sanson 0.0% - 1 Arthur Taylor 0.0% - 1 Aseem Kishore 0.0% - 1 Bart Teeuwisse 0.0% - 1 Cameron Howey 0.0% - 1 Chad Weider 0.0% - 1 Craig Barnes 0.0% - 1 Eran Hammer-Lahav 0.0% - 1 Gregory McWhirter 0.0% - 1 Guillermo Rauch 0.0% - 1 Jae Kwon 0.0% - 1 Jakub Nesetril 0.0% - 1 Joshua Peek 0.0% - 1 Jxck 0.0% - 1 AJ ONeal 0.0% - 1 Michael Hemesath 0.0% - 1 Morten Siebuhr 0.0% - 1 Samori Gorse 0.0% - 1 Tom Jensen 0.0% - -## Node Compatibility - - Connect `< 1.x` is compatible with node 0.2.x - - - Connect `1.x` is compatible with node 0.4.x - - - Connect (_master_) `2.x` is compatible with node 0.6.x - -## CLA - - [http://sencha.com/cla](http://sencha.com/cla) - -## License - -View the [LICENSE](https://github.com/senchalabs/connect/blob/master/LICENSE) file. The [Silk](http://www.famfamfam.com/lab/icons/silk/) icons used by the `directory` middleware created by/copyright of [FAMFAMFAM](http://www.famfamfam.com/). diff --git a/node_modules/express/node_modules/connect/index.js b/node_modules/express/node_modules/connect/index.js deleted file mode 100644 index 23240ee..0000000 --- a/node_modules/express/node_modules/connect/index.js +++ /dev/null @@ -1,4 +0,0 @@ - -module.exports = process.env.CONNECT_COV - ? require('./lib-cov/connect') - : require('./lib/connect'); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/lib/cache.js b/node_modules/express/node_modules/connect/lib/cache.js deleted file mode 100644 index 052fcdb..0000000 --- a/node_modules/express/node_modules/connect/lib/cache.js +++ /dev/null @@ -1,81 +0,0 @@ - -/*! - * Connect - Cache - * Copyright(c) 2011 Sencha Inc. - * MIT Licensed - */ - -/** - * Expose `Cache`. - */ - -module.exports = Cache; - -/** - * LRU cache store. - * - * @param {Number} limit - * @api private - */ - -function Cache(limit) { - this.store = {}; - this.keys = []; - this.limit = limit; -} - -/** - * Touch `key`, promoting the object. - * - * @param {String} key - * @param {Number} i - * @api private - */ - -Cache.prototype.touch = function(key, i){ - this.keys.splice(i,1); - this.keys.push(key); -}; - -/** - * Remove `key`. - * - * @param {String} key - * @api private - */ - -Cache.prototype.remove = function(key){ - delete this.store[key]; -}; - -/** - * Get the object stored for `key`. - * - * @param {String} key - * @return {Array} - * @api private - */ - -Cache.prototype.get = function(key){ - return this.store[key]; -}; - -/** - * Add a cache `key`. - * - * @param {String} key - * @return {Array} - * @api private - */ - -Cache.prototype.add = function(key){ - // initialize store - var len = this.keys.push(key); - - // limit reached, invalidate LRU - if (len > this.limit) this.remove(this.keys.shift()); - - var arr = this.store[key] = []; - arr.createdAt = new Date; - return arr; -}; diff --git a/node_modules/express/node_modules/connect/lib/connect.js b/node_modules/express/node_modules/connect/lib/connect.js deleted file mode 100644 index 024fce4..0000000 --- a/node_modules/express/node_modules/connect/lib/connect.js +++ /dev/null @@ -1,92 +0,0 @@ -/*! - * Connect - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var EventEmitter = require('events').EventEmitter - , proto = require('./proto') - , utils = require('./utils') - , path = require('path') - , basename = path.basename - , fs = require('fs'); - -// node patches - -require('./patch'); - -// expose createServer() as the module - -exports = module.exports = createServer; - -/** - * Framework version. - */ - -exports.version = '2.7.5'; - -/** - * Expose mime module. - */ - -exports.mime = require('./middleware/static').mime; - -/** - * Expose the prototype. - */ - -exports.proto = proto; - -/** - * Auto-load middleware getters. - */ - -exports.middleware = {}; - -/** - * Expose utilities. - */ - -exports.utils = utils; - -/** - * Create a new connect server. - * - * @return {Function} - * @api public - */ - -function createServer() { - function app(req, res, next){ app.handle(req, res, next); } - utils.merge(app, proto); - utils.merge(app, EventEmitter.prototype); - app.route = '/'; - app.stack = []; - for (var i = 0; i < arguments.length; ++i) { - app.use(arguments[i]); - } - return app; -}; - -/** - * Support old `.createServer()` method. - */ - -createServer.createServer = createServer; - -/** - * Auto-load bundled middleware with getters. - */ - -fs.readdirSync(__dirname + '/middleware').forEach(function(filename){ - if (!/\.js$/.test(filename)) return; - var name = basename(filename, '.js'); - function load(){ return require('./middleware/' + name); } - exports.middleware.__defineGetter__(name, load); - exports.__defineGetter__(name, load); -}); diff --git a/node_modules/express/node_modules/connect/lib/index.js b/node_modules/express/node_modules/connect/lib/index.js deleted file mode 100644 index 2618ddc..0000000 --- a/node_modules/express/node_modules/connect/lib/index.js +++ /dev/null @@ -1,50 +0,0 @@ - -/** - * Connect is a middleware framework for node, - * shipping with over 18 bundled middleware and a rich selection of - * 3rd-party middleware. - * - * var app = connect() - * .use(connect.logger('dev')) - * .use(connect.static('public')) - * .use(function(req, res){ - * res.end('hello world\n'); - * }) - * .listen(3000); - * - * Installation: - * - * $ npm install connect - * - * Middleware: - * - * - [logger](logger.html) request logger with custom format support - * - [csrf](csrf.html) Cross-site request forgery protection - * - [compress](compress.html) Gzip compression middleware - * - [basicAuth](basicAuth.html) basic http authentication - * - [bodyParser](bodyParser.html) extensible request body parser - * - [json](json.html) application/json parser - * - [urlencoded](urlencoded.html) application/x-www-form-urlencoded parser - * - [multipart](multipart.html) multipart/form-data parser - * - [timeout](timeout.html) request timeouts - * - [cookieParser](cookieParser.html) cookie parser - * - [session](session.html) session management support with bundled MemoryStore - * - [cookieSession](cookieSession.html) cookie-based session support - * - [methodOverride](methodOverride.html) faux HTTP method support - * - [responseTime](responseTime.html) calculates response-time and exposes via X-Response-Time - * - [staticCache](staticCache.html) memory cache layer for the static() middleware - * - [static](static.html) streaming static file server supporting `Range` and more - * - [directory](directory.html) directory listing middleware - * - [vhost](vhost.html) virtual host sub-domain mapping middleware - * - [favicon](favicon.html) efficient favicon server (with default icon) - * - [limit](limit.html) limit the bytesize of request bodies - * - [query](query.html) automatic querystring parser, populating `req.query` - * - [errorHandler](errorHandler.html) flexible error handler - * - * Links: - * - * - list of [3rd-party](https://github.com/senchalabs/connect/wiki) middleware - * - GitHub [repository](http://github.com/senchalabs/connect) - * - [test documentation](https://github.com/senchalabs/connect/blob/gh-pages/tests.md) - * - */ \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/lib/middleware/basicAuth.js b/node_modules/express/node_modules/connect/lib/middleware/basicAuth.js deleted file mode 100644 index bc7ec97..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/basicAuth.js +++ /dev/null @@ -1,103 +0,0 @@ - -/*! - * Connect - basicAuth - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../utils') - , unauthorized = utils.unauthorized; - -/** - * Basic Auth: - * - * Enfore basic authentication by providing a `callback(user, pass)`, - * which must return `true` in order to gain access. Alternatively an async - * method is provided as well, invoking `callback(user, pass, callback)`. Populates - * `req.user`. The final alternative is simply passing username / password - * strings. - * - * Simple username and password - * - * connect(connect.basicAuth('username', 'password')); - * - * Callback verification - * - * connect() - * .use(connect.basicAuth(function(user, pass){ - * return 'tj' == user & 'wahoo' == pass; - * })) - * - * Async callback verification, accepting `fn(err, user)`. - * - * connect() - * .use(connect.basicAuth(function(user, pass, fn){ - * User.authenticate({ user: user, pass: pass }, fn); - * })) - * - * @param {Function|String} callback or username - * @param {String} realm - * @api public - */ - -module.exports = function basicAuth(callback, realm) { - var username, password; - - // user / pass strings - if ('string' == typeof callback) { - username = callback; - password = realm; - if ('string' != typeof password) throw new Error('password argument required'); - realm = arguments[2]; - callback = function(user, pass){ - return user == username && pass == password; - } - } - - realm = realm || 'Authorization Required'; - - return function(req, res, next) { - var authorization = req.headers.authorization; - - if (req.user) return next(); - if (!authorization) return unauthorized(res, realm); - - var parts = authorization.split(' '); - - if (parts.length !== 2) return next(utils.error(400)); - - var scheme = parts[0] - , credentials = new Buffer(parts[1], 'base64').toString() - , index = credentials.indexOf(':'); - - if ('Basic' != scheme || index < 0) return next(utils.error(400)); - - var user = credentials.slice(0, index) - , pass = credentials.slice(index + 1); - - // async - if (callback.length >= 3) { - var pause = utils.pause(req); - callback(user, pass, function(err, user){ - if (err || !user) return unauthorized(res, realm); - req.user = req.remoteUser = user; - next(); - pause.resume(); - }); - // sync - } else { - if (callback(user, pass)) { - req.user = req.remoteUser = user; - next(); - } else { - unauthorized(res, realm); - } - } - } -}; - diff --git a/node_modules/express/node_modules/connect/lib/middleware/bodyParser.js b/node_modules/express/node_modules/connect/lib/middleware/bodyParser.js deleted file mode 100644 index 9f692cd..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/bodyParser.js +++ /dev/null @@ -1,61 +0,0 @@ - -/*! - * Connect - bodyParser - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var multipart = require('./multipart') - , urlencoded = require('./urlencoded') - , json = require('./json'); - -/** - * Body parser: - * - * Parse request bodies, supports _application/json_, - * _application/x-www-form-urlencoded_, and _multipart/form-data_. - * - * This is equivalent to: - * - * app.use(connect.json()); - * app.use(connect.urlencoded()); - * app.use(connect.multipart()); - * - * Examples: - * - * connect() - * .use(connect.bodyParser()) - * .use(function(req, res) { - * res.end('viewing user ' + req.body.user.name); - * }); - * - * $ curl -d 'user[name]=tj' http://local/ - * $ curl -d '{"user":{"name":"tj"}}' -H "Content-Type: application/json" http://local/ - * - * View [json](json.html), [urlencoded](urlencoded.html), and [multipart](multipart.html) for more info. - * - * @param {Object} options - * @return {Function} - * @api public - */ - -exports = module.exports = function bodyParser(options){ - var _urlencoded = urlencoded(options) - , _multipart = multipart(options) - , _json = json(options); - - return function bodyParser(req, res, next) { - _json(req, res, function(err){ - if (err) return next(err); - _urlencoded(req, res, function(err){ - if (err) return next(err); - _multipart(req, res, next); - }); - }); - } -}; \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/lib/middleware/compress.js b/node_modules/express/node_modules/connect/lib/middleware/compress.js deleted file mode 100644 index e71ea8c..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/compress.js +++ /dev/null @@ -1,152 +0,0 @@ -/*! - * Connect - compress - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var zlib = require('zlib'); - -/** - * Supported content-encoding methods. - */ - -exports.methods = { - gzip: zlib.createGzip - , deflate: zlib.createDeflate -}; - -/** - * Default filter function. - */ - -exports.filter = function(req, res){ - return /json|text|javascript/.test(res.getHeader('Content-Type')); -}; - -/** - * Compress: - * - * Compress response data with gzip/deflate. - * - * Filter: - * - * A `filter` callback function may be passed to - * replace the default logic of: - * - * exports.filter = function(req, res){ - * return /json|text|javascript/.test(res.getHeader('Content-Type')); - * }; - * - * Options: - * - * All remaining options are passed to the gzip/deflate - * creation functions. Consult node's docs for additional details. - * - * - `chunkSize` (default: 16*1024) - * - `windowBits` - * - `level`: 0-9 where 0 is no compression, and 9 is slow but best compression - * - `memLevel`: 1-9 low is slower but uses less memory, high is fast but uses more - * - `strategy`: compression strategy - * - * @param {Object} options - * @return {Function} - * @api public - */ - -module.exports = function compress(options) { - options = options || {}; - var names = Object.keys(exports.methods) - , filter = options.filter || exports.filter; - - return function compress(req, res, next){ - var accept = req.headers['accept-encoding'] - , vary = res.getHeader('Vary') - , write = res.write - , end = res.end - , stream - , method; - - // vary - if (!vary) { - res.setHeader('Vary', 'Accept-Encoding'); - } else if (!~vary.indexOf('Accept-Encoding')) { - res.setHeader('Vary', vary + ', Accept-Encoding'); - } - - // proxy - - res.write = function(chunk, encoding){ - if (!this.headerSent) this._implicitHeader(); - return stream - ? stream.write(new Buffer(chunk, encoding)) - : write.call(res, chunk, encoding); - }; - - res.end = function(chunk, encoding){ - if (chunk) this.write(chunk, encoding); - return stream - ? stream.end() - : end.call(res); - }; - - res.on('header', function(){ - var encoding = res.getHeader('Content-Encoding') || 'identity'; - - // already encoded - if ('identity' != encoding) return; - - // default request filter - if (!filter(req, res)) return; - - // SHOULD use identity - if (!accept) return; - - // head - if ('HEAD' == req.method) return; - - // default to gzip - if ('*' == accept.trim()) method = 'gzip'; - - // compression method - if (!method) { - for (var i = 0, len = names.length; i < len; ++i) { - if (~accept.indexOf(names[i])) { - method = names[i]; - break; - } - } - } - - // compression method - if (!method) return; - - // compression stream - stream = exports.methods[method](options); - - // header fields - res.setHeader('Content-Encoding', method); - res.removeHeader('Content-Length'); - - // compression - - stream.on('data', function(chunk){ - write.call(res, chunk); - }); - - stream.on('end', function(){ - end.call(res); - }); - - stream.on('drain', function() { - res.emit('drain'); - }); - }); - - next(); - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/cookieParser.js b/node_modules/express/node_modules/connect/lib/middleware/cookieParser.js deleted file mode 100644 index 5da23f2..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/cookieParser.js +++ /dev/null @@ -1,62 +0,0 @@ - -/*! - * Connect - cookieParser - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('./../utils') - , cookie = require('cookie'); - -/** - * Cookie parser: - * - * Parse _Cookie_ header and populate `req.cookies` - * with an object keyed by the cookie names. Optionally - * you may enabled signed cookie support by passing - * a `secret` string, which assigns `req.secret` so - * it may be used by other middleware. - * - * Examples: - * - * connect() - * .use(connect.cookieParser('optional secret string')) - * .use(function(req, res, next){ - * res.end(JSON.stringify(req.cookies)); - * }) - * - * @param {String} secret - * @return {Function} - * @api public - */ - -module.exports = function cookieParser(secret){ - return function cookieParser(req, res, next) { - if (req.cookies) return next(); - var cookies = req.headers.cookie; - - req.secret = secret; - req.cookies = {}; - req.signedCookies = {}; - - if (cookies) { - try { - req.cookies = cookie.parse(cookies); - if (secret) { - req.signedCookies = utils.parseSignedCookies(req.cookies, secret); - req.signedCookies = utils.parseJSONCookies(req.signedCookies); - } - req.cookies = utils.parseJSONCookies(req.cookies); - } catch (err) { - err.status = 400; - return next(err); - } - } - next(); - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/cookieSession.js b/node_modules/express/node_modules/connect/lib/middleware/cookieSession.js deleted file mode 100644 index 402fd55..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/cookieSession.js +++ /dev/null @@ -1,117 +0,0 @@ - -/*! - * Connect - cookieSession - * Copyright(c) 2011 Sencha Inc. - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('./../utils') - , Cookie = require('./session/cookie') - , debug = require('debug')('connect:cookieSession') - , signature = require('cookie-signature') - , crc32 = require('buffer-crc32'); - -/** - * Cookie Session: - * - * Cookie session middleware. - * - * var app = connect(); - * app.use(connect.cookieParser()); - * app.use(connect.cookieSession({ secret: 'tobo!', cookie: { maxAge: 60 * 60 * 1000 }})); - * - * Options: - * - * - `key` cookie name defaulting to `connect.sess` - * - `secret` prevents cookie tampering - * - `cookie` session cookie settings, defaulting to `{ path: '/', httpOnly: true, maxAge: null }` - * - `proxy` trust the reverse proxy when setting secure cookies (via "x-forwarded-proto") - * - * Clearing sessions: - * - * To clear the session simply set its value to `null`, - * `cookieSession()` will then respond with a 1970 Set-Cookie. - * - * req.session = null; - * - * @param {Object} options - * @return {Function} - * @api public - */ - -module.exports = function cookieSession(options){ - // TODO: utilize Session/Cookie to unify API - options = options || {}; - var key = options.key || 'connect.sess' - , trustProxy = options.proxy; - - return function cookieSession(req, res, next) { - - // req.secret is for backwards compatibility - var secret = options.secret || req.secret; - if (!secret) throw new Error('`secret` option required for cookie sessions'); - - // default session - req.session = {}; - var cookie = req.session.cookie = new Cookie(options.cookie); - - // pathname mismatch - if (0 != req.originalUrl.indexOf(cookie.path)) return next(); - - // cookieParser secret - if (!options.secret && req.secret) { - req.session = req.signedCookies[key] || {}; - req.session.cookie = cookie; - } else { - // TODO: refactor - var rawCookie = req.cookies[key]; - if (rawCookie) { - var unsigned = utils.parseSignedCookie(rawCookie, secret); - if (unsigned) { - var originalHash = crc32.signed(unsigned); - req.session = utils.parseJSONCookie(unsigned) || {}; - req.session.cookie = cookie; - } - } - } - - res.on('header', function(){ - // removed - if (!req.session) { - debug('clear session'); - cookie.expires = new Date(0); - res.setHeader('Set-Cookie', cookie.serialize(key, '')); - return; - } - - delete req.session.cookie; - - // check security - var proto = (req.headers['x-forwarded-proto'] || '').toLowerCase() - , tls = req.connection.encrypted || (trustProxy && 'https' == proto) - , secured = cookie.secure && tls; - - // only send secure cookies via https - if (cookie.secure && !secured) return debug('not secured'); - - // serialize - debug('serializing %j', req.session); - var val = 'j:' + JSON.stringify(req.session); - - // compare hashes, no need to set-cookie if unchanged - if (originalHash == crc32.signed(val)) return debug('unmodified session'); - - // set-cookie - val = 's:' + signature.sign(val, secret); - val = cookie.serialize(key, val); - debug('set-cookie %j', cookie); - res.setHeader('Set-Cookie', val); - }); - - next(); - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/csrf.js b/node_modules/express/node_modules/connect/lib/middleware/csrf.js deleted file mode 100644 index e3c353e..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/csrf.js +++ /dev/null @@ -1,73 +0,0 @@ -/*! - * Connect - csrf - * Copyright(c) 2011 Sencha Inc. - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../utils'); - -/** - * Anti CSRF: - * - * CRSF protection middleware. - * - * By default this middleware generates a token named "_csrf" - * which should be added to requests which mutate - * state, within a hidden form field, query-string etc. This - * token is validated against the visitor's `req.session._csrf` - * property. - * - * The default `value` function checks `req.body` generated - * by the `bodyParser()` middleware, `req.query` generated - * by `query()`, and the "X-CSRF-Token" header field. - * - * This middleware requires session support, thus should be added - * somewhere _below_ `session()` and `cookieParser()`. - * - * Options: - * - * - `value` a function accepting the request, returning the token - * - * @param {Object} options - * @api public - */ - -module.exports = function csrf(options) { - options = options || {}; - var value = options.value || defaultValue; - - return function(req, res, next){ - // generate CSRF token - var token = req.session._csrf || (req.session._csrf = utils.uid(24)); - - // ignore these methods - if ('GET' == req.method || 'HEAD' == req.method || 'OPTIONS' == req.method) return next(); - - // determine value - var val = value(req); - - // check - if (val != token) return next(utils.error(403)); - - next(); - } -}; - -/** - * Default value function, checking the `req.body` - * and `req.query` for the CSRF token. - * - * @param {IncomingMessage} req - * @return {String} - * @api private - */ - -function defaultValue(req) { - return (req.body && req.body._csrf) - || (req.query && req.query._csrf) - || (req.headers['x-csrf-token']); -} diff --git a/node_modules/express/node_modules/connect/lib/middleware/directory.js b/node_modules/express/node_modules/connect/lib/middleware/directory.js deleted file mode 100644 index 1c925a7..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/directory.js +++ /dev/null @@ -1,229 +0,0 @@ - -/*! - * Connect - directory - * Copyright(c) 2011 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -// TODO: icon / style for directories -// TODO: arrow key navigation -// TODO: make icons extensible - -/** - * Module dependencies. - */ - -var fs = require('fs') - , parse = require('url').parse - , utils = require('../utils') - , path = require('path') - , normalize = path.normalize - , extname = path.extname - , join = path.join; - -/*! - * Icon cache. - */ - -var cache = {}; - -/** - * Directory: - * - * Serve directory listings with the given `root` path. - * - * Options: - * - * - `hidden` display hidden (dot) files. Defaults to false. - * - `icons` display icons. Defaults to false. - * - `filter` Apply this filter function to files. Defaults to false. - * - * @param {String} root - * @param {Object} options - * @return {Function} - * @api public - */ - -exports = module.exports = function directory(root, options){ - options = options || {}; - - // root required - if (!root) throw new Error('directory() root path required'); - var hidden = options.hidden - , icons = options.icons - , filter = options.filter - , root = normalize(root); - - return function directory(req, res, next) { - if ('GET' != req.method && 'HEAD' != req.method) return next(); - - var accept = req.headers.accept || 'text/plain' - , url = parse(req.url) - , dir = decodeURIComponent(url.pathname) - , path = normalize(join(root, dir)) - , originalUrl = parse(req.originalUrl) - , originalDir = decodeURIComponent(originalUrl.pathname) - , showUp = path != root && path != root + '/'; - - // null byte(s), bad request - if (~path.indexOf('\0')) return next(utils.error(400)); - - // malicious path, forbidden - if (0 != path.indexOf(root)) return next(utils.error(403)); - - // check if we have a directory - fs.stat(path, function(err, stat){ - if (err) return 'ENOENT' == err.code - ? next() - : next(err); - - if (!stat.isDirectory()) return next(); - - // fetch files - fs.readdir(path, function(err, files){ - if (err) return next(err); - if (!hidden) files = removeHidden(files); - if (filter) files = files.filter(filter); - files.sort(); - - // content-negotiation - for (var key in exports) { - if (~accept.indexOf(key) || ~accept.indexOf('*/*')) { - exports[key](req, res, files, next, originalDir, showUp, icons); - return; - } - } - - // not acceptable - next(utils.error(406)); - }); - }); - }; -}; - -/** - * Respond with text/html. - */ - -exports.html = function(req, res, files, next, dir, showUp, icons){ - fs.readFile(__dirname + '/../public/directory.html', 'utf8', function(err, str){ - if (err) return next(err); - fs.readFile(__dirname + '/../public/style.css', 'utf8', function(err, style){ - if (err) return next(err); - if (showUp) files.unshift('..'); - str = str - .replace('{style}', style) - .replace('{files}', html(files, dir, icons)) - .replace('{directory}', dir) - .replace('{linked-path}', htmlPath(dir)); - res.setHeader('Content-Type', 'text/html'); - res.setHeader('Content-Length', str.length); - res.end(str); - }); - }); -}; - -/** - * Respond with application/json. - */ - -exports.json = function(req, res, files){ - files = JSON.stringify(files); - res.setHeader('Content-Type', 'application/json'); - res.setHeader('Content-Length', files.length); - res.end(files); -}; - -/** - * Respond with text/plain. - */ - -exports.plain = function(req, res, files){ - files = files.join('\n') + '\n'; - res.setHeader('Content-Type', 'text/plain'); - res.setHeader('Content-Length', files.length); - res.end(files); -}; - -/** - * Map html `dir`, returning a linked path. - */ - -function htmlPath(dir) { - var curr = []; - return dir.split('/').map(function(part){ - curr.push(part); - return '' + part + ''; - }).join(' / '); -} - -/** - * Map html `files`, returning an html unordered list. - */ - -function html(files, dir, useIcons) { - return '
    ' + files.map(function(file){ - var icon = '' - , classes = []; - - if (useIcons && '..' != file) { - icon = icons[extname(file)] || icons.default; - icon = ''; - classes.push('icon'); - } - - return '
  • ' - + icon + file + '
  • '; - - }).join('\n') + '
'; -} - -/** - * Load and cache the given `icon`. - * - * @param {String} icon - * @return {String} - * @api private - */ - -function load(icon) { - if (cache[icon]) return cache[icon]; - return cache[icon] = fs.readFileSync(__dirname + '/../public/icons/' + icon, 'base64'); -} - -/** - * Filter "hidden" `files`, aka files - * beginning with a `.`. - * - * @param {Array} files - * @return {Array} - * @api private - */ - -function removeHidden(files) { - return files.filter(function(file){ - return '.' != file[0]; - }); -} - -/** - * Icon map. - */ - -var icons = { - '.js': 'page_white_code_red.png' - , '.c': 'page_white_c.png' - , '.h': 'page_white_h.png' - , '.cc': 'page_white_cplusplus.png' - , '.php': 'page_white_php.png' - , '.rb': 'page_white_ruby.png' - , '.cpp': 'page_white_cplusplus.png' - , '.swf': 'page_white_flash.png' - , '.pdf': 'page_white_acrobat.png' - , 'default': 'page_white.png' -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/errorHandler.js b/node_modules/express/node_modules/connect/lib/middleware/errorHandler.js deleted file mode 100644 index 4a84edc..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/errorHandler.js +++ /dev/null @@ -1,86 +0,0 @@ -/*! - * Connect - errorHandler - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../utils') - , fs = require('fs'); - -// environment - -var env = process.env.NODE_ENV || 'development'; - -/** - * Error handler: - * - * Development error handler, providing stack traces - * and error message responses for requests accepting text, html, - * or json. - * - * Text: - * - * By default, and when _text/plain_ is accepted a simple stack trace - * or error message will be returned. - * - * JSON: - * - * When _application/json_ is accepted, connect will respond with - * an object in the form of `{ "error": error }`. - * - * HTML: - * - * When accepted connect will output a nice html stack trace. - * - * @return {Function} - * @api public - */ - -exports = module.exports = function errorHandler(){ - return function errorHandler(err, req, res, next){ - if (err.status) res.statusCode = err.status; - if (res.statusCode < 400) res.statusCode = 500; - if ('test' != env) console.error(err.stack); - var accept = req.headers.accept || ''; - // html - if (~accept.indexOf('html')) { - fs.readFile(__dirname + '/../public/style.css', 'utf8', function(e, style){ - fs.readFile(__dirname + '/../public/error.html', 'utf8', function(e, html){ - var stack = (err.stack || '') - .split('\n').slice(1) - .map(function(v){ return '
  • ' + v + '
  • '; }).join(''); - html = html - .replace('{style}', style) - .replace('{stack}', stack) - .replace('{title}', exports.title) - .replace('{statusCode}', res.statusCode) - .replace(/\{error\}/g, utils.escape(err.toString())); - res.setHeader('Content-Type', 'text/html; charset=utf-8'); - res.end(html); - }); - }); - // json - } else if (~accept.indexOf('json')) { - var error = { message: err.message, stack: err.stack }; - for (var prop in err) error[prop] = err[prop]; - var json = JSON.stringify({ error: error }); - res.setHeader('Content-Type', 'application/json'); - res.end(json); - // plain text - } else { - res.writeHead(res.statusCode, { 'Content-Type': 'text/plain' }); - res.end(err.stack); - } - }; -}; - -/** - * Template title, framework authors may override this value. - */ - -exports.title = 'Connect'; diff --git a/node_modules/express/node_modules/connect/lib/middleware/favicon.js b/node_modules/express/node_modules/connect/lib/middleware/favicon.js deleted file mode 100644 index ef54354..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/favicon.js +++ /dev/null @@ -1,80 +0,0 @@ -/*! - * Connect - favicon - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var fs = require('fs') - , utils = require('../utils'); - -/** - * Favicon: - * - * By default serves the connect favicon, or the favicon - * located by the given `path`. - * - * Options: - * - * - `maxAge` cache-control max-age directive, defaulting to 1 day - * - * Examples: - * - * Serve default favicon: - * - * connect() - * .use(connect.favicon()) - * - * Serve favicon before logging for brevity: - * - * connect() - * .use(connect.favicon()) - * .use(connect.logger('dev')) - * - * Serve custom favicon: - * - * connect() - * .use(connect.favicon('public/favicon.ico')) - * - * @param {String} path - * @param {Object} options - * @return {Function} - * @api public - */ - -module.exports = function favicon(path, options){ - var options = options || {} - , path = path || __dirname + '/../public/favicon.ico' - , maxAge = options.maxAge || 86400000 - , icon; // favicon cache - - return function favicon(req, res, next){ - if ('/favicon.ico' == req.url) { - if (icon) { - res.writeHead(200, icon.headers); - res.end(icon.body); - } else { - fs.readFile(path, function(err, buf){ - if (err) return next(err); - icon = { - headers: { - 'Content-Type': 'image/x-icon' - , 'Content-Length': buf.length - , 'ETag': '"' + utils.md5(buf) + '"' - , 'Cache-Control': 'public, max-age=' + (maxAge / 1000) - }, - body: buf - }; - res.writeHead(200, icon.headers); - res.end(icon.body); - }); - } - } else { - next(); - } - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/json.js b/node_modules/express/node_modules/connect/lib/middleware/json.js deleted file mode 100644 index 17e5918..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/json.js +++ /dev/null @@ -1,86 +0,0 @@ - -/*! - * Connect - json - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../utils') - , _limit = require('./limit'); - -/** - * noop middleware. - */ - -function noop(req, res, next) { - next(); -} - -/** - * JSON: - * - * Parse JSON request bodies, providing the - * parsed object as `req.body`. - * - * Options: - * - * - `strict` when `false` anything `JSON.parse()` accepts will be parsed - * - `reviver` used as the second "reviver" argument for JSON.parse - * - `limit` byte limit disabled by default - * - * @param {Object} options - * @return {Function} - * @api public - */ - -exports = module.exports = function(options){ - var options = options || {} - , strict = options.strict !== false; - - var limit = options.limit - ? _limit(options.limit) - : noop; - - return function json(req, res, next) { - if (req._body) return next(); - req.body = req.body || {}; - - if (!utils.hasBody(req)) return next(); - - // check Content-Type - if ('application/json' != utils.mime(req)) return next(); - - // flag as parsed - req._body = true; - - // parse - limit(req, res, function(err){ - if (err) return next(err); - var buf = ''; - req.setEncoding('utf8'); - req.on('data', function(chunk){ buf += chunk }); - req.on('end', function(){ - var first = buf.trim()[0]; - - if (0 == buf.length) { - return next(utils.error(400, 'invalid json, empty body')); - } - - if (strict && '{' != first && '[' != first) return next(utils.error(400, 'invalid json')); - try { - req.body = JSON.parse(buf, options.reviver); - } catch (err){ - err.body = buf; - err.status = 400; - return next(err); - } - next(); - }); - }); - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/limit.js b/node_modules/express/node_modules/connect/lib/middleware/limit.js deleted file mode 100644 index 09bd1c4..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/limit.js +++ /dev/null @@ -1,78 +0,0 @@ - -/*! - * Connect - limit - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../utils'), - brokenPause = utils.brokenPause; - -/** - * Limit: - * - * Limit request bodies to the given size in `bytes`. - * - * A string representation of the bytesize may also be passed, - * for example "5mb", "200kb", "1gb", etc. - * - * connect() - * .use(connect.limit('5.5mb')) - * .use(handleImageUpload) - * - * @param {Number|String} bytes - * @return {Function} - * @api public - */ - -module.exports = function limit(bytes){ - if ('string' == typeof bytes) bytes = utils.parseBytes(bytes); - if ('number' != typeof bytes) throw new Error('limit() bytes required'); - return function limit(req, res, next){ - var received = 0 - , len = req.headers['content-length'] - ? parseInt(req.headers['content-length'], 10) - : null; - - // self-awareness - if (req._limit) return next(); - req._limit = true; - - // limit by content-length - if (len && len > bytes) return next(utils.error(413)); - - // limit - if (brokenPause) { - listen(); - } else { - req.on('newListener', function handler(event) { - if (event !== 'data') return; - - req.removeListener('newListener', handler); - // Start listening at the end of the current loop - // otherwise the request will be consumed too early. - // Sideaffect is `limit` will miss the first chunk, - // but that's not a big deal. - // Unfortunately, the tests don't have large enough - // request bodies to test this. - process.nextTick(listen); - }); - }; - - next(); - - function listen() { - req.on('data', function(chunk) { - received += Buffer.isBuffer(chunk) - ? chunk.length : - Buffer.byteLength(chunk); - - if (received > bytes) req.destroy(); - }); - }; - }; -}; \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/lib/middleware/logger.js b/node_modules/express/node_modules/connect/lib/middleware/logger.js deleted file mode 100644 index de72244..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/logger.js +++ /dev/null @@ -1,339 +0,0 @@ -/*! - * Connect - logger - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var bytes = require('bytes'); - -/*! - * Log buffer. - */ - -var buf = []; - -/*! - * Default log buffer duration. - */ - -var defaultBufferDuration = 1000; - -/** - * Logger: - * - * Log requests with the given `options` or a `format` string. - * - * Options: - * - * - `format` Format string, see below for tokens - * - `stream` Output stream, defaults to _stdout_ - * - `buffer` Buffer duration, defaults to 1000ms when _true_ - * - `immediate` Write log line on request instead of response (for response times) - * - * Tokens: - * - * - `:req[header]` ex: `:req[Accept]` - * - `:res[header]` ex: `:res[Content-Length]` - * - `:http-version` - * - `:response-time` - * - `:remote-addr` - * - `:date` - * - `:method` - * - `:url` - * - `:referrer` - * - `:user-agent` - * - `:status` - * - * Formats: - * - * Pre-defined formats that ship with connect: - * - * - `default` ':remote-addr - - [:date] ":method :url HTTP/:http-version" :status :res[content-length] ":referrer" ":user-agent"' - * - `short` ':remote-addr - :method :url HTTP/:http-version :status :res[content-length] - :response-time ms' - * - `tiny` ':method :url :status :res[content-length] - :response-time ms' - * - `dev` concise output colored by response status for development use - * - * Examples: - * - * connect.logger() // default - * connect.logger('short') - * connect.logger('tiny') - * connect.logger({ immediate: true, format: 'dev' }) - * connect.logger(':method :url - :referrer') - * connect.logger(':req[content-type] -> :res[content-type]') - * connect.logger(function(tokens, req, res){ return 'some format string' }) - * - * Defining Tokens: - * - * To define a token, simply invoke `connect.logger.token()` with the - * name and a callback function. The value returned is then available - * as ":type" in this case. - * - * connect.logger.token('type', function(req, res){ return req.headers['content-type']; }) - * - * Defining Formats: - * - * All default formats are defined this way, however it's public API as well: - * - * connect.logger.format('name', 'string or function') - * - * @param {String|Function|Object} format or options - * @return {Function} - * @api public - */ - -exports = module.exports = function logger(options) { - if ('object' == typeof options) { - options = options || {}; - } else if (options) { - options = { format: options }; - } else { - options = {}; - } - - // output on request instead of response - var immediate = options.immediate; - - // format name - var fmt = exports[options.format] || options.format || exports.default; - - // compile format - if ('function' != typeof fmt) fmt = compile(fmt); - - // options - var stream = options.stream || process.stdout - , buffer = options.buffer; - - // buffering support - if (buffer) { - var realStream = stream - , interval = 'number' == typeof buffer - ? buffer - : defaultBufferDuration; - - // flush interval - setInterval(function(){ - if (buf.length) { - realStream.write(buf.join('')); - buf.length = 0; - } - }, interval); - - // swap the stream - stream = { - write: function(str){ - buf.push(str); - } - }; - } - - return function logger(req, res, next) { - req._startTime = new Date; - - // immediate - if (immediate) { - var line = fmt(exports, req, res); - if (null == line) return; - stream.write(line + '\n'); - // proxy end to output logging - } else { - var end = res.end; - res.end = function(chunk, encoding){ - res.end = end; - res.end(chunk, encoding); - var line = fmt(exports, req, res); - if (null == line) return; - stream.write(line + '\n'); - }; - } - - - next(); - }; -}; - -/** - * Compile `fmt` into a function. - * - * @param {String} fmt - * @return {Function} - * @api private - */ - -function compile(fmt) { - fmt = fmt.replace(/"/g, '\\"'); - var js = ' return "' + fmt.replace(/:([-\w]{2,})(?:\[([^\]]+)\])?/g, function(_, name, arg){ - return '"\n + (tokens["' + name + '"](req, res, "' + arg + '") || "-") + "'; - }) + '";' - return new Function('tokens, req, res', js); -}; - -/** - * Define a token function with the given `name`, - * and callback `fn(req, res)`. - * - * @param {String} name - * @param {Function} fn - * @return {Object} exports for chaining - * @api public - */ - -exports.token = function(name, fn) { - exports[name] = fn; - return this; -}; - -/** - * Define a `fmt` with the given `name`. - * - * @param {String} name - * @param {String|Function} fmt - * @return {Object} exports for chaining - * @api public - */ - -exports.format = function(name, str){ - exports[name] = str; - return this; -}; - -/** - * Default format. - */ - -exports.format('default', ':remote-addr - - [:date] ":method :url HTTP/:http-version" :status :res[content-length] ":referrer" ":user-agent"'); - -/** - * Short format. - */ - -exports.format('short', ':remote-addr - :method :url HTTP/:http-version :status :res[content-length] - :response-time ms'); - -/** - * Tiny format. - */ - -exports.format('tiny', ':method :url :status :res[content-length] - :response-time ms'); - -/** - * dev (colored) - */ - -exports.format('dev', function(tokens, req, res){ - var status = res.statusCode - , len = parseInt(res.getHeader('Content-Length'), 10) - , color = 32; - - if (status >= 500) color = 31 - else if (status >= 400) color = 33 - else if (status >= 300) color = 36; - - len = isNaN(len) - ? '' - : len = ' - ' + bytes(len); - - return '\033[90m' + req.method - + ' ' + req.originalUrl + ' ' - + '\033[' + color + 'm' + res.statusCode - + ' \033[90m' - + (new Date - req._startTime) - + 'ms' + len - + '\033[0m'; -}); - -/** - * request url - */ - -exports.token('url', function(req){ - return req.originalUrl || req.url; -}); - -/** - * request method - */ - -exports.token('method', function(req){ - return req.method; -}); - -/** - * response time in milliseconds - */ - -exports.token('response-time', function(req){ - return new Date - req._startTime; -}); - -/** - * UTC date - */ - -exports.token('date', function(){ - return new Date().toUTCString(); -}); - -/** - * response status code - */ - -exports.token('status', function(req, res){ - return res.statusCode; -}); - -/** - * normalized referrer - */ - -exports.token('referrer', function(req){ - return req.headers['referer'] || req.headers['referrer']; -}); - -/** - * remote address - */ - -exports.token('remote-addr', function(req){ - if (req.ip) return req.ip; - var sock = req.socket; - if (sock.socket) return sock.socket.remoteAddress; - return sock.remoteAddress; -}); - -/** - * HTTP version - */ - -exports.token('http-version', function(req){ - return req.httpVersionMajor + '.' + req.httpVersionMinor; -}); - -/** - * UA string - */ - -exports.token('user-agent', function(req){ - return req.headers['user-agent']; -}); - -/** - * request header - */ - -exports.token('req', function(req, res, field){ - return req.headers[field.toLowerCase()]; -}); - -/** - * response header - */ - -exports.token('res', function(req, res, field){ - return (res._headers || {})[field.toLowerCase()]; -}); - diff --git a/node_modules/express/node_modules/connect/lib/middleware/methodOverride.js b/node_modules/express/node_modules/connect/lib/middleware/methodOverride.js deleted file mode 100644 index aaf4014..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/methodOverride.js +++ /dev/null @@ -1,40 +0,0 @@ - -/*! - * Connect - methodOverride - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Method Override: - * - * Provides faux HTTP method support. - * - * Pass an optional `key` to use when checking for - * a method override, othewise defaults to _\_method_. - * The original method is available via `req.originalMethod`. - * - * @param {String} key - * @return {Function} - * @api public - */ - -module.exports = function methodOverride(key){ - key = key || "_method"; - return function methodOverride(req, res, next) { - req.originalMethod = req.originalMethod || req.method; - - // req.body - if (req.body && key in req.body) { - req.method = req.body[key].toUpperCase(); - delete req.body[key]; - // check X-HTTP-Method-Override - } else if (req.headers['x-http-method-override']) { - req.method = req.headers['x-http-method-override'].toUpperCase(); - } - - next(); - }; -}; - diff --git a/node_modules/express/node_modules/connect/lib/middleware/multipart.js b/node_modules/express/node_modules/connect/lib/middleware/multipart.js deleted file mode 100644 index 7b26fae..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/multipart.js +++ /dev/null @@ -1,133 +0,0 @@ -/*! - * Connect - multipart - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var formidable = require('formidable') - , _limit = require('./limit') - , utils = require('../utils') - , qs = require('qs'); - -/** - * noop middleware. - */ - -function noop(req, res, next) { - next(); -} - -/** - * Multipart: - * - * Parse multipart/form-data request bodies, - * providing the parsed object as `req.body` - * and `req.files`. - * - * Configuration: - * - * The options passed are merged with [formidable](https://github.com/felixge/node-formidable)'s - * `IncomingForm` object, allowing you to configure the upload directory, - * size limits, etc. For example if you wish to change the upload dir do the following. - * - * app.use(connect.multipart({ uploadDir: path })); - * - * Options: - * - * - `limit` byte limit defaulting to none - * - `defer` defers processing and exposes the Formidable form object as `req.form`. - * `next()` is called without waiting for the form's "end" event. - * This option is useful if you need to bind to the "progress" event, for example. - * - * @param {Object} options - * @return {Function} - * @api public - */ - -exports = module.exports = function(options){ - options = options || {}; - - var limit = options.limit - ? _limit(options.limit) - : noop; - - return function multipart(req, res, next) { - if (req._body) return next(); - req.body = req.body || {}; - req.files = req.files || {}; - - if (!utils.hasBody(req)) return next(); - - // ignore GET - if ('GET' == req.method || 'HEAD' == req.method) return next(); - - // check Content-Type - if ('multipart/form-data' != utils.mime(req)) return next(); - - // flag as parsed - req._body = true; - - // parse - limit(req, res, function(err){ - if (err) return next(err); - - var form = new formidable.IncomingForm - , data = {} - , files = {} - , done; - - Object.keys(options).forEach(function(key){ - form[key] = options[key]; - }); - - function ondata(name, val, data){ - if (Array.isArray(data[name])) { - data[name].push(val); - } else if (data[name]) { - data[name] = [data[name], val]; - } else { - data[name] = val; - } - } - - form.on('field', function(name, val){ - ondata(name, val, data); - }); - - form.on('file', function(name, val){ - ondata(name, val, files); - }); - - form.on('error', function(err){ - if (!options.defer) { - err.status = 400; - next(err); - } - done = true; - }); - - form.on('end', function(){ - if (done) return; - try { - req.body = qs.parse(data); - req.files = qs.parse(files); - if (!options.defer) next(); - } catch (err) { - form.emit('error', err); - } - }); - - form.parse(req); - - if (options.defer) { - req.form = form; - next(); - } - }); - } -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/query.js b/node_modules/express/node_modules/connect/lib/middleware/query.js deleted file mode 100644 index 93fc5d3..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/query.js +++ /dev/null @@ -1,46 +0,0 @@ -/*! - * Connect - query - * Copyright(c) 2011 TJ Holowaychuk - * Copyright(c) 2011 Sencha Inc. - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var qs = require('qs') - , parse = require('../utils').parseUrl; - -/** - * Query: - * - * Automatically parse the query-string when available, - * populating the `req.query` object. - * - * Examples: - * - * connect() - * .use(connect.query()) - * .use(function(req, res){ - * res.end(JSON.stringify(req.query)); - * }); - * - * The `options` passed are provided to qs.parse function. - * - * @param {Object} options - * @return {Function} - * @api public - */ - -module.exports = function query(options){ - return function query(req, res, next){ - if (!req.query) { - req.query = ~req.url.indexOf('?') - ? qs.parse(parse(req).query, options) - : {}; - } - - next(); - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/responseTime.js b/node_modules/express/node_modules/connect/lib/middleware/responseTime.js deleted file mode 100644 index 62abc04..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/responseTime.js +++ /dev/null @@ -1,32 +0,0 @@ - -/*! - * Connect - responseTime - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Reponse time: - * - * Adds the `X-Response-Time` header displaying the response - * duration in milliseconds. - * - * @return {Function} - * @api public - */ - -module.exports = function responseTime(){ - return function(req, res, next){ - var start = new Date; - - if (res._responseTime) return next(); - res._responseTime = true; - - res.on('header', function(){ - var duration = new Date - start; - res.setHeader('X-Response-Time', duration + 'ms'); - }); - - next(); - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/session.js b/node_modules/express/node_modules/connect/lib/middleware/session.js deleted file mode 100644 index 9be6c8b..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/session.js +++ /dev/null @@ -1,356 +0,0 @@ - -/*! - * Connect - session - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Session = require('./session/session') - , debug = require('debug')('connect:session') - , MemoryStore = require('./session/memory') - , signature = require('cookie-signature') - , Cookie = require('./session/cookie') - , Store = require('./session/store') - , utils = require('./../utils') - , parse = utils.parseUrl - , crc32 = require('buffer-crc32'); - -// environment - -var env = process.env.NODE_ENV; - -/** - * Expose the middleware. - */ - -exports = module.exports = session; - -/** - * Expose constructors. - */ - -exports.Store = Store; -exports.Cookie = Cookie; -exports.Session = Session; -exports.MemoryStore = MemoryStore; - -/** - * Warning message for `MemoryStore` usage in production. - */ - -var warning = 'Warning: connection.session() MemoryStore is not\n' - + 'designed for a production environment, as it will leak\n' - + 'memory, and will not scale past a single process.'; - -/** - * Session: - * - * Setup session store with the given `options`. - * - * Session data is _not_ saved in the cookie itself, however - * cookies are used, so we must use the [cookieParser()](cookieParser.html) - * middleware _before_ `session()`. - * - * Examples: - * - * connect() - * .use(connect.cookieParser()) - * .use(connect.session({ secret: 'keyboard cat', key: 'sid', cookie: { secure: true }})) - * - * Options: - * - * - `key` cookie name defaulting to `connect.sid` - * - `store` session store instance - * - `secret` session cookie is signed with this secret to prevent tampering - * - `cookie` session cookie settings, defaulting to `{ path: '/', httpOnly: true, maxAge: null }` - * - `proxy` trust the reverse proxy when setting secure cookies (via "x-forwarded-proto") - * - * Cookie option: - * - * By default `cookie.maxAge` is `null`, meaning no "expires" parameter is set - * so the cookie becomes a browser-session cookie. When the user closes the - * browser the cookie (and session) will be removed. - * - * ## req.session - * - * To store or access session data, simply use the request property `req.session`, - * which is (generally) serialized as JSON by the store, so nested objects - * are typically fine. For example below is a user-specific view counter: - * - * connect() - * .use(connect.favicon()) - * .use(connect.cookieParser()) - * .use(connect.session({ secret: 'keyboard cat', cookie: { maxAge: 60000 }})) - * .use(function(req, res, next){ - * var sess = req.session; - * if (sess.views) { - * res.setHeader('Content-Type', 'text/html'); - * res.write('

    views: ' + sess.views + '

    '); - * res.write('

    expires in: ' + (sess.cookie.maxAge / 1000) + 's

    '); - * res.end(); - * sess.views++; - * } else { - * sess.views = 1; - * res.end('welcome to the session demo. refresh!'); - * } - * } - * )).listen(3000); - * - * ## Session#regenerate() - * - * To regenerate the session simply invoke the method, once complete - * a new SID and `Session` instance will be initialized at `req.session`. - * - * req.session.regenerate(function(err){ - * // will have a new session here - * }); - * - * ## Session#destroy() - * - * Destroys the session, removing `req.session`, will be re-generated next request. - * - * req.session.destroy(function(err){ - * // cannot access session here - * }); - * - * ## Session#reload() - * - * Reloads the session data. - * - * req.session.reload(function(err){ - * // session updated - * }); - * - * ## Session#save() - * - * Save the session. - * - * req.session.save(function(err){ - * // session saved - * }); - * - * ## Session#touch() - * - * Updates the `.maxAge` property. Typically this is - * not necessary to call, as the session middleware does this for you. - * - * ## Session#cookie - * - * Each session has a unique cookie object accompany it. This allows - * you to alter the session cookie per visitor. For example we can - * set `req.session.cookie.expires` to `false` to enable the cookie - * to remain for only the duration of the user-agent. - * - * ## Session#maxAge - * - * Alternatively `req.session.cookie.maxAge` will return the time - * remaining in milliseconds, which we may also re-assign a new value - * to adjust the `.expires` property appropriately. The following - * are essentially equivalent - * - * var hour = 3600000; - * req.session.cookie.expires = new Date(Date.now() + hour); - * req.session.cookie.maxAge = hour; - * - * For example when `maxAge` is set to `60000` (one minute), and 30 seconds - * has elapsed it will return `30000` until the current request has completed, - * at which time `req.session.touch()` is called to reset `req.session.maxAge` - * to its original value. - * - * req.session.cookie.maxAge; - * // => 30000 - * - * Session Store Implementation: - * - * Every session store _must_ implement the following methods - * - * - `.get(sid, callback)` - * - `.set(sid, session, callback)` - * - `.destroy(sid, callback)` - * - * Recommended methods include, but are not limited to: - * - * - `.length(callback)` - * - `.clear(callback)` - * - * For an example implementation view the [connect-redis](http://github.com/visionmedia/connect-redis) repo. - * - * @param {Object} options - * @return {Function} - * @api public - */ - -function session(options){ - var options = options || {} - , key = options.key || 'connect.sid' - , store = options.store || new MemoryStore - , cookie = options.cookie || {} - , trustProxy = options.proxy - , storeReady = true; - - // notify user that this store is not - // meant for a production environment - if ('production' == env && store instanceof MemoryStore) { - console.warn(warning); - } - - // generates the new session - store.generate = function(req){ - req.sessionID = utils.uid(24); - req.session = new Session(req); - req.session.cookie = new Cookie(cookie); - }; - - store.on('disconnect', function(){ storeReady = false; }); - store.on('connect', function(){ storeReady = true; }); - - return function session(req, res, next) { - // self-awareness - if (req.session) return next(); - - // Handle connection as if there is no session if - // the store has temporarily disconnected etc - if (!storeReady) return debug('store is disconnected'), next(); - - // pathname mismatch - if (0 != req.originalUrl.indexOf(cookie.path || '/')) return next(); - - // backwards compatibility for signed cookies - // req.secret is passed from the cookie parser middleware - var secret = options.secret || req.secret; - - // ensure secret is available or bail - if (!secret) throw new Error('`secret` option required for sessions'); - - // parse url - var originalHash - , originalId; - - // expose store - req.sessionStore = store; - - // grab the session cookie value and check the signature - var rawCookie = req.cookies[key]; - - // get signedCookies for backwards compat with signed cookies - var unsignedCookie = req.signedCookies[key]; - - if (!unsignedCookie && rawCookie) { - unsignedCookie = utils.parseSignedCookie(rawCookie, secret); - } - - // set-cookie - res.on('header', function(){ - if (!req.session) return; - var cookie = req.session.cookie - , proto = (req.headers['x-forwarded-proto'] || '').split(',')[0].toLowerCase().trim() - , tls = req.connection.encrypted || (trustProxy && 'https' == proto) - , secured = cookie.secure && tls - , isNew = unsignedCookie != req.sessionID; - - // only send secure cookies via https - if (cookie.secure && !secured) return debug('not secured'); - - // long expires, handle expiry server-side - if (!isNew && cookie.hasLongExpires) return debug('already set cookie'); - - // browser-session length cookie - if (null == cookie.expires) { - if (!isNew) return debug('already set browser-session cookie'); - // compare hashes and ids - } else if (originalHash == hash(req.session) && originalId == req.session.id) { - return debug('unmodified session'); - } - - var val = 's:' + signature.sign(req.sessionID, secret); - val = cookie.serialize(key, val); - debug('set-cookie %s', val); - res.setHeader('Set-Cookie', val); - }); - - // proxy end() to commit the session - var end = res.end; - res.end = function(data, encoding){ - res.end = end; - if (!req.session) return res.end(data, encoding); - debug('saving'); - req.session.resetMaxAge(); - req.session.save(function(err){ - if (err) console.error(err.stack); - debug('saved'); - res.end(data, encoding); - }); - }; - - // generate the session - function generate() { - store.generate(req); - } - - // get the sessionID from the cookie - req.sessionID = unsignedCookie; - - // generate a session if the browser doesn't send a sessionID - if (!req.sessionID) { - debug('no SID sent, generating session'); - generate(); - next(); - return; - } - - // generate the session object - var pause = utils.pause(req); - debug('fetching %s', req.sessionID); - store.get(req.sessionID, function(err, sess){ - // proxy to resume() events - var _next = next; - next = function(err){ - _next(err); - pause.resume(); - }; - - // error handling - if (err) { - debug('error %j', err); - if ('ENOENT' == err.code) { - generate(); - next(); - } else { - next(err); - } - // no session - } else if (!sess) { - debug('no session found'); - generate(); - next(); - // populate req.session - } else { - debug('session found'); - store.createSession(req, sess); - originalId = req.sessionID; - originalHash = hash(sess); - next(); - } - }); - }; -}; - -/** - * Hash the given `sess` object omitting changes - * to `.cookie`. - * - * @param {Object} sess - * @return {String} - * @api private - */ - -function hash(sess) { - return crc32.signed(JSON.stringify(sess, function(key, val){ - if ('cookie' != key) return val; - })); -} diff --git a/node_modules/express/node_modules/connect/lib/middleware/session/cookie.js b/node_modules/express/node_modules/connect/lib/middleware/session/cookie.js deleted file mode 100644 index cdce2a5..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/session/cookie.js +++ /dev/null @@ -1,140 +0,0 @@ - -/*! - * Connect - session - Cookie - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../../utils') - , cookie = require('cookie'); - -/** - * Initialize a new `Cookie` with the given `options`. - * - * @param {IncomingMessage} req - * @param {Object} options - * @api private - */ - -var Cookie = module.exports = function Cookie(options) { - this.path = '/'; - this.maxAge = null; - this.httpOnly = true; - if (options) utils.merge(this, options); - this.originalMaxAge = undefined == this.originalMaxAge - ? this.maxAge - : this.originalMaxAge; -}; - -/*! - * Prototype. - */ - -Cookie.prototype = { - - /** - * Set expires `date`. - * - * @param {Date} date - * @api public - */ - - set expires(date) { - this._expires = date; - this.originalMaxAge = this.maxAge; - }, - - /** - * Get expires `date`. - * - * @return {Date} - * @api public - */ - - get expires() { - return this._expires; - }, - - /** - * Set expires via max-age in `ms`. - * - * @param {Number} ms - * @api public - */ - - set maxAge(ms) { - this.expires = 'number' == typeof ms - ? new Date(Date.now() + ms) - : ms; - }, - - /** - * Get expires max-age in `ms`. - * - * @return {Number} - * @api public - */ - - get maxAge() { - return this.expires instanceof Date - ? this.expires.valueOf() - Date.now() - : this.expires; - }, - - /** - * Return cookie data object. - * - * @return {Object} - * @api private - */ - - get data() { - return { - originalMaxAge: this.originalMaxAge - , expires: this._expires - , secure: this.secure - , httpOnly: this.httpOnly - , domain: this.domain - , path: this.path - } - }, - - /** - * Check if the cookie has a reasonably large max-age. - * - * @return {Boolean} - * @api private - */ - - get hasLongExpires() { - var week = 604800000; - return this.maxAge > (4 * week); - }, - - /** - * Return a serialized cookie string. - * - * @return {String} - * @api public - */ - - serialize: function(name, val){ - return cookie.serialize(name, val, this.data); - }, - - /** - * Return JSON representation of this cookie. - * - * @return {Object} - * @api private - */ - - toJSON: function(){ - return this.data; - } -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/session/memory.js b/node_modules/express/node_modules/connect/lib/middleware/session/memory.js deleted file mode 100644 index fb93939..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/session/memory.js +++ /dev/null @@ -1,129 +0,0 @@ - -/*! - * Connect - session - MemoryStore - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Store = require('./store'); - -/** - * Initialize a new `MemoryStore`. - * - * @api public - */ - -var MemoryStore = module.exports = function MemoryStore() { - this.sessions = {}; -}; - -/** - * Inherit from `Store.prototype`. - */ - -MemoryStore.prototype.__proto__ = Store.prototype; - -/** - * Attempt to fetch session by the given `sid`. - * - * @param {String} sid - * @param {Function} fn - * @api public - */ - -MemoryStore.prototype.get = function(sid, fn){ - var self = this; - process.nextTick(function(){ - var expires - , sess = self.sessions[sid]; - if (sess) { - sess = JSON.parse(sess); - expires = 'string' == typeof sess.cookie.expires - ? new Date(sess.cookie.expires) - : sess.cookie.expires; - if (!expires || new Date < expires) { - fn(null, sess); - } else { - self.destroy(sid, fn); - } - } else { - fn(); - } - }); -}; - -/** - * Commit the given `sess` object associated with the given `sid`. - * - * @param {String} sid - * @param {Session} sess - * @param {Function} fn - * @api public - */ - -MemoryStore.prototype.set = function(sid, sess, fn){ - var self = this; - process.nextTick(function(){ - self.sessions[sid] = JSON.stringify(sess); - fn && fn(); - }); -}; - -/** - * Destroy the session associated with the given `sid`. - * - * @param {String} sid - * @api public - */ - -MemoryStore.prototype.destroy = function(sid, fn){ - var self = this; - process.nextTick(function(){ - delete self.sessions[sid]; - fn && fn(); - }); -}; - -/** - * Invoke the given callback `fn` with all active sessions. - * - * @param {Function} fn - * @api public - */ - -MemoryStore.prototype.all = function(fn){ - var arr = [] - , keys = Object.keys(this.sessions); - for (var i = 0, len = keys.length; i < len; ++i) { - arr.push(this.sessions[keys[i]]); - } - fn(null, arr); -}; - -/** - * Clear all sessions. - * - * @param {Function} fn - * @api public - */ - -MemoryStore.prototype.clear = function(fn){ - this.sessions = {}; - fn && fn(); -}; - -/** - * Fetch number of sessions. - * - * @param {Function} fn - * @api public - */ - -MemoryStore.prototype.length = function(fn){ - fn(null, Object.keys(this.sessions).length); -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/session/session.js b/node_modules/express/node_modules/connect/lib/middleware/session/session.js deleted file mode 100644 index 0dd4b40..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/session/session.js +++ /dev/null @@ -1,116 +0,0 @@ - -/*! - * Connect - session - Session - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../../utils'); - -/** - * Create a new `Session` with the given request and `data`. - * - * @param {IncomingRequest} req - * @param {Object} data - * @api private - */ - -var Session = module.exports = function Session(req, data) { - Object.defineProperty(this, 'req', { value: req }); - Object.defineProperty(this, 'id', { value: req.sessionID }); - if ('object' == typeof data) utils.merge(this, data); -}; - -/** - * Update reset `.cookie.maxAge` to prevent - * the cookie from expiring when the - * session is still active. - * - * @return {Session} for chaining - * @api public - */ - -Session.prototype.touch = function(){ - return this.resetMaxAge(); -}; - -/** - * Reset `.maxAge` to `.originalMaxAge`. - * - * @return {Session} for chaining - * @api public - */ - -Session.prototype.resetMaxAge = function(){ - this.cookie.maxAge = this.cookie.originalMaxAge; - return this; -}; - -/** - * Save the session data with optional callback `fn(err)`. - * - * @param {Function} fn - * @return {Session} for chaining - * @api public - */ - -Session.prototype.save = function(fn){ - this.req.sessionStore.set(this.id, this, fn || function(){}); - return this; -}; - -/** - * Re-loads the session data _without_ altering - * the maxAge properties. Invokes the callback `fn(err)`, - * after which time if no exception has occurred the - * `req.session` property will be a new `Session` object, - * although representing the same session. - * - * @param {Function} fn - * @return {Session} for chaining - * @api public - */ - -Session.prototype.reload = function(fn){ - var req = this.req - , store = this.req.sessionStore; - store.get(this.id, function(err, sess){ - if (err) return fn(err); - if (!sess) return fn(new Error('failed to load session')); - store.createSession(req, sess); - fn(); - }); - return this; -}; - -/** - * Destroy `this` session. - * - * @param {Function} fn - * @return {Session} for chaining - * @api public - */ - -Session.prototype.destroy = function(fn){ - delete this.req.session; - this.req.sessionStore.destroy(this.id, fn); - return this; -}; - -/** - * Regenerate this request's session. - * - * @param {Function} fn - * @return {Session} for chaining - * @api public - */ - -Session.prototype.regenerate = function(fn){ - this.req.sessionStore.regenerate(this.req, fn); - return this; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/session/store.js b/node_modules/express/node_modules/connect/lib/middleware/session/store.js deleted file mode 100644 index 54294cb..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/session/store.js +++ /dev/null @@ -1,84 +0,0 @@ - -/*! - * Connect - session - Store - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var EventEmitter = require('events').EventEmitter - , Session = require('./session') - , Cookie = require('./cookie'); - -/** - * Initialize abstract `Store`. - * - * @api private - */ - -var Store = module.exports = function Store(options){}; - -/** - * Inherit from `EventEmitter.prototype`. - */ - -Store.prototype.__proto__ = EventEmitter.prototype; - -/** - * Re-generate the given requests's session. - * - * @param {IncomingRequest} req - * @return {Function} fn - * @api public - */ - -Store.prototype.regenerate = function(req, fn){ - var self = this; - this.destroy(req.sessionID, function(err){ - self.generate(req); - fn(err); - }); -}; - -/** - * Load a `Session` instance via the given `sid` - * and invoke the callback `fn(err, sess)`. - * - * @param {String} sid - * @param {Function} fn - * @api public - */ - -Store.prototype.load = function(sid, fn){ - var self = this; - this.get(sid, function(err, sess){ - if (err) return fn(err); - if (!sess) return fn(); - var req = { sessionID: sid, sessionStore: self }; - sess = self.createSession(req, sess); - fn(null, sess); - }); -}; - -/** - * Create session from JSON `sess` data. - * - * @param {IncomingRequest} req - * @param {Object} sess - * @return {Session} - * @api private - */ - -Store.prototype.createSession = function(req, sess){ - var expires = sess.cookie.expires - , orig = sess.cookie.originalMaxAge; - sess.cookie = new Cookie(sess.cookie); - if ('string' == typeof expires) sess.cookie.expires = new Date(expires); - sess.cookie.originalMaxAge = orig; - req.session = new Session(req, sess); - return req.session; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/static.js b/node_modules/express/node_modules/connect/lib/middleware/static.js deleted file mode 100644 index f69b58e..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/static.js +++ /dev/null @@ -1,95 +0,0 @@ -/*! - * Connect - static - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var send = require('send') - , utils = require('../utils') - , parse = utils.parseUrl - , url = require('url'); - -/** - * Static: - * - * Static file server with the given `root` path. - * - * Examples: - * - * var oneDay = 86400000; - * - * connect() - * .use(connect.static(__dirname + '/public')) - * - * connect() - * .use(connect.static(__dirname + '/public', { maxAge: oneDay })) - * - * Options: - * - * - `maxAge` Browser cache maxAge in milliseconds. defaults to 0 - * - `hidden` Allow transfer of hidden files. defaults to false - * - `redirect` Redirect to trailing "/" when the pathname is a dir. defaults to true - * - `index` Default file name, defaults to 'index.html' - * - * @param {String} root - * @param {Object} options - * @return {Function} - * @api public - */ - -exports = module.exports = function static(root, options){ - options = options || {}; - - // root required - if (!root) throw new Error('static() root path required'); - - // default redirect - var redirect = false !== options.redirect; - - return function static(req, res, next) { - if ('GET' != req.method && 'HEAD' != req.method) return next(); - var path = parse(req).pathname; - var pause = utils.pause(req); - - function resume() { - next(); - pause.resume(); - } - - function directory() { - if (!redirect) return resume(); - var pathname = url.parse(req.originalUrl).pathname; - res.statusCode = 301; - res.setHeader('Location', pathname + '/'); - res.end('Redirecting to ' + utils.escape(pathname) + '/'); - } - - function error(err) { - if (404 == err.status) return resume(); - next(err); - } - - send(req, path) - .maxage(options.maxAge || 0) - .root(root) - .index(options.index || 'index.html') - .hidden(options.hidden) - .on('error', error) - .on('directory', directory) - .pipe(res); - }; -}; - -/** - * Expose mime module. - * - * If you wish to extend the mime table use this - * reference to the "mime" module in the npm registry. - */ - -exports.mime = send.mime; diff --git a/node_modules/express/node_modules/connect/lib/middleware/staticCache.js b/node_modules/express/node_modules/connect/lib/middleware/staticCache.js deleted file mode 100644 index 7354a8f..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/staticCache.js +++ /dev/null @@ -1,231 +0,0 @@ - -/*! - * Connect - staticCache - * Copyright(c) 2011 Sencha Inc. - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../utils') - , Cache = require('../cache') - , fresh = require('fresh'); - -/** - * Static cache: - * - * Enables a memory cache layer on top of - * the `static()` middleware, serving popular - * static files. - * - * By default a maximum of 128 objects are - * held in cache, with a max of 256k each, - * totalling ~32mb. - * - * A Least-Recently-Used (LRU) cache algo - * is implemented through the `Cache` object, - * simply rotating cache objects as they are - * hit. This means that increasingly popular - * objects maintain their positions while - * others get shoved out of the stack and - * garbage collected. - * - * Benchmarks: - * - * static(): 2700 rps - * node-static: 5300 rps - * static() + staticCache(): 7500 rps - * - * Options: - * - * - `maxObjects` max cache objects [128] - * - `maxLength` max cache object length 256kb - * - * @param {Object} options - * @return {Function} - * @api public - */ - -module.exports = function staticCache(options){ - var options = options || {} - , cache = new Cache(options.maxObjects || 128) - , maxlen = options.maxLength || 1024 * 256; - - console.warn('connect.staticCache() is deprecated and will be removed in 3.0'); - console.warn('use varnish or similar reverse proxy caches.'); - - return function staticCache(req, res, next){ - var key = cacheKey(req) - , ranges = req.headers.range - , hasCookies = req.headers.cookie - , hit = cache.get(key); - - // cache static - // TODO: change from staticCache() -> cache() - // and make this work for any request - req.on('static', function(stream){ - var headers = res._headers - , cc = utils.parseCacheControl(headers['cache-control'] || '') - , contentLength = headers['content-length'] - , hit; - - // dont cache set-cookie responses - if (headers['set-cookie']) return hasCookies = true; - - // dont cache when cookies are present - if (hasCookies) return; - - // ignore larger files - if (!contentLength || contentLength > maxlen) return; - - // don't cache partial files - if (headers['content-range']) return; - - // dont cache items we shouldn't be - // TODO: real support for must-revalidate / no-cache - if ( cc['no-cache'] - || cc['no-store'] - || cc['private'] - || cc['must-revalidate']) return; - - // if already in cache then validate - if (hit = cache.get(key)){ - if (headers.etag == hit[0].etag) { - hit[0].date = new Date; - return; - } else { - cache.remove(key); - } - } - - // validation notifiactions don't contain a steam - if (null == stream) return; - - // add the cache object - var arr = []; - - // store the chunks - stream.on('data', function(chunk){ - arr.push(chunk); - }); - - // flag it as complete - stream.on('end', function(){ - var cacheEntry = cache.add(key); - delete headers['x-cache']; // Clean up (TODO: others) - cacheEntry.push(200); - cacheEntry.push(headers); - cacheEntry.push.apply(cacheEntry, arr); - }); - }); - - if (req.method == 'GET' || req.method == 'HEAD') { - if (ranges) { - next(); - } else if (!hasCookies && hit && !mustRevalidate(req, hit)) { - res.setHeader('X-Cache', 'HIT'); - respondFromCache(req, res, hit); - } else { - res.setHeader('X-Cache', 'MISS'); - next(); - } - } else { - next(); - } - } -}; - -/** - * Respond with the provided cached value. - * TODO: Assume 200 code, that's iffy. - * - * @param {Object} req - * @param {Object} res - * @param {Object} cacheEntry - * @return {String} - * @api private - */ - -function respondFromCache(req, res, cacheEntry) { - var status = cacheEntry[0] - , headers = utils.merge({}, cacheEntry[1]) - , content = cacheEntry.slice(2); - - headers.age = (new Date - new Date(headers.date)) / 1000 || 0; - - switch (req.method) { - case 'HEAD': - res.writeHead(status, headers); - res.end(); - break; - case 'GET': - if (utils.conditionalGET(req) && fresh(req.headers, headers)) { - headers['content-length'] = 0; - res.writeHead(304, headers); - res.end(); - } else { - res.writeHead(status, headers); - - function write() { - while (content.length) { - if (false === res.write(content.shift())) { - res.once('drain', write); - return; - } - } - res.end(); - } - - write(); - } - break; - default: - // This should never happen. - res.writeHead(500, ''); - res.end(); - } -} - -/** - * Determine whether or not a cached value must be revalidated. - * - * @param {Object} req - * @param {Object} cacheEntry - * @return {String} - * @api private - */ - -function mustRevalidate(req, cacheEntry) { - var cacheHeaders = cacheEntry[1] - , reqCC = utils.parseCacheControl(req.headers['cache-control'] || '') - , cacheCC = utils.parseCacheControl(cacheHeaders['cache-control'] || '') - , cacheAge = (new Date - new Date(cacheHeaders.date)) / 1000 || 0; - - if ( cacheCC['no-cache'] - || cacheCC['must-revalidate'] - || cacheCC['proxy-revalidate']) return true; - - if (reqCC['no-cache']) return true; - - if (null != reqCC['max-age']) return reqCC['max-age'] < cacheAge; - - if (null != cacheCC['max-age']) return cacheCC['max-age'] < cacheAge; - - return false; -} - -/** - * The key to use in the cache. For now, this is the URL path and query. - * - * 'http://example.com?key=value' -> '/?key=value' - * - * @param {Object} req - * @return {String} - * @api private - */ - -function cacheKey(req) { - return utils.parseUrl(req).path; -} diff --git a/node_modules/express/node_modules/connect/lib/middleware/timeout.js b/node_modules/express/node_modules/connect/lib/middleware/timeout.js deleted file mode 100644 index dba4654..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/timeout.js +++ /dev/null @@ -1,55 +0,0 @@ -/*! - * Connect - timeout - * Ported from https://github.com/LearnBoost/connect-timeout - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var debug = require('debug')('connect:timeout'); - -/** - * Timeout: - * - * Times out the request in `ms`, defaulting to `5000`. The - * method `req.clearTimeout()` is added to revert this behaviour - * programmatically within your application's middleware, routes, etc. - * - * The timeout error is passed to `next()` so that you may customize - * the response behaviour. This error has the `.timeout` property as - * well as `.status == 408`. - * - * @param {Number} ms - * @return {Function} - * @api public - */ - -module.exports = function timeout(ms) { - ms = ms || 5000; - - return function(req, res, next) { - var id = setTimeout(function(){ - req.emit('timeout', ms); - }, ms); - - req.on('timeout', function(){ - if (res.headerSent) return debug('response started, cannot timeout'); - var err = new Error('Response timeout'); - err.timeout = ms; - err.status = 503; - next(err); - }); - - req.clearTimeout = function(){ - clearTimeout(id); - }; - - res.on('header', function(){ - clearTimeout(id); - }); - - next(); - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/urlencoded.js b/node_modules/express/node_modules/connect/lib/middleware/urlencoded.js deleted file mode 100644 index cceafc0..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/urlencoded.js +++ /dev/null @@ -1,78 +0,0 @@ - -/*! - * Connect - urlencoded - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var utils = require('../utils') - , _limit = require('./limit') - , qs = require('qs'); - -/** - * noop middleware. - */ - -function noop(req, res, next) { - next(); -} - -/** - * Urlencoded: - * - * Parse x-ww-form-urlencoded request bodies, - * providing the parsed object as `req.body`. - * - * Options: - * - * - `limit` byte limit disabled by default - * - * @param {Object} options - * @return {Function} - * @api public - */ - -exports = module.exports = function(options){ - options = options || {}; - - var limit = options.limit - ? _limit(options.limit) - : noop; - - return function urlencoded(req, res, next) { - if (req._body) return next(); - req.body = req.body || {}; - - if (!utils.hasBody(req)) return next(); - - // check Content-Type - if ('application/x-www-form-urlencoded' != utils.mime(req)) return next(); - - // flag as parsed - req._body = true; - - // parse - limit(req, res, function(err){ - if (err) return next(err); - var buf = ''; - req.setEncoding('utf8'); - req.on('data', function(chunk){ buf += chunk }); - req.on('end', function(){ - try { - req.body = buf.length - ? qs.parse(buf, options) - : {}; - next(); - } catch (err){ - err.body = buf; - next(err); - } - }); - }); - } -}; diff --git a/node_modules/express/node_modules/connect/lib/middleware/vhost.js b/node_modules/express/node_modules/connect/lib/middleware/vhost.js deleted file mode 100644 index abbb050..0000000 --- a/node_modules/express/node_modules/connect/lib/middleware/vhost.js +++ /dev/null @@ -1,40 +0,0 @@ - -/*! - * Connect - vhost - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Vhost: - * - * Setup vhost for the given `hostname` and `server`. - * - * connect() - * .use(connect.vhost('foo.com', fooApp)) - * .use(connect.vhost('bar.com', barApp)) - * .use(connect.vhost('*.com', mainApp)) - * - * The `server` may be a Connect server or - * a regular Node `http.Server`. - * - * @param {String} hostname - * @param {Server} server - * @return {Function} - * @api public - */ - -module.exports = function vhost(hostname, server){ - if (!hostname) throw new Error('vhost hostname required'); - if (!server) throw new Error('vhost server required'); - var regexp = new RegExp('^' + hostname.replace(/[^*\w]/g, '\\$&').replace(/[*]/g, '(?:.*?)') + '$', 'i'); - if (server.onvhost) server.onvhost(hostname); - return function vhost(req, res, next){ - if (!req.headers.host) return next(); - var host = req.headers.host.split(':')[0]; - if (!regexp.test(host)) return next(); - if ('function' == typeof server) return server(req, res, next); - server.emit('request', req, res); - }; -}; diff --git a/node_modules/express/node_modules/connect/lib/patch.js b/node_modules/express/node_modules/connect/lib/patch.js deleted file mode 100644 index 7cf0012..0000000 --- a/node_modules/express/node_modules/connect/lib/patch.js +++ /dev/null @@ -1,79 +0,0 @@ - -/*! - * Connect - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var http = require('http') - , res = http.ServerResponse.prototype - , setHeader = res.setHeader - , _renderHeaders = res._renderHeaders - , writeHead = res.writeHead; - -// apply only once - -if (!res._hasConnectPatch) { - - /** - * Provide a public "header sent" flag - * until node does. - * - * @return {Boolean} - * @api public - */ - - res.__defineGetter__('headerSent', function(){ - return this._header; - }); - - /** - * Set header `field` to `val`, special-casing - * the `Set-Cookie` field for multiple support. - * - * @param {String} field - * @param {String} val - * @api public - */ - - res.setHeader = function(field, val){ - var key = field.toLowerCase() - , prev; - - // special-case Set-Cookie - if (this._headers && 'set-cookie' == key) { - if (prev = this.getHeader(field)) { - val = Array.isArray(prev) - ? prev.concat(val) - : [prev, val]; - } - // charset - } else if ('content-type' == key && this.charset) { - val += '; charset=' + this.charset; - } - - return setHeader.call(this, field, val); - }; - - /** - * Proxy to emit "header" event. - */ - - res._renderHeaders = function(){ - if (!this._emittedHeader) this.emit('header'); - this._emittedHeader = true; - return _renderHeaders.call(this); - }; - - res.writeHead = function(){ - if (!this._emittedHeader) this.emit('header'); - this._emittedHeader = true; - return writeHead.apply(this, arguments); - }; - - res._hasConnectPatch = true; -} diff --git a/node_modules/express/node_modules/connect/lib/proto.js b/node_modules/express/node_modules/connect/lib/proto.js deleted file mode 100644 index b304cf7..0000000 --- a/node_modules/express/node_modules/connect/lib/proto.js +++ /dev/null @@ -1,230 +0,0 @@ - -/*! - * Connect - HTTPServer - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var http = require('http') - , utils = require('./utils') - , debug = require('debug')('connect:dispatcher'); - -// prototype - -var app = module.exports = {}; - -// environment - -var env = process.env.NODE_ENV || 'development'; - -/** - * Utilize the given middleware `handle` to the given `route`, - * defaulting to _/_. This "route" is the mount-point for the - * middleware, when given a value other than _/_ the middleware - * is only effective when that segment is present in the request's - * pathname. - * - * For example if we were to mount a function at _/admin_, it would - * be invoked on _/admin_, and _/admin/settings_, however it would - * not be invoked for _/_, or _/posts_. - * - * Examples: - * - * var app = connect(); - * app.use(connect.favicon()); - * app.use(connect.logger()); - * app.use(connect.static(__dirname + '/public')); - * - * If we wanted to prefix static files with _/public_, we could - * "mount" the `static()` middleware: - * - * app.use('/public', connect.static(__dirname + '/public')); - * - * This api is chainable, so the following is valid: - * - * connect() - * .use(connect.favicon()) - * .use(connect.logger()) - * .use(connect.static(__dirname + '/public')) - * .listen(3000); - * - * @param {String|Function|Server} route, callback or server - * @param {Function|Server} callback or server - * @return {Server} for chaining - * @api public - */ - -app.use = function(route, fn){ - // default route to '/' - if ('string' != typeof route) { - fn = route; - route = '/'; - } - - // wrap sub-apps - if ('function' == typeof fn.handle) { - var server = fn; - fn.route = route; - fn = function(req, res, next){ - server.handle(req, res, next); - }; - } - - // wrap vanilla http.Servers - if (fn instanceof http.Server) { - fn = fn.listeners('request')[0]; - } - - // strip trailing slash - if ('/' == route[route.length - 1]) { - route = route.slice(0, -1); - } - - // add the middleware - debug('use %s %s', route || '/', fn.name || 'anonymous'); - this.stack.push({ route: route, handle: fn }); - - return this; -}; - -/** - * Handle server requests, punting them down - * the middleware stack. - * - * @api private - */ - -app.handle = function(req, res, out) { - var stack = this.stack - , fqdn = ~req.url.indexOf('://') - , removed = '' - , slashAdded = false - , index = 0; - - function next(err) { - var layer, path, status, c; - - if (slashAdded) { - req.url = req.url.substr(1); - slashAdded = false; - } - - req.url = removed + req.url; - req.originalUrl = req.originalUrl || req.url; - removed = ''; - - // next callback - layer = stack[index++]; - - // all done - if (!layer || res.headerSent) { - // delegate to parent - if (out) return out(err); - - // unhandled error - if (err) { - // default to 500 - if (res.statusCode < 400) res.statusCode = 500; - debug('default %s', res.statusCode); - - // respect err.status - if (err.status) res.statusCode = err.status; - - // production gets a basic error message - var msg = 'production' == env - ? http.STATUS_CODES[res.statusCode] - : err.stack || err.toString(); - - // log to stderr in a non-test env - if ('test' != env) console.error(err.stack || err.toString()); - if (res.headerSent) return req.socket.destroy(); - res.setHeader('Content-Type', 'text/plain'); - res.setHeader('Content-Length', Buffer.byteLength(msg)); - if ('HEAD' == req.method) return res.end(); - res.end(msg); - } else { - debug('default 404'); - res.statusCode = 404; - res.setHeader('Content-Type', 'text/plain'); - if ('HEAD' == req.method) return res.end(); - res.end('Cannot ' + req.method + ' ' + utils.escape(req.originalUrl)); - } - return; - } - - try { - path = utils.parseUrl(req).pathname; - if (undefined == path) path = '/'; - - // skip this layer if the route doesn't match. - if (0 != path.toLowerCase().indexOf(layer.route.toLowerCase())) return next(err); - - c = path[layer.route.length]; - if (c && '/' != c && '.' != c) return next(err); - - // Call the layer handler - // Trim off the part of the url that matches the route - removed = layer.route; - req.url = req.url.substr(removed.length); - - // Ensure leading slash - if (!fqdn && '/' != req.url[0]) { - req.url = '/' + req.url; - slashAdded = true; - } - - debug('%s', layer.handle.name || 'anonymous'); - var arity = layer.handle.length; - if (err) { - if (arity === 4) { - layer.handle(err, req, res, next); - } else { - next(err); - } - } else if (arity < 4) { - layer.handle(req, res, next); - } else { - next(); - } - } catch (e) { - next(e); - } - } - next(); -}; - -/** - * Listen for connections. - * - * This method takes the same arguments - * as node's `http.Server#listen()`. - * - * HTTP and HTTPS: - * - * If you run your application both as HTTP - * and HTTPS you may wrap them individually, - * since your Connect "server" is really just - * a JavaScript `Function`. - * - * var connect = require('connect') - * , http = require('http') - * , https = require('https'); - * - * var app = connect(); - * - * http.createServer(app).listen(80); - * https.createServer(options, app).listen(443); - * - * @return {http.Server} - * @api public - */ - -app.listen = function(){ - var server = http.createServer(this); - return server.listen.apply(server, arguments); -}; diff --git a/node_modules/express/node_modules/connect/lib/public/directory.html b/node_modules/express/node_modules/connect/lib/public/directory.html deleted file mode 100644 index 2d63704..0000000 --- a/node_modules/express/node_modules/connect/lib/public/directory.html +++ /dev/null @@ -1,81 +0,0 @@ - - - - - listing directory {directory} - - - - - -
    -

    {linked-path}

    - {files} -
    - - \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/lib/public/error.html b/node_modules/express/node_modules/connect/lib/public/error.html deleted file mode 100644 index a6d3faf..0000000 --- a/node_modules/express/node_modules/connect/lib/public/error.html +++ /dev/null @@ -1,14 +0,0 @@ - - - - {error} - - - -
    -

    {title}

    -

    {statusCode} {error}

    -
      {stack}
    -
    - - diff --git a/node_modules/express/node_modules/connect/lib/public/favicon.ico b/node_modules/express/node_modules/connect/lib/public/favicon.ico deleted file mode 100644 index 895fc96a76b68b4924f1c51d022e1b82fa0f461f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1406 zcmZQzU<5(|0R}M0U}azs1F|%L7$l?s#Ec9aKoZP=&}i&OouUjIY8@C}uZw4x5z5N2 zvEG^C^vXtt_xtJ?p3O32c(KTx;lsgZhW%5M85Sf}k-mf`L80)|7ga~M{a znlL=>m1Q`#uoPkbC~GtXMnhnDh5$DU1D6mx+;2QAKt3ZQFH}H~1y~00GcqzVg9QXw z_<%(a7y$V|wJgF=E>MgE#Aid|14}IyCM5MhEnxFF;pTzOK(>#80puP=gnOXwAd33` mpMe2}f66m3eB@_fcnmb^7!b1nO#opK8zcsj1F30)+jEP);68^d)m`eN0o>(5%D`Q(1;j>g@G;xlf`0VBQ`PFY?6)!N&f?*K}$p; zB!U=NBn{eB8${1}&-2_L*HuZp@ZP1@clS@cHp)4iM1ewzw59vko7eMM{e9z|%NNdX z0V;`?KKSzTCvTm5bc{L^CIKLUxc2X{i{ISz$8Sgf{q)1nXTP{`{s?9mQ$4&hPiKC- zY8q7(Y1Xu5iCf33=O4Vy(+|zQ?rW#gkKB0f%}?+6{G*qT22|DQB-73`YzA{N4W^=s zq0kQYcbtFfz zLz)H<&|z(Y4kBG67=JY6c|L1R-#TR>fC$3^Y%QEnYO1xHsf)+GU`3F<{J0kR(;pbF3)zyg$H+idfnl-wl5Wkh!vUH z4Z32YP=l_}1rZd1W_D&^$A($A+&a0e&P?xx0!ctY2}*<#p+qPVN*B(YzvAWXa*%bzq z7Fz41LKILT(GWohi9|LgIzSZBhb*Zf6R6O}WYQ4GOi&71s9lmll0x6;8&ILOl$j(c z0Z1T(6Tg09{?wd{moFHNN6PS?$|e>1MxSJ(0Z7o2)J-Zv|>acY@f`(Y@g7GwsEj5NLQo+q|HsxQ5}XSX_d@*^A9ZT9=A{W~j+$GyI1 zc4oqTHx@1FlRjw4XWyPN5i2~l_F3@aBk!0yu^aoRDvXy}8@HCjUVQUsuSH4$T5|r< zzZOn^?Wfa6y|Q($Hx4{ws+)wX6-HP4zo!S?4KJ@7PG@G3G{CjXs(p*kIrj6rHs7_y z+=<-=Q62s9FuWa^X~WKgJIAAZJR&XBB002ovPDHLkV1jCMPILeO diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_attach.png b/node_modules/express/node_modules/connect/lib/public/icons/page_attach.png deleted file mode 100644 index 89ee2da0753040d1ba0a3487473a715a8fe89322..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 794 zcmV+#1LgdQP)i_t#ewV_0K6;=bl;e_Jt7$~$sQ)q$+ia<4Ec+jeaGt9oWH@O|2`W6&O0t!k{B9sUvLWxkCaPsd9W(`fa z;j-|^ZI^2XnzhgZWYRW-kP&J>DWPo`%;JaBX}or79k=+Jo@h%4Eo72tqev+cB?PjP zO<|ByL#>Tehyq$jR74O$B9WDW1`tK`LzYyL3A9iAcRxLkJ`I)n}v%Od-3H>j$OTBtk>(k-9o?8PqI=0 zB&f-+KOXVnjyKJlf4iHOtnuiE_4+ZVJ$dHjU<^o^YCjQ-wt^!;rPpBv(@pFO{9rdw98 z_s@3+yta93oyfL>7AD5}r=|`zS3Gm$_|(iSl8XBd9k%=91J0j2=ivT5cJ18ZmDjh{ z$-RMd{jQ#X79#Sc diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_code.png b/node_modules/express/node_modules/connect/lib/public/icons/page_code.png deleted file mode 100644 index f7ea90419d950f9e69d977a1f5847456d96a5f0b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 818 zcmV-21I_%2P)@LCln44|RX7Ti z0HI3&7jPq){odH{?_{%nYVq_;n_c4WbUpvU(&Cvnj!vq|kVC-vpF6vp^;;e0mm6HW z+WPzA`AZ|;pPp$&dNjzrc??4rt`k%Q1l*u-BPD0MQ}Fbm8jnsyezNt7+u{23>t7Em zJtETY?ja9KrVs^!LJ$xEMF3-bAZO;-IQJavE60KA7fO$VY_%N)R6s>g5mW>fL4&aR z*EVgKKTBXm!=L?S0?xM zYqL@C$|EDF2q*3zWW7;PDZ}SK*IE8;i!3U62=qn80C&*I1Le7WwNP5EcX;_oh2dJn zf#HgBe4@r$GcjHjmj2vAfT%(YN?}kK=(*+1*DkNNc1H5R++vfBMhACi<5uFUU+N4+ z<&U*CPmWi}REa7C6-t>2im1CWv5Jkefxa6>)dEj-CAW wWa{_}BJ!}~75?MkfaCnj>Dn=~vkLS70Pk`;z)@TQj{pDw07*qoM6N<$f@imYHUIzs diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_copy.png b/node_modules/express/node_modules/connect/lib/public/icons/page_copy.png deleted file mode 100644 index 195dc6d6c365d298e466026b37c1959d96119ea7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 663 zcmV;I0%-k-P)^@R5;6Z z(>-WZK@^7J_sq=QY_e{46@P+~LNG}sRzZsxQHvCsN*h5ir6^j7pq-$xu$N#V1gx}9 zClV7;5)7zih-s3DB)G=7|99>ji@So7-P24n=VQ(@GctDX!^_@$bj%oviY6e4Dh;od zooe%Wvs8LEKQ&&bL&@bwi=STIAI@!-gB2jC5+?y?VR~VkrNxam-`6*8&po|RZ5LpS zNKdJ%c4bTX`XjKsnecf%W>1%6WT?pKNdLLq{=(f(Col?P1+oq@R>)W(n=x!|*BIIh z6DJGw_w`)u6yN|vAhMteYK5#b%r5^v+VCFl1IGssaclZZMS{vs-LJ2$)n7DAr6==K z<29#%AXsBsDoO}SBaXR#_Ap!JKx)(1)3O2pj0_dYWz5By*X74fRT01$Fk%P_RzOMDtV?GU{nsYq#K8iy zb6qzLYDj`_f5$BwC*WE(t0m#xYJ*=jC2|HQYHh=pf#QG7oowi`h!L!{DB$8|qY{~X zu8@sU1tWq;n$XThR0%;45mdqXM892|{CJ@0DS*}>?ami06Q_^tvM~Y3K(_-`#m!8f z8f!QIrH4y#61;0Ym0cCoLl8{IPombPHtnn7%SbTdI&G-d>ZQo!_wBMF9nzX!g8HVY xYTJPGciz9XMh3w2fmZ(7v{)r*QZD48?mrio{~IaoqP z|1Ep}yDQG09bP~E^Dk?@JiKQJ z6-pO(3~IOP)IYisL6D6;oAEd;E%zR}{U$rMRNuD6nQV7nesKS>)yLo7JuDCrD>Abi zbj3uW23?^GA}9jQ{M^8v?ejL?HaT7AX5WPZNkBmfN`w-jL?{tT7ykZt$%Yln?p_m~ z-?>&d(LD(jAd}h=LPltPQbO$*Wbyl@G-_k5jXbb#qffHY03>M1jfEqoPJQ6Mr=Byp=^jfzePZV1 zLjCmNi31hdIJHa%e;5g=1(`u3BRzfeExY%=VCu{loOr{`%2hUR*x>tL^W_TTaj);0 zpPR6CUD1+0>4TQ6zVfH3TQ;%l6#(_%yspK@3gcmG#Q4!WCPyLU93nMKk7E2pcA=l45({2jNho>sdF*A~bA zxX?-cp~y_z_kFf+yqu3m#QiB}03?Z&9vvR5TNgj<)($Vm)xq5G>|o2sFMag&6aNF+ WAT1?sQBYt20000iHtsh1EzPArg^Q zIZrOk#rNsfjaSbMAL;<4h;Z=jvu8dzyz8N&Nb7=z03ZUw?9z%8KQEa6yM5=kUnka& z3?FJk2}L7q>na=T#;<7U*P91xfF`;`6%pVgWgRy0?1ZryL@%z52=-!fGXWGEn4M351L4<+7eDgwo|moqXT+s1&Kmn>-uQQ8mL7XY)w5Zk*(g+<3Y3tmkR!bL zOUKaUtj_pX26sH+=Iorwu}MGd`_%O-_sS}8VpG#fJA)Fcs#ezwtZf?q?Ac70mDv`rVs{$od?VPKeqf<-kUjNtS6ecB*mq<&M97K^6IVsDO zt2$Ru!b+>2S<}_H>$RcInusU_8PMNdf(W{sNlJ3FkrwMJPeBPO#d}Y^a{9TH(#{Y) l0D?dWAV4eUJX#h`!2gmISk&ZKd4B)^002ovPDHLkV1g&sd|Lnj diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_error.png b/node_modules/express/node_modules/connect/lib/public/icons/page_error.png deleted file mode 100644 index f07f449a44ff2761bfc7b752db3d08d0e1238b02..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 793 zcmV+!1LpjRP)lHwhvrAu0-@MQwt}+5~MQTtu}C0%;W( z1<{R?aHBz*g;pk%AyQVBR_Zu5m~;ES_vxI-O!vIF|H*|T{l`n#garr?$RMk>)?Y48 z(ZF2yTneKb};DNWF+jK)IF`6_IfJ{i|F3o%Q+l&4_HGBD|ACE8na_6>L z=s{^>-C(a7J$6=8A_%h5W!1K6dcL!D?XX+Ndk)oei?UundDpX_E&1Y&`)3P8#Ny0s z2Ag7_&ZPhyGj%)g&S6V2LNun1;iBQm#Fwlfv zgyESZR$X}2P;=RW!2zid1r$hBL{K7>2qi*f7>pT1=RdT3@-anEoH{ z={KFOO;Dh#bV*jaN>}M>RZqQd`S=6O9C3KpI~I>l%QFYfo;jqQYe5fcn`)+)zMm6P z4X&L(>gnN0!%J4^rhX->?$S5bY<=GEU%jc!KLL8sww-Eg;h z`H-yBHa)yfojYT}&G*GFc$<(Yja_q=lZvj66DC^O5%$B)|Z(CeD=n`|eM04SP; z>-=-l+xdJjA~vR6^xB#o{ehf~tSM`iwaQv$O<8NIHA}W_WOw*~ XD^gE}t;YAo00000NkvXXu0mjfl6ZQ> diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_excel.png b/node_modules/express/node_modules/connect/lib/public/icons/page_excel.png deleted file mode 100644 index eb6158eb5ca9c4b64c81e70e0fd894dbc8e2bed9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 817 zcmV-11J3-3P)hdKqhFO_H1|Vn1E?(=|cjAh_&P}y{{^`u< zcYD@PK?$)4i~7o9*6F_$F$4lR(d4AAvrez^(88Hk+)+B7E)M3jc=Ewl4$S^`_qwSF zA%qz@=c2EOsz@0qB1IqsqJU)HaG&+}%`-OM8YqW{K85hqj@4&V9vz!Cm_n0-W#f;# zeEpJde%vdgRn5?(+PY=W*z~|lT2-mtown({ll8&3S5+lWz5K}LTRW{k{eJn3Qz!SU zQ`@qI_n2;K?RG~pYJ9=dj-RWgG;P&wEuoAxL~Q)<>x0b=dED^Outj&xQ^rA;u3pw| zca_ClTh_d9cxXg_U!lLRl0`xU@$=UXO|_dRdtXfKwPawmnf(LC7u}-U>8k6}3u|{8 zs9LHr>MIJZGD^r9h|q2yF24Wuh+PM^yMN9GP1$khlDkdyCY}D{kg}jEf-(kW5jBY> z0rB29ZhG-r=i_R{;+1k0?A-sBM;AP6(k1i9ZuixR4?MqXOvphQgCYj~RnXKKL~J9Q zIDNz~XMfO{ZhUg&BTtMyXJqDVqc<x643?SvKHx004lTvR=0a5$dUw z>Xb470000$S;ka1sfH4I-R8njUol7M4dtApOahDylptpYswf1hD#CwFoz6oEA>(SIECU&IQ%a}GXdnC!9$70`0uH1B00000NkvXXu0mjf!8w72 diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_find.png b/node_modules/express/node_modules/connect/lib/public/icons/page_find.png deleted file mode 100644 index 2f193889f7ea091c292acdd684c595dcb206b5c4..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 879 zcmV-#1CacQP)@+1&aazfGU7ezSm^v zpACwO+tu0su66!(dT=`e05DeeCnCFJW(8|RKtKa{4LGONnx2V85A4m%PEQ?MEtR-esdM$pB-`H542D0)N2zSC6Imf)4L8?>%ZrW+H>xCKi$unm zvGZq-*Q%Aahx;C*=l+K%-?>XB)6TB$-L$r*`RUvlA`xP1NG2?)ge8@TQ4EN|Jks0u zcDg;oFC#-#R`YbWB`D?Q`1#y7l$LXhjSLf8AvQuB84}i#j0^!#g{VE#(K7h@5pFHy zSenl=@XBEdxp`h2Ji>CR%=qXJ7!e|?paKet-~;#ok#jETyeB(5&Bkhp;!+;51~G=) zH?L7xmDUu_h+a$+xuWom;AWW!mS$%%+436Rjc@}y?l1134kgD0AOf$OmjOR zstUlshZk$ZC!bAyIg{Y29z#&@3SJ;6D4+_eFume9^#TmMccC5u0J!ZCTnO6m$lnD| z5JeFHf`Xs~1vP>RLKI1GKDY<~pjr2&bi(fX;6Nj-ss@Ds0CcoO0H{JsEQkm{q03skAA);_bv3q{k31qwVo&s-q`Z?_e+j^w(WL? zl+uETs5+~xBU2};OqEE9ETLGwsMGe1%iTRNue)9}|0~E4B*@5#oRXZ9oRXZ9TqRep zPrGZuoOON4n@=uPbyP1y4G=+HktC6l(gZoFD>@_lXDrN?wo+zozGt3P=Qh+3L7+}q z2!WK7geXLnO3Vw;o12Skp%_E#+N#9;DDWP?Q-VS3B$v~Ha)dDzWn0zG~O(^_1!n0HYp-( z+;wPIdoFgQlpYV!10V>5@a)1LyGBMvoa}miyp(bxbMTM-FYNyx;V@TfYddyT00000 LNkvXXu0mjf90!wr diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_go.png b/node_modules/express/node_modules/connect/lib/public/icons/page_go.png deleted file mode 100644 index 80fe1ed0cc75fbb67e9398ae686641f8fb287238..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 779 zcmV+m1N8ifP)JNR2Ufr z!Apo%Wf%b9=l#x^8AolrB&K9H?Pg_|78WA8(M3toqE%3B#7*srcF`i*xhupr27%Nr ziguah1+mN)U5GNmyEfQ3-e%_i&-vb-Co#(>FJ+EhQEwjRYVQ(&UYy{U@%vbY||>@4x=B^vIqabI?L* z;-S&DS^V3-ni4^fl|HMkOEbgX)(390>A}|VIypb5Xee4g;7ck zwKh^A3Mi1Mh@eC$5lV!}Fw%sP623U`g3pd_Hr5sunLvTskx4}&Gm%Q6L}(l4x}jCe z*81q1_4-O*ffi~_nMslo?EQ8t*&Ec(pzEw$vc}pKn_Qp0>D7Jr>ATNC3w@9f|Y+U&+)#!t7l&wKp+nP{PQsb+fb=Yf!Fu&5j8vpRj{FT>jD z>d>$sx;A&+`n$HcF}&sYKSyR;=(=9tvvOj@hUG;~4qTYk^_@E=?$*^_pVh_bGnOt~ z;pEw)j{SK$XVc;qy181rT655gW9NG{(yeablViIL>cDI_ux8m>Pp{tY$J0lgo4#ax za?j0EA3s0S!f>{~ykN9h_RhM&g3K(E`q(dE(Rd49+%xMeR9{qlWnmd{s#(SQ>PmFtSQqUjAtB;_Vvt6}AS_5YgM`Uqu`yva+H8^=4U$e4gHb}u zAQ2N{V3A%pO|?Pv?tb6z=jC}SiRa$G^v3q?*6XcYz$p|cq{uLj@#~Fi`J(>5{@&&N zy%T^+;>8cXx%|o77anP?&W1?1A(>-T49z9pyeCl@7YI+Si zKti7=B~``}TImz(G{0PnlQA3P#MAd}sorMjkP!50B7$nAkU^%#nl{Q9lW0@}9fE-> zN(q7tRuiC_T1r|BBtVBTlQ2+70$Rf;eF`Z;lx46Cpu-rEgb)EBKq(b^W8l<^We(`D z43?0=01z<3G6+UUv6`CsWCk6^93!#+<;ws7007{zS3k2k9-zZKFO~(k`>s0y006+1 zgF_jyIhsL-`FMf~JL~C=cV75(CrJ|q;MVO961G=O zm9d)YpJg5g(4i_HKL75eSE}mq$Y}r}hyVdcV~p>6a}oXr80q`oj%+s700000NkvXX Hu0mjfPs|!l diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_key.png b/node_modules/express/node_modules/connect/lib/public/icons/page_key.png deleted file mode 100644 index d6626cb09eb11a298b90a8a27b0d8eab41f49a82..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 801 zcmV++1K#|JP)$lC4gU2-`f*>nhR-;k6IP7e>YO!0^w)WK%3$w02v-#>5Ep64PCP| zJihT#O|N+nT7XR2h7dAB?UEAOhJF^mol1i`QtQB`HSY}RE7=r! z)zaVIHr5?>v2Gz&fdYw&2ug$!p+txby(aWZ7(4QT)l2`jX7eMQ{>)lG6ev(fWKxmH zOr%mM5$6B%u~qGtCf40#`mbGj3s!n+^%wnJ&#rl>g<4Z)lB5J6f!?|AP275)Zswr* z%T}4~{;_(?waU!#?JabbF3Cy-kf0{R{z}6$e=5yMQKt3BPcl2>zoTPMqMwF;3!_n|>sT?~bK_-2O_m+o>GJ6h zt=+g$4n7y%1qVJI7*5Yw(hqM=JusY{d}*?U(Oj*gT655eZ>Ksn(qrd7v3}DX1}C>` z+X+8@+4-pVq_fxG zlU}~Ye!0+%>J+pPk+0wV{GM$QaYM?5ux)w2z59=S&H2+K?;gH$bZGzL&g5>G ft+noNiyiPkP9r@8gT|RZ00000NkvXXu0mjfuqTIu diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_lightning.png b/node_modules/express/node_modules/connect/lib/public/icons/page_lightning.png deleted file mode 100644 index 7e568703d6432c530224e443771a04fc1e2e59c6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 839 zcmV-N1GxN&P)73{`^;G#xwAtHz%LU)4b zqTRU=ve1RNa3QT=ZM7C`iJ~QFQ*9t7<~pX$d^7W%^M8FFCkcQj-~0ZRTBCB(J0^iD z-~e!d9LR`f3#|=(>$bPvx_D-~2jC%pJ=n_e_OK zeJ_2b-KdDDh@@UlzBSMC;EPygH_MwjWBnPGQegihBV73D?-x9PlHL9A=(Vg=8^d<4 z<9r=UkxuIm)*CO=9e###7PztDxUv}e?$0)rQicmYhV`pQ%S!g@;K(?TVfhM#E?bM| z=B0gfb6h@a8bf5FVT-SV~6}?X}9lK@@Yynoty&1zdZP@?RfODsl=2XzzU% zS8gIN43How+9%bK2S@Xbc`O>`z5`%^;pXGy8^4f>9^3!Sp@|O&)m;dOa3q6d;4P-l zca|=H_{G&m?D_+&-}r{u-J$5T=(X4R&)q|O^gN8cgv;s#@5sEPT5_Z)oFo9Ac>l+I zc4ng5zHpps|9)<_Rw>5bKzE(M1j)dFWI_%OH$BJSz0?T+02W0)_a>#vFqb!*d|5wB zzBUN|M&ty51O@=i?kiDrjQ{{}e|^rU?OS|RdxxP1p5mAw36cX72#`R6UsoeCQFI~! z0ATITp!vfeYyQ?Dr=^5BAshfEa0nB~JG?nUa2Aur006MC*<9`)86SPS(W^`H2n+xi ztOWohsFfVfVWrI7PSKW}BmkyPoj(-|J?ES|BGd-}fIxr{00@ANnO*ZR`#)pee4I5T Rmm>fG002ovPDHLkV1i|Hc$okI diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_link.png b/node_modules/express/node_modules/connect/lib/public/icons/page_link.png deleted file mode 100644 index 312eab0914ab59271384686255d1be913a6b3add..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 830 zcmV-E1Ht@>P)VWgGzD=Y79#JI$lhEn`|2MpRa?Bt#-nSD~P0P(mbVe{KrOBoKnSsk>m|ML{6l zBosu@om4j#WzNLRAk+{k1JRvL(MfE&vvbb->v>W{*z*1_uMP}0cIRX*?mz+wk%*#O z%0D-+$B*g1nRkvI+_3E8Pr1NC6@5M&4vWaLCnNlr;lNlr4i91z&)eBGqL{L{GNu;Fof}GS9{gM5BJuH;2QWk8yuOZdB3pGR#s8bd~ zAmt<>3Q=YH$t5YJ5;7@+8Uh6=ktBgY6#6Pa%2F?h910?U8cLT43KAj$Z1*==ra&gILO{WkHfs(--F=bly9l~${z@AT>V$oat!YAD@M zBE0v_F{`g#^wOSP-u~!wvlmXdd*uqFqoZ0^{&nEMDU+=!>({S0wrQhFmoB}Yq1)~0 z^{A)L8Xjzdr(W4_exYO6u3a{4*kIeXZMJUR>Q=ksjW_p!rAwDKYUs90>6Q|C>56o@ zbrSq^Xk7Gq#>dAsn@un`Hz$?w$;Ss`%jV7L%9ShgHFV~C)6M>B`Tp%|nqPc&G*A3| z)Qe+}vT4-x^t1~XE@(6wR;^lPWMo9n*~E!cy~)YRsT2{`?fqeIw-e7N@mOA%UcCIq z_kH(_EK7|>pM*1Wt2^DaDAp|cvp*@(ZZDKpYkKC^?97(`0sb&XTXy7N#sB~S07*qo IM6N<$f;YmCWB>pF diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_paintbrush.png b/node_modules/express/node_modules/connect/lib/public/icons/page_paintbrush.png deleted file mode 100644 index 246a2f0b426faa0c7f5ba009e32b1deaf88d1288..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 813 zcmV+|1JeA7P)otxGRZMDZ!_a~nK|b_-`n%VosaL{KDuPV10`(1LIen8kX2Xff$3BE zah#djvFGJ&eE^89Pk*-O^+&d>FC~^GjRYVQ(uuPJyS|-v?9lxA-+tM5>1Qu*n+Ir1 z6KhA>X4$XDH6?-|E5oe1E?pQ5-M;2xw_ex!x}I2+b=}mPFW$U%^;o(Zg*LP!K^1kP%8ynsD^= z1y^6xD1#GLjO{VLdh@0GKY7;d$+NGukV)GRLPn^=q=dF%B#XaJrNP`0E6=}e&Gj3d zKJbQre*WXt!60_DnIzgMQc6S#fvjXxsE1v7;T;njHkdy2miIqAS(nX~o%cO+q+b#h z5tIleLWvL=dQE8OC#{%y*Tnku&K`Tuub&_ELI0t_ea{@3f>Jv&sYqld(%}3_GY3Dm z;O{3*Y?v^A`a|D;^qrM=ykI)U6QHd%WhO~VF!SGjGn0GOZrc3mGZudNl9{Q#X5&-F zuGwVReFLBjE5jr!!^-5*L%!I%PkYH#Hs5rMrEBl^)9)9XTD;xjHFxVZMc3~Dw6#k$ z(-S}RE$bgMHv6Z`mS5|u$$78sp4G-8b@lVkl`HtEv+MGn!F&bKcHPi$$oP_;=BrPf z$(~b3&p3CsuQxhoV$%jIR;`lB-s7FDX)xCTXuJ7ZyIQk96uIR=HBt%-P?N*bp`)EF zq14c}QM+O70NTOa@V~_)&GMZ$^cQDlkyOCa(H3Mf+6xhCuZh`VSN{cQBl5Ys9{cp( rh`2H3A^=GuC6HjQ*7|*0>;m{7QlnX3z3MSD00000NkvXXu0mjfR5FYo diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_paste.png b/node_modules/express/node_modules/connect/lib/public/icons/page_paste.png deleted file mode 100644 index 968f073fdddc1cc0f0800b1ac4001cd9a55f053d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 703 zcmV;w0zmzVP)AVs!l4K}n~L(tL`6d4Up4iSWnZ3Qg~4n+_J zDGk-qQdogO5JUtO-d5pRp7Nd7_r1^a|M&Zq%mn9Oe((|e0sw%Ur!K7T1pojj=U#f? zQM`qbQrM^DPkwa?DK_be^~z<~RgSMIa<`xP_4P7gg2jCwJ{9^k!fsU=#Ti|%I3p;>90Qd+7|~0h&mIklA#nb>ATL2+v$&u)OBgB z;nsHb)I&QRKeX40H~~cIZxCd}5C} z=79lXoXK%6YlyLtsV$~bSm?Upq|DJh#{|*a7XMm`4QJWZ>s6nL2R1|&J z0VPEwJ9?!n`o5PKAjc->P1Gi8BY*%!5&FVp=#)$mMJYul1Jton}gujiUf??eOy!x&!tsjxy;=Q3_DdcXx=a^OBhW0N~`A@4xB0a*%F? l+@c^sQA%W+?pa#c`9H5UNfS6T{e=Jk002ovPDHLkV1grvM=byV diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_red.png b/node_modules/express/node_modules/connect/lib/public/icons/page_red.png deleted file mode 100644 index 0b18247da5850f3c2486373a3e179acd2772e8aa..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 641 zcmV-{0)G98P);68^@7JE5sw#jpE*579S@TLkU(6yap1yN*Zuy>-hV%Q_v4Ar&!63c8OBr(ZRhFu z_kWs36-AmgZCT>x!RqM;Zu9tqvoHI~k@UmYo_g(*J3c%2{N8}7I+|qKPQzv}7t>%W zsu&9G)UmCzkDYSw{fBnuW4j;1fKV_nicw`$8C6D=F_qu`zUiK$2Oc?5UVY+D(`I@R zW`KlwqLftWHH3Z2_XVNfKn>VgT~k=@- z+N>c>0|@A_HbI9Jn`v0~7cfIF(TS69zaomDS1QtgvaBBfGEPLHccO2~3jc>n^6}^HAEh-2#VxC7YYcDXv!L9X= z-R*SOUvIs;n`8(LxP4~^2|JsiN^hq6cU}5dn4v1~Kl{eT7pm&f$PoY`hyVe%y8G*S bxB&kH@RuR86sIAS00000NkvXXu0mjfZBi!% diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_refresh.png b/node_modules/express/node_modules/connect/lib/public/icons/page_refresh.png deleted file mode 100644 index cf347c7d4685128a4a447abb9fb8e939417644f4..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 858 zcmV-g1Eu_lP)`6pHR2Ufr z!EI=jWf;Km|8+n2IrHqe<9xxFVk)&(Nh?w$Xk`TAyvb=#e=0aySC z00NkRDM597_LiNIJ2M^qhuTvB004REvvU8@of{r?P8tmo3;+Pk0F0@*jAMhdOkS&1 zhJPPfQa;pP0|4+Yk%#j>X}o-s#EF1_DMV93FsfPP`G*>Ks>L&)Q}w2g%slu0kBfW1 z+*$*0BC^oTl6>OGIq(9BgG4|C90Dk-N_mPazGrQ7uHZ|>BLD!-KmZ)z1e^#?1Sf(M z!6m}K(^b|i%$TcA5bC}r$tAA?0C)g1@CgWliJ;NAk&ZF+-w#}$`-3nZ32C6IVKrHp zr+(!L2hRfF&AsTw>_@ z1y23;E%Oz}?q^Q2d($ayO;-sON2t7$w(Z|o0Pw1YnSp^}0PI+I5HnDNsFCA?oorkG z5sUIGIq=FSyxcj+xlhkm0en=52Bx3@02o12gdAU$_i?v6iyFMuc7P9#zQ-Hf; zVuV$t9P5`m)F2w1?t6{<8%wk{w-PP#Sj#%1MbsjrSI6n;D_@8q9`~W98dNQf$j=iI z6~hpgww&be%X_HI50Hhx@W==u4TLPB;ei-J-1}G8wH}|{i#Lk-WZAyfv}k4y0|fvU zZTy^$u6L>2nWo(NDSV2@MRD}JQ4(c%G%=dG@_vxH?>gcH#*Ue2HC}9sapf8X?R$Z;XEnm&g zW99mh)5jNw008mK8)r^`_{yH0rNn%u1|SpC(tjf#om=+r#lh+?Kb>DVb9`|C0Bvbv zN3U(>f4-tAC1hosRoA7p(b(hL*V}(j>ug<`&U)|l$6o$)!>PBQ9RQSwn9asj2p*|xhU*R^vq?*Twb0t!lm5}`yW5lRy-U0ZYK?8to!;o!r!XeOE$ z0HB3T+6EEoI4PlR=wonwqJ+TvCoWh&$?CAPVYcU= zD{DS0?AkOtb@-hh^ZLq~FMjxYf19X?pa_YqtgZGvv2TaxcF#KT?O%=_*a-kW_;N|D zakkWsOe!)HsT5WRBiC+p;N-c>0Qwy(1D2MDBC595oXSiR07)sKNk-%9*rDBOO^HUD zZW#;)R&EZpqha<(HK$(tZYU#V29<@0qCXgU{gXeGpc_|pTqQD-WO|}%yKZbeX7k*H z2W~CK$v8NBAq~czrc5A(v51g0Wma7`G8}f=ZcuAiYYxZan@gP(;Ku66M6?bquGiHe z3Q0ya)%Lvk@kLixZfZyU@#UFbv+>pYhcj8TRKSr_sWG8i^X~UA**LvbD3(_Lba3xm ziYcpup*A9qJ$?AA=Og05lndxfwr`!C+O~h|B~4 z01q8H`StcY);%&mId7_+)76ovRpeNWRp&4M?#jx@|E-)x%P*A6t^fc407*qoM6N<$ Ef@ddc(f|Me diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white.png deleted file mode 100644 index 8b8b1ca0000bc8fa8d0379926736029f8fabe364..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 294 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!60wlNoGJgf6SkfJR9T^zbpD<_bdI{u9mbgZg z1m~xflqVLYGB~E>C#5QQ<|d}62BjvZR2H60wE-&H;pyTSqH(@-Vl>|&1p(LP>kg~E zYiz5X^`c$+%8#zC{u)yfe-5 zmgid={Z3k(ERKCKrE7DF;=x4^O+ pzO8rLO8p|Ip=x)jHOtWj`bJBmKdh_V<`47(gQu&X%Q~loCIFbEay|e6 diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_acrobat.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_acrobat.png deleted file mode 100644 index 8f8095e46fa4965700afe1f9d065d8a37b101676..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 591 zcmV-V0~O9lw>B8WRlD)Gm}Jrz31u-X&&gn2lvjs=i{7nIaL6v2==uw+8Lcs(8j27 z;|c`rmSv@Lx!heopGP^^Ieb3f=R!%Lpp$}iMS-&P3EJ)s48wrJ_Ni0~k|c47D2nj= z{jS6bt|kFpFf|p5cM`_&0Zh|`rfEp0(}=}lT#(6RpzAsUfxv^LSYX>WlAaN$>)*J5 z0#sE+JRUD8iT9*fz{)_^7@6P&!sEjTcD+I9Z4YjT1`wH@fV{cEvneYGFU%maIEU2s55&K(LixD|{p-uiS@?KNj zk-Go8G$hH6g002ovPDHLkV1hVj1#|!a diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_actionscript.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_actionscript.png deleted file mode 100644 index 159b24075191fc259cfd80c797a1b0d74c168422..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 664 zcmV;J0%!e+P)7Z7t2}reCh0o`+ zAlt$F2tW%oO@m<=(B8a-_VgLl#~yUMUDWG!0qFPppd^03e+x1WpkO1NhIaKD2A)-@ z=Py8(Wi%R%JtYZG#sTKH@6Z+&!S3Edf8jFJJNKuva#KJQD3X^7;H^fd2di znEN&c58aUG>`>P{Vqq$kLb+TP{?I!d4(|o59X_%|nVEZq2Rk60n7072SWJ{64CV?3 zgS!EB=eYxwQ>P2&$}(iT6UMvuFgHHIEdNA29!EBtg=v~X!DxxEH~}L2zn|52%xalaq@DTdhh{EVwv0IaQ=!?daer zTKp4I`l8SDt;d{8Q`5Ko;BXUi&oAG1l4}59P-{|^S(Rmord5s6qsh<&m@Ab^wqCD) zHyRD}lKLDzpYN&@q5&*47mGzGiqcXpmqR9#K|CH8kXS4RNs`(iEF%HjP%f8ItyaZK z6$%Apvsok(2>~dTO5jTZfq;N?0ch4l01f$k9?4{~Youl-#x{UDMr#AFIkz@SDwPtQ z$gQ^$2|*(Ps9LQiav_8o8Ne<=Zx1*M*syo80sEO1tB%>5 zfdHB`1z+!R@?ghPRKmL)hWEvZE$=*54ose*0JiUNTM_)cMDXhxEKg(?-pD=y<)L4J zT0dSyD0&NhJ$^_8Ko9uom%-ZM4BTM{Tw$9qyPj=-9W;N(Wi@3*-Q4pq`Gcp}^vvNr zyd&PsmG>fpCSZz?K}UIEd;HGgG%0MG>ymxKPwy{>wy(m*Atq7)0000~7 zMNw2LQirBVQoa8G3P(rY+l;L4iy+JwSqmy$9JlSkk z&*$^Eg+c)@!R|v4gdc8+TTn&eWHO0VD&>$!B%o;;WLf4CNs=Inq9d`xA4otCWHK38 zmc{pkX`0Y=9g3oGK{}lVy~OYL|C5lQ&U^l;wrg|7w=BcA9L4-r411?K7f`@348&rw zXD#uW)DK;H`hxO}u%=@Cj{;#u#_;bb1_KgUOT2Hp6;)MvC6P$vQP3=g1O5#aU%I!K zZ1dc@f}YvG&*Spnplm2rIp^VdA^HydZ0X1axdms2!RKi5x-SFA4p@ zC@N|PI$ryHL@t-(!zBsf2-+sYAukhDHU7Lxm88-p zDk^c;sHj}OKUc4lGZU}6umlGVNAJx0%sKDOFwQx|V2pVvxhYKe|L9TNk!~md3BVrm zYPDL8Hk*yU-ER3~LGwJ7N`0ZV&nOhBI{~~A;@ND*=kxg?#^W&`4u`zk?Mg_e)8XlK z`T#M+OaR1!<#Nf_>$S`xrqd}OjYhoJ>q)?3vEX8pY&I()ERjfjrXM$k7e+-Qs3Ihj zNyOuEQ2EGYG7ro!o6VOBQEwuV2z)*tR8>WxP{616FY)p1Pn1d}#9}cxolZC$4n(6- z35hJq0;FlHC{ zp*iF(lgUK(E`($(s9pJ8Kn?(M734H_63WHtf}6SQQ_MXEP!#0|&@>J8dL5TfG&tBw z#tYn{TCGZvAr>cca%YYn^!t73tg8OOJ2FvJ(`YpCyVZi*?Ur+1uUA$hAg8-aK)c;e zQ<)!XwHh|n&ND=$@^)>aF-`~n}#*WMkD*M|f8r$i*z7+W! qF|A!t*4fE(R`<_YIkN&?Jng?3oQ|aAqClPi0000t>5xmo{nArfL4CJwMMm+N`pQ3p^Le$?rMud6Rbxz!-yG7bz2z$^USP5(;udf(gfQ zG~f68y^)EvcNWp#bUoDt2=h+^%o-?-|mo~iieWqLNP<0m@2PTB7ftyb= z@H`K$>v9Pr5X`L|rw&CEN2(9SB7A2SE;d|j9@*F}sd(@*2l|P*fWfK>1drZUrtUA7 zNXO~pKn1cjf~~TLbje1g>EPPzN2GH#UIBxJ{}S9=E`{zs-w#hO?vcH+hJxroI5v?j zD!4lP0WXq8zUx3RAP@|Gq$}6wXCjFLZY^YSWBxN9#&g)ro$%5}aYn#y=tJ_aIT%4d z5d4u`rlp!};XGmbZkJE*kYJoi&N0pd*yxY{0${xD;;Q1h^8f$<07*qoM6N<$f(}t7 Ae*gdg diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_code.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_code.png deleted file mode 100644 index 0c76bd1297751b66230f74719504b2adb02b1615..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 603 zcmV-h0;K(kP)^~*-1fljz_B$LUvK}k?BNXe#Y!m=zM!!V#}8bncK5m;8VP zw86G*RI63?Cd%b9bX|ueNlZ|wR6rj|r_)VIP@r2imh3?SN+^{|kY%~8B{maJ@F*OK z&VH9LwOeGt#DRjj0~v~8`>iO7!Ybi;zE$va`A^T#yW`y44;k^#O~K5*jD=qcUhPSc zvyy~q;5H_1WT1l~cqje9yfa+l!hu6xjdOJ8s;8E^+=QQ$tw p?%p!Hy#YapB=@+^9(46X{{RQg%9y;OKjr`c002ovPDHLkV1g7l326WT diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_code_red.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_code_red.png deleted file mode 100644 index 87a69145075afd8f8fd8b391c5da1249ec8b2889..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 587 zcmV-R0<`^!P)LWh{^|hy<@Q*xw+qo|KpY<+vaXbbW{L4q( zTsjXEJvb}e%bgb=o%W0h?4u1;^bWTqH8}5Th002ovPDHLkV1nrS0P+9; diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_coldfusion.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_coldfusion.png deleted file mode 100644 index c66011fb0fbdcbf210483d676b7131542a0e282b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 592 zcmV-W0k7R5;6x zlV4BMP#DI+Z{WQcKZBTk0lfkj5F$ztWhP#lcuyb@0@rA^#Kpu5KLA&Rgc}o#aSmis zrZC__xY^&#cI&!!{c|4Q_tcec*#b>|Y15wPcY2=o3;-Bl=(t4;6Ok*pL)-{*A;GX^ zS(@WGp6j~k1wBVR9)BB_gar`}HyRBXh7nM!)u5^>N~MyN6bc0-5{W?44iB<`2biXb zR;wk?jIQg@G!5l)SqhrXCU}x$GU-dY1sra}0uCq@153FUULT=jNwSk}0WBjKz}Jdu z<5gB*<^XtpAmp3m^ZEXQZWd1krhft}CoYaF4cSMvTJ01}X3X37KYdx-D0$c{doUe8 ztY{vlGr-e*;N!WAV%_hgUawyYrhegW>^F)pv%uUTFslHn; zvJ)l{%w(~{!O4`KTmK{Q{zCYltLfs&4?nz|6IdlqHCvX;|HGv~!QW?8P~_d#e0$v$ z)5XHEz{3>qMiH`1+qNYf?huS+@L`J9_$cjJF)Hf?@pu;)`9}BXwGUM{2!{y-4|Z{L zG>z?O%Cp8P5T#j1DID7u_*(Jg?7iss8AZQ+&;u_J{FmILf((9eoiL5nGUe>Fgq*U$z0000 zJ3A|*qoWOonz+4ZQ0KNhDB07SX1?#FrNy8%K)_l}y&kh`*KYdy`Y99&tgNgMLSSrc z?B?+B@HO@P-jS~z2Rgc6yy~Y~%>oJpBxsb$5<&nRLqiuR7K=@0SZj~jTs|sv_jWVX zGe?WflejOaq|Vec=s9+ahmXbyJ|T)Sl*?s82sr2H?Ce~HD5WI+Sz&tmWrN()wI2}+ zKqg92t*l^-#ae~;9%KFlWkmwnY=-UK`_|%ICZ#P1gdjK<2n38VXsuC7{WiU!fZFmm zW~Sda9(Qi@pxO}$ARY+;t##Ao27usOqNt7Hwq6K7G1il@xitj=LIM&{N&#SuX;x4x zmG6FhCg-$PI;hQ=;1iZ>F>^~@)IPi;l}fX?SZ!QiO=X<|pSVkNpJuLHzW(FT_~W-v z?vFpkyE>8ee4d=7wKauH5~dd_M7d2Aa=ICC{Nj7Blqv&DQEP#j_VeWV&WXL>c=LLK zsmYg^_JiDb;%U!UxO%qjFAvsDFj-kzT2$GbV(ZopPM$i$z`!7jvEk07BcC=6FMt4` z*0u3Sy`0b~%#(0000K diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_copy.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_copy.png deleted file mode 100644 index a9f31a278e17993d8d4e13beac2f9d5f7b42d08f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 309 zcmV-50m}Y~P)sF~CC`eaI+m%Y8jfzomMvZQaNUIT3LIrJ$h)_W{ zwF|LDNlB-g`Hb_G$;>3F$9JF3WYR|3fy2C+_wH}*xp!_4fF2UN4lt#d26oXwru}hT z0+0%Vz-l&|Tdh_L-Ng1G2*RBtBncRx;99K)&+}s0whhxXp{go}$g&Jk6k|vfypI5M z!1sNGVaV?!*L7i87Bo%cfO@?S`bajL{R<($@$|PtgBRcCGIJ_2a|&kO>G-s2aR3E4 zjssoScUa;zIdOeGHBnH13G)W-zt$kUQgNfG;96b=v&4NzRt&@7nN%v3HsG`<<+F$cumMs448N!W3r&2Z*b~D5^$^d6Jxn@SFK5Q8*uKSR7x{I|H-_N1f+AD zSYC5@2K4OKL$==F9U@CH;ONNL(W}oZICHn;d?~pw?GRIsH*x-68Oy6SuK`)`{E)46 z9^3(-HXa#X89SBv?u_YP)WjsQrp;}0X?Bxrvf12IKW8>3t`e~W9|JS<{btTNbNT@EQIWBSNJTX8AMGXD z-SsH|s#>j9Xf~VMtyT-YMD}5^SWHTY5->o`k|d#AE_YQd79j`%GMS7FNvG3b7^Vy9 zn0HYCJy5MyQLoqKnW|JOp-?D*<2V^msZ>BOv0ANd2n7t@{=V;sZrQ>3c})5_%ms4z z7!qXwHHe~!QFj8aR~&*-3F?O|;#(ESIXP~Os%|~y^7c15*q5`gz2-5ol!fU92NIGT z_ves+>+Tf3gfcL?!nimYmR}cw*|BGULzI^7!;k#3K^YO#;!+vM@N~(99+<;fdqr zYPJm+pXYFYk;neQyXXEcTQDNQx57i`Okp9A#n?<7!{#tnKJdsF>utb@JH7dU01gfL zEK2hoPZAnO5+je3&^i*hWM`qCW^vLK!O*?U-#IvXV?#6koWqrwnD{j&K`7N>^tR3G z8zr1(qVOzcF#nF1&0MZ5C$l8*E^Uth0000zE0Ay_3@1Z_7#f-XWL#E{8Al7>L$ z0Rx7lnddoqAyfT%&#`$;v0@*5YdW3w z7mLNoa=FAshK% zDiy@zakyMAxr-H?iQDZi^!t5;Eno2A=?>mMx`Vg(Z!?<53LHLvfTPa`$mjDcX*Qdv zR;ylN4OH+m)fVX&Z#yZpUae;ss@a$K&})gHovkhr@w#xyPVlfVgXti1_357y%I-UHDvRWYvPEX+#g+j4Q9ayba zh7uQN1j%HQgA=Fp9DfODAU^*3*FCs^6IpO7xg`RUXyP)(;=d!ly=#I^l3e0Cub`{H Z`5PU3+D2e&<<>s`J(VpX#y^kqzQ;#=2x({YMw9Q&ndHT&`BD$#%Ql?{+)-OuSA`r}MWJ zVg+2Gc(GW}a=BERPNy^;kEz$|38dTYlFQ{%5S!g@|8f8D_!Nu9_Ni2glF1}xG8xi! zorc39&F6EPOeWOt_XS`W2H_Bo$MXugy}SEctJQj=(TLXTHL(jRXfzs>NF=0SHk;94 zF!&HjdZNX(3U3;LY64IMX__Xv%_wjLC!J2`0Jw?X=zPK$C$`&dYPDKaC={e16bcE@ zgun^<0k;ak*=xLE)@(Lqu~MmsFoMCLY&0Qog`NO(h@kyxaA%EbwJLy8sU*Vi`~52K zX0wrqW;_LmMq@evX4iAM9Od(Q0eHP$1%L|xAh@vrqB`HPQLon}f3aAka=9!3hr=O- z5F9`#J_7Jhah=U(4RjaRhkS4Xkk98kDz-`i!r|~~AQ1TFcDw(@<8g{aBE)l)PNxNE zI(RPyc>9e{@WGSMU%i7*v{!&P$WLz25)0oc=Dl-yy%xYZAm4b-rttL7UjR#%`#j_F R;_m(?iiXTHIMmcoLoO94I8;j@ zv^2DJ5#orqydFJX|Gm$_Bi_vyew+j6{r}$Qc@D1%fQqeAhJj)1!z4pP83k2MV2~s! zSt^w(<#HLFVBg_#xz1W8ioi(WY&Hu~6zil?DI^jJgu`K35(hkP)H%@Imesbg#5!Ps_$Ni*SiR8&sKb9?M`0-mH)gtg&YgRX#*TXz@Z+| z;|2H@xzE0TfuORhuO2k6#K8#sW^J`mQ0+E@$K`QkFV+DTlI$w{GJ;zid{*v9xeIe_ z$|Bp`@iKkgoFK3{4Z)#DWKV~W4K@5WZN+Ql_7%YxNqSx7%cWud&cX>)_PvD*UzxZg a%Kia9Rjz_59@~-t0000)l$0ECbfb-0$}>7z|u>IvuoHEmW&j4lzv=KA+EpIObc7e7{sGA)QVmnM@*^ z%|h38^m;wC+ilpk%>l#V5LCqP_y2_Cayd^XlX;j*r54R*lW!zbqtSpQNyz8(JVmWm zyV4S$2{Uhyc{Cb0QQbf{ZGT{Kr zvJi{K_&f+q^Pv4MK$hhS4TgFj_FD*rLOePdE-E^T7ZzTFCRUB`*?9&h(a#C!-v8lWG#k3AOJQaUey6Oasked^kDPe=Khg@7s584 zg`XfS1)&u*_c;I76#%`kkBfiZgKKo@0)9d6vZw=ExQUtV?eW{Y1Xv}=4X(2zy85d> Y0C^(qLv?Ui{{R3007*qoM6N<$f-gW7od5s; diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_edit.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_edit.png deleted file mode 100644 index b93e77600def75c9a144d3d0a5088a62c02cbb0b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 618 zcmV-w0+s!VP)$>5Y&axjp2O=VLu>*f>1L;s0)kkvKC!*u?s6CVL=HJ6oP~pNfZc; zsKr=bq;7MITw8NXw{SZm%59TId2x_9BQ zV86`NuvGI!>o^V!Na!=$7GJE{Cq`b+XwknM{UcGHFTTfmuS+ zm-zYC!P3+zmY;SG$?!fYkOih`QYaLxyF}A86h$GGN}kFj)_o*0e zjPMP%zTG7FYMAfO2Nn1D`D0Cj?Wl>5q%@CE10nX)KxpNmwk+!IWkzywiYD( zqUXiYYIq3qcRyMGJ;IY`(Gz~E$J$zu2+R{)xGlE*88b3WK6V*J>}2iPY1HH|tER0W z_+^^FdppY?o)Gt5M2`%xwRDH@R3G}^i1l4|6uchm0X0f!@&YdVLB5K&dd7Rv{)DXX zt^&vP;}kqj3f>94j+4xd93>s|Q!Ezi>?r8(Il$P}PFxSqu{d*!Y%*#cX(R0f|Juz# z3o0_xI14Al->1uky@W-rCI_%l&>PK^TXNSN{byMk2AI5vbwp!K-%-@!-vPR3iikL1L7HA!^!~ChCFU#lnGzp88=I z67V8PHBo4(l$u?-AKmT8?#_0rKW9dUNRbpLc`}piywAM9$xZ-3fR1C75T(BjCn-l* zjUcci2oXXo-}iqun@#)+`W@kL_-U&|2>MxZy~3IdmRm&8b)9!2%ksg3R)nNnT*TJOC=6{2hG86Dz+<^p6qfG5$i^UNUh+u)CD7O2 zK>Ioazn;U|+X0x$=feveYZL1W*Fm%e5P1sajd#eW#^5(ddx76*pt$^)b}$Q4oPabL zLc^HF>Z{8za;f$LtN0P$6C?1{X*jtXkRJ8IEeyiSzencvH3Ux_y>y^}wfJrRCQN#9 z?&e+C>sSAfrE%mZD5RfZ`gSndD)=P?+nG5Oq$zmY&-v+gc7R6c0u8^Ke#|XOq?gF@othF3zFpM8Il<8BJrWqBtF>b#_ye4{0)Xbu6j&@UIhRE002ov JPDHLkV1nWI9dZBw diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_excel.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_excel.png deleted file mode 100644 index b977d7e52e2446ea01201c5c7209ac3a05f12c9f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 663 zcmV;I0%-k-P)^@R5;6x zlTS!gQ5431_q{u#M2 zg&W%y6a}>qj1Z|7Vu&-DW6d~k-n;jnHsjb-q#u0C^W!_5^C=MlKq<8oNCQ6qS00!X z5eI;XP=g!^f}j{hku}E1zZ?XCjE;`p19k(Rh%^AQQ54xysU+ocx$c#f61Z4HnT#3u~FR(3>BnZniMIF4DouI8Hi4u>cAK%EN)5PO(ip3(% zIgBx+QYirR){Z8QwV$9Z(Mpt=L-Or3#bf-G@66}txq0yc*T(zNTBDT0T8rO^JeNbSI-Tzf5!pBioy4NwAN^?iN#{;fH1Jke4Xa`^fR8m z%h6dq%xX)S?7`zae))(Xst^Scp6B8FejQW?RLTM8@0=vnnntuRGBM2dpo>gbCnTD= z^<;=JuqdSf@O>Z8^XdR?s+KEfhDdB_#ahFj^giCtzT(s8kA$AViyTqaAR;KGaLzUU z<=GqA4bRwpX|IG~*x>pZ!@zLr`XQ`od>m(`;jz|M_*1GDO#$7;n74ppb8=eiqh760 x0yt}J1#p`gw$`o!R{d7zU9~!Un@nJV{4bstt4Au+Up@c;002ovPDHLkV1kWhGjjj{ diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_find.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_find.png deleted file mode 100644 index 581843637079359a6a58fcdccf0763690c67b063..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 676 zcmV;V0$crwP)_k3`4d{s8lK_6bi^@vq&To98fNoK}7)fx$e2^Y&@<^jR_Ee+8}KG;X`@ z@bCyiolqX>bb1ZIs%QGnjzFU~L8H~d?e;*XP(h(S262}XyZ3a0h07r{KV?E70l+e- zE`%3x|M5#q+;HOC(h@A^M)7Rn13dm0&>K$j%k_F4wOWlsNCIH+!c_#{eS&TL8v4yc zcpnPEY`cQzZ$ILq{U-MA6Z6Z|1p!FZjQ}tXSb25J@HphEqX-6Hqo?-_Zn@{d#>2Ml zJGhxTAd&emK$lV-QK&VM&ix0Xy{GyS3Wp(+E1^8BhD3T0a)m-Lw@Lu4zQRrP)9(3F z^>$hh@N>OAXrmPYunLi|fJ$_*5i`46;M>~*5D{bp>-OL3{+!MJa`3kv~Q#QfQ%c z)1s}QE<_XaYBG;IuRF=td#+}fi4h(6HgoUyJLi0t(*dA^B)%@8kkG&bdM5P5^Z5WF z%d%>m^SbN0XeV)wbUOXn5Ag#A$gJx+7-OCkMM1S%MWIlTkbFLmOeW(&n&wUd&;`>p zVcRy$Z{K0=?SpNnP^;BYEEXleFbq(UY&LrXX$6qkJ~)8+b{=jj3HEXds;Z(?D%}}L zX3`39&dy=Zyar!ehA}e>w)(*vrCct{PI9^2Jpj&OZS8<3-@{0(gNv%1{)zAiLY+_^ zl}e>Ofd4&#Irj#7>=o=Uhv5IJ@?sN0^J|(WL2Uun$4}si6}TG-s3T#p&6GE<<2W)O zf{^Y2HlO#*QDvTp3v&d@;8*}aUC4lisG9(w7@d5Y8y)}U#FwCkqp*Mcgme4{&gGRf zlBfd`nF9cQBKB2_L{F8G2)7pAf$i)Ds`|}-c>pc^LRW{w4SQ)3N^BbZx)6BlCZts! zKph%`(m#xg-q3I7=(us;9<)*2%iuQ1J`oV3gU6V~T}^JU5714JN33&GwEEru0d}Uo U{MPL+lmGw#07*qoM6N<$f^vibe*gdg diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_freehand.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_freehand.png deleted file mode 100644 index 8d719df5205f7415ce657e5c277db4533c82f346..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 639 zcmV-_0)YLAP)p{{sC7)XB-g4w*W1a1)XtvxrMYa1o?wn&v~3 zHnC|#(>B_M1d`_7gfzLiHy=0c<2kQQdXu*33(xYN_xYW39(cz9jEVT%VokB8|DoF~ z8u%Q5sdl@4VB7X#uh+v_;yOGY&pRi?378ghv)P1cngiAAb<}D#l*?rWDV0j_dc6Zk z-|P~AJZQCA=yWcQjG8fYnimzj*3KqTfN0Cy!G^$7)+bQ$+mHVd1J zvwOR^5Lm<|R+uyB1Nu4vL?d4qa3tn?9H7SZH@~u=fFHEDfSH|bHU6kh0O3%cLdyny z{`9S2Sw~WMy0MPy!64i`jdk4Z3>^+KIL_fN2V_d&ywBt`^IJpxUI$=YAph~5`;xCe Z{s%Y0vkUXDnO6V+002ovPDHLkV1loX8z=w( diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_gear.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_gear.png deleted file mode 100644 index 106f5aa3611a4807ec8c21701c631730275089a4..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 402 zcmV;D0d4+?P)<@FR}JvtGRKa0_WfK^c7uXaFH3q@Y!Hnl8VySc`OtkPN3;#l*y*l23+99h*9JzA00}rAC!#M1dZ#v9YOBH|eC*${MmzzYjBu!!-< zK8tujf&(6i)1biy*F>4{f*Kd(IU-JsG&#b_@NgTnx@40)2@2%c;*=?-2Za=}O}7&( w%_K#(S>e1j&gfY?mR})n>>0+8p`iTe2d1K2h8#$+)&Kwi07*qoM6N<$f(2cptN;K2 diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_get.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_get.png deleted file mode 100644 index e4a1ecba1b60e54f3777717ed105cdde745b7184..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 516 zcmV+f0{i`mP)o)wchR-92qq~y6`XqbKmElbB3z{pkZs0VPF`CFvS?7jDn^mFo>d9Y&06* z&1MsS!M-CH3ee+h_sy)Ms%B*ec3R0RpVi9?*mU84yoq(Bw8 z<4(999dJJE!V%pWT~HGRIAb;(#O%2K3?uRpz}AfgE8e9q&OSdr^e^}lC$QXZz;S2A z)w>^oHy>?v)q--`!pmuBe96PxP0u*inQvyFW(llfv9 zXV1s*Jh`y2H%B3ZTA(AzpsQ?hb6_PyZ=c1?_B4fbl>G%!@ubJln=!)x0000#DY{xaiib^#X=YT4@yE_&2#eBulEdzjE`u&@G%2(&u{J-<}d(^uY4W_kMfEX z@!X)AR9F&FL?RJyJRUzvBoeqN{5kY`z3wcM0+du73~_0|*lac! z42Dw(Eg1o{Ash}P8jXrqN+1w`*XxDD;ShmPCZC7#4;wWbHoMvBl$=zF-`?*9j*Nbjd=v@OWt_BgKxP-3wd zy37?ATx&$b+&zRM!K;BD%Okw`Sb@&Pak8$KRX19jWZmC0&n*Ggv%j8nvSPDFw zEkV65AGOoBQ8kf`R|}Px*&INNS%osq9b{Fq2I(x6@xM>tg=vRLF?I`0rWzHyRc>}g~)F_Qn`A>)C_iwK%Z zrIJ;xR)UI1Y4Ozts|-Nho;q zVk9-bX)%F~!;63iu$Fk=VJn3~fmb5S@@)ZqjBT2{f`vT`b2}zxb0$o;EF@G3&BHK^ zc)`1kUzo^Qkk$?KFKHNBD?nP-MJ3b@&4fg;g5l2wMi^g?9qj+~@b;62o_U1_S1J`g z7m^UMg25FX1MJ5AQxAJ5F5WDt=$=-@JV-!LHA2vuxl9kN>PS8x??^AINH6LjF*#nbk4}=n3gfWp$kEX5IpHS zYiQ{@d7Nl&d$#+7-TckP&Q}N91e-C#5QQ<|d}62BjvZR2H60wE-&H<>}%WqH(_V;zPbB1rgSSSC(0? zWlQ#?N3UgnJ9m2C29w!SwoOo5_2Iq!<8vCyEoDoj@#oV($oJEg6Bj@;nD|2g8 s%L|>IZ381yx9RvPhV4J)*SeoEV4lyr#k*`nfWBbxboFyt=akR{0DpOPi2wiq diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_key.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_key.png deleted file mode 100644 index d61648452284da1bc28b10385f95b5d2bf027901..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 616 zcmV-u0+;=XP)-tZUVHjYHp;RjQ0M0pRlXN=mLv{hk9Ebp9&~+Wj-T9IkpzWPWd#fZ)d=zV^~S`;LE*!&u-?g42^wwN&Xr1~#d5ifl_2*B1OoS}CDno^8a50ArfE8;stQF>AP54J@H~%T zFz84s;dO!QJKD36(~!QOg!t_^gfFcSKDU4yK0+Ypg$NT^mIYcQ6bk*3P(lBLh7Df_ zTu=2xC#+-_%)|{Cv8zz0t|0y4D5M`xAc{gwOKc`ou<*&VjUREFHs1qd<_xSkKeTBt zgyCi=@jj;&Ns^GsWWaIUl0Y2azcDlF@u{(P*!+EH;lnU~b|Lv{4|4Hdkh!qoQHiE$ zY#y>KFA0QEw=4Z|uV{0A^`Y=D}hB$GP&<$bi8q(u;p^0(my3Rz7fP}|# zGZ&#uor4@c3q9r|f?H6-UZmfgKx(iV(MQ`MPWB>iC~SxnN5H*zb*A3#zWwgu&c|}3 zn^g87H{pdeasl%Lhmab&jC?lES}7C?4BFDNA<}20hoY@w_IU%i*T;}}wh!589}7~7 z#Ug`-R~4j&+K_y4kW@X7qLr-)S5qVKU)tO;+kXJ++{vPI@{hVK|PhMVVx_`)vx~zUs}c9O-Ok{00000NkvXXu0mjf DS5_-g diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_link.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_link.png deleted file mode 100644 index bf7bd1c9bfd78d689c73ba67cf914182933ee68c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 614 zcmV-s0-61ZP)OOAS;jTeL{ZSdz-%)SMH9tDF;N4B6%j=d15J&5qy`F#vB?Ar zqS1nH@%ny_XSI*Y>) z1f5QYdmzT>YciP<3WehS<{GovEaLGv27>{*-7f0&I$yJ^L%ZGPv1YT$V|u;*+ZCWz ztHI~CDVsuy($SfR6-`N~K?9GTB#l%%0h7 z-q`K-y~E)+s8lMyTrPL8^_pUo)9G|SluG5pPqw6!LJB_PzyJUM07*qoM6N<$f^=yZ AYybcN diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_magnify.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_magnify.png deleted file mode 100644 index f6b74cc40f82fc83e4dfa6e9647ccc1b34e6ed7e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 554 zcmV+_0@eMAP)Vb2f>2}Fa82O3m(Ob=t*sniin`NpInLyMJgI`saru@YOPfh zy4g0#G*cV!#N%;Gq9_VH9?v%kjS3Rb1j8^;C={$Gp=lbj z*(?%?geA!5^Pok%UauwjA)v4g2`HedDw4_Mk4hhBQt?e7YJ5(hcj|3dNu^TOPGnjB zTTsqd3GIZ=Bb`n=7no)dflv&K(lsWw?lH6T1Yht0F9qgIuzh}ym0%n<3d3EBWB*pg z+G!I0lbAEXyd>k|QNuwr4=KX1D+tLPv)j@C1=N4sA4NF9A>HcO3G47*Y6!+SrUH-7 z1hb;^#S=r|`aMh>J#dWruAEf}gcR(DRUC`ZUev&$Sbh0SgLiTXeeHEU<$_YV;9281 zym`igIE%Sm8DpDw7@71Tv^EB5xSdUR*0$Mqp+Wq8OoaZtOg52&)zZ;;M=7#C1Yd6x svjx>8ad4e2x|*xHHwRjcjs6zA0XLDUqKT6dS^xk507*qoM6N<$f*wetruo^Ag2=LamM1T#~4RmC^m`_ zs}H7d&XJ}mg+hU?tu0noRvyjI&o2SRAeYZFesYkts79I^jJ7!A7%6nJwq8O?iT55M z1OQ` zbL{!Cp5o*IRmE9PInMCSPjTwfT~J+EYkz}tjxY=fg5Yf6EQ@DG$0kMJ9h^&$W}9BU zP1oj2;?MWVkKIEl)r=Y;L^Cx2q|>!)qJJ8zE7-V*-Cf7V8_2#1c0N975t~+&QUpQJ z5(uo(-O_`%Rj@U@t>JYAgd!>L?0Idxtd#oW2gc!jinsAEva8|kF4#Ic**mmml_{d^$s}Q5Q)KCys4sfck5bP1SyeHwh2`A<@N&t2j0^lgHC_^(pAAPCNjwl+>AN%C4Ll>_8Hjda%9 oS~i=#*e)>KsPlg0=2)Qg6BCqJ=F8HdGXMYp07*qoM6N<$f@i)wr~m)} diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_office.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_office.png deleted file mode 100644 index a65bcb3e1e9613cd9e4950850db43d7025a5fdf9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 779 zcmV+m1N8ifP)JNR5;6x zll@CmQ5eTZ^k*a#RQf}fVOdd`5NJh6S(>6Cf$wEW#f&JyAR#GAn9>Gml;nOf3WCDa z5({5&UB$(IF?G#$x4X@Ickg!Y-HU!Z_rzX=qAq-XI_LS^=lOCT0|0{#{kBkYDS7{3 zD`iu%E=`cDX_^#^#n$5SIQ|4Zhsk8>N|zXHXG@*41$i-7`Jr{8`3S_OEcmY|RF48wXkk?WpdVM4OePBSbfh z#4_=eXJg@3epx~gi>QbUmO}Bm(ENN3+@c?jWiKvSrm(o|W}Ud*?vy~fn1!V~Cl4kB zI-;c!8f~-v)jX82%EG($>?;KSD$64f2&4qQ#=Yyrcpy$57RAVuV#vKMP)0hT$r6m# zc^F^XaJ8R9Q|}x^NoJYIvYZkq-z}Tnj@UJK2l2H zG}p+VvtjP2Z%bsb$~7QLJ9#pC0dKi`ppOd^_V;ME6tdzC0PtV|r=@e@37O`%0k^=5^`%cf$eu00N17Ro!{^30krz>a%3j34C?*{Mt2^a4~ zK=P+Qq%|f;Tc&+9ps;@Mw`EE%rgs&#y=j6BUGg96oIqdwj9-fiy*N(|@o)eD002ov JPDHLkV1j#pUPb@_ diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_paint.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_paint.png deleted file mode 100644 index 23a37b891c2f5faa3b8128d45373ceab794ca609..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 688 zcmV;h0#E&kP)PK^TXt2QS_@2qt2T|9~baC-vaPn=ziepcfAwB$0!O2Q)E}1e8!q+9)KT5JVKU z7HNY}h##OS-BxWHWjD0wrPDeEfUClHs%Q6&2u@FTOJkKMQN|_Rlw6rQz$gPzqGNtj z#ruSeFeh835JJFiM6vp@6M5bXj%k7CMt%SIwfbF_fD-3*Os`9Ly_Q3WQ_SX33E{pX z9_WIeeTCGQ3wYALpBcK+P-iuw;3i&7xCua37k5# z`>c`M@sGeC7cdsdz`aE9lOz!hPholbyz%T85LYf6O*@SA+9&+^7k>+4M8$A8iNQq{ zQvn8k?-+dU`Z@gK0z$EtPV#+`^OH`R@cE-cuE&_!D)SZGxmQxeobP_Zwq zMEgi6ePN45N`|V1so0uE8^}1xw8s;VM%Ai@7} z2-&Cyvez_-O4?6uv{zTaj|YeYEk34i~K@`8YW2g{x* zc;7z3lItpVy_et{Z-ZZ)<@*%{l7Ao8mu@V7*gz<_1##mwW*%LEwCdzNsVLYx2*T-J z#HeQ*_a=R~KDdVNk$EVgAIRl$oQi_(`_IrdJciDpH|Xe{K-YsMtc!cRnFi$qzsr4z z5*$;ecov%3->1{YNy6-Gf(Ecy&_I$CjI#laeuE+S120^|Vjsf)W&i*H07*qoM6N<$ Ef)^4A_ab^avY?n0hpS-#mn_4{O$e%cm-@NH=3`90Wq+3`~HKArSdfX`&Z12 z(CY$VW-MNtXX4xy%yUeE?}*~0-|iByA@ZrwXgph4S*bhcc5{HB!DFVm_v}P*g7+Q~K}7K0lcp(^N@X>U zV`{ZpeIf${R6Hgg4FL^`X$Eu75k(PE6ycl$AW0Ic)#@rR7Z(7;V?i-dR1K935Jgcx zPfkwK>2wGokf!Nih^ARp6-6arYFG#(9Ta!x93nFEjoA==z(g?#sDg?Owk?Mg7K+>l zWYsf(<`#+$h9Sp6gFOg_dd+80SkUpk&xM7h0`Sov9W73spU;GP073|VfZ&Gd$J$*0<~TV5aPS|qWH57|VJz+d0000vYep8SaFV10Q$h+;hIUPX_=v5b}%>Tm<(&j1&5;I!55C)oN0s(P%ZB zP3Q#ahfpXKWF@S?jm4U#fv)QovMhrriclyNs6-G12#3R##4PSZ0VY(dRWJ;Lwuq{# zAW0Gwi$yA^R4RZ!;W+L`f&%x{=D^VK#BBWL4Ys{;*!A7Q;!=dN<&D8*GzGaF4`hV4 zDbY0{NrMX>ZqF=0((gR5-zL$kC*b)!fwu{Euru|XrG<$^n#@)7i_>rCmRxnDq>$Y%gJaCkRd|tE*a2x05Pe!I^e13o69#&RQZ36s0 zB=O|K2Yi(jsMqThn}9t?f5E-)L^naZ+db$&%M$!bCdm=jv7?t_lB?3&%Ltq(>ESw? c;MI421LCcoDG!2@;{X5v07*qoM6N<$f`UZt7XSbN diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_picture.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_picture.png deleted file mode 100644 index 134b6693687b2fa5fe36d48a9c0b8001f937c741..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 650 zcmV;50(Jd~P)VHAd+bMNh~)LLRqN>D)-jd9UvB%+hyKX5U|&4t0)fzgD-MPpQ$nHU%yoz=vI zMGb>1Xu!6Hw$NT~@Au<4P-+{9;Uw?&oj31uzH>xX0T7Xkz!(tn|Ed9-s_FqyReC13 z(ll)vW1O{Ck5ihay12Ob2ABc@RUI;zHpaMiyRDs0r>|D4rHw{ItJSJnYjt~jTbGuW z`X(~}?!&86q40R8<4zYw;$qi0^3ec=c&<&H;r`8W%H=Xymf^i;Wo6~<+}zx2UMzpC z*6MZN?(FMv`n|KO3(KFiUaucP0;Z!@LcUNa%8#vGK5aZ>wDgB0Gi=t*argWJcdlMQ z2#MpEX0wU+9&0U?N(F#OgpviU_Y{jYMsj65U3|PjwUOY}lUYj?MTiK_Il}NCVx-Eh zDx-TzMk7se+M#W_>?A1-x}ZXw3kkyz5kW)_hkjsi@RhKadN#H$Hq)$07*qoM6N<$f}lhwPXGV_ diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_powerpoint.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_powerpoint.png deleted file mode 100644 index c4eff0387d5888c638ba09473ba6d2369f7b56f0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 588 zcmV-S0<-;zP)HU2HvUSp%6 z*n}iP63IK?dpo;h@sj9~pcxo;VVTc-XLiP@DgefqE#NE=@oyUd-&HjLpsLIuSFXV-EMck)oQ(A`s%*^&wf0(rNiNHsU%=0Rw;WC z(kbc37l6fo`-0uR!pYkYv8U^3?nsh^@pw!K0TH3uYyx1_2>|JbXPmfskJ|1YAw9w! z9`N)1^Aesr;y5Nr5-ODn)oOL|CGi}f9!&iVwpK$khlIX10X$H6^A_stBJqvLhU$?V`QXqKme*s~gVDJ4A;LTs_e15jhc1;By a82kqHEPVYFAD2!50000JNR5;6( zlS@kiVHAe7MZY2;Xi-5)WxDDgv@tCUl*&p14T@Z~3ThM5LP4tuQfLu@EnG;nXc<8S z6&3BN?fx-cv-Kp6>HRiNTHE>$X( zD&=w+?GWC>?RLAGC6Yix;an~UmSt)tSf}1VS6N1N2ONORdD? zaj}w6DAZZdOud9Ep?M?{iQWbE5^9HLLZZF|1kdy0Tu4InEuboP9@nvbZ-P0n4AZTy zyMRIxRDmUE#LdqYuD=-Qz4N^bC`_#S7vcLn1M}{J(Wl3#c4VWczu&)AjUlh(11>gp>f`wv{KnjF%!aA*Jk N002ovPDHLkV1kkt*XsZP diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_ruby.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_ruby.png deleted file mode 100644 index f59b7c4365fa1720af1aa04eb47167ddaa6eeed4..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 626 zcmV-&0*(ENP)ZS(e|#C2>JN4>y}l*tQ*E7zP@R2CCJnkW?xa6bgk%(hgtZ z0=~d?U3i`+Mvi4!&~+WPT1^NX#{u6&QIx+DE(oR{&T5&-ovF?@wGw)P&AtpHZa|G%V*GUUqL@@!d4V$`8=##4)ytY959JG zdc&Kho)&AL70^i z!PEmeeDWCB-UbK(*4JST44^tV2z_J(dn~+vBMJT97_7rzFio=~XczIv?PQ5$v%u~y zu(bteXb5I1h2zCV{Jc2~V{{yzZipgsP6;k264$*#5q?GzCm|CPa9CKqm4b116h3Pu z?+%Cm52plC8|5P0@igf2GV1KkCfk{Zecu=G@VNrf>s%g9c5D%@cfxVb6$nY`1IW=4 zt10QqSps_2JLp0f3I0j0u>#qA;v!+T))KEbCg|mo3q0pG{OR}p0fPds8+K~d>Hq)$ M07*qoM6N<$g1S2e3jhEB diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_stack.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_stack.png deleted file mode 100644 index 44084add79b9a0fc3354d16bbd4b4b5ff8095da7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 317 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!60wlNoGJgf6SkfJR9T^zbpD<_bdI{u9mbgZg z1m~xflqVLYGB~E>C#5QQ<|d}62BjvZR2H60wE-$R@9E+gqH(@-qA%AW0|7U8+xDRI z0k`B18}ImRw2g{jTGP$Pmx3yI6F_2s&$|`cJ!i0UN zB3H;=r{#{FwLaNVJ&hZl9+MTHGx1T^-A=Q0?hRb#8a~x50X%;`b6ik3cw=#XdxWy= zgrpBoDjpwP&g9<9h3x!k_B!?vuTJVkmIJ-U N;OXk;vd$@?2>|rNdMN+^ diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_star.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_star.png deleted file mode 100644 index 3a1441c9a12062a4bb3d706000d3ca14399aebca..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 565 zcmV-50?Pe~P)SCZIX8XZzY2l?gCw6LlgWJ5Avz#QX4|&mI8LN)w~J1vgL=KLAhlWz*=#m~gyvxa z&;iC6gb?aZvMdXxX`<0+D1hs_pqJ!wxqlEH;CJ)je~uL(gpi@v>!I0f_Kl=E(E+Tq z26na*9gribxx-Oft(HnstyXUUy!39&E-cI%J5Rsy;(PGZH{g{ty!HVC&yGPT3H8x# zw{^gBPW)O0FMoh{k%l<`1a|To_Wl&u&-GXm8izU|&<&utILc4wc6s@u1bmTz6x{qg zTw@7=FQRcg&r`h+gcR$*Jbv+*DPk7v)B@e0o2 z6IlBXW&8xh@9)YKiV~2>+z&XKd24JT55YWz&JtfvCg4r^~bLP79-yS@n$OW00000NkvXXu0mjf DStt2z diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_swoosh.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_swoosh.png deleted file mode 100644 index e7708292adabf4821612bfca032cbd019c63180b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 634 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 diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_width.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_width.png deleted file mode 100644 index 1eb880947ddf3e745c29e8d9dc90f09c7e6e323c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 309 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!60wlNoGJgf6SkfJR9T^zbpD<_bdI{u9mbgZg z1m~xflqVLYGB~E>C#5QQ<|d}62BjvZR2H60wE-$R?&;zfqH(@;q9b3Efq-lM(nr^( z=EYR73-9e)UYMWsXy%?aZsD68Yyv^2$~6QgEcljw%kx>O(f-gQ?@fOOx3A-0+Qw?O zRx~W)kn~Qe2d6f9nMG#g9Q04Mk==M~N!Dglvxk!fgVh#w@ZV$IY1+Xc`d{d2UcaP~ zfWp)_Ivqj}l2SPy^9ZWy6rG9Yx4v67_uA&&9|XA~5-#3)W3%em1peD8RWH^#O%XoM zxMPud%}GTj#~*+7JMxTd!`{^Q+>(D3*|@KV`*G2;{QnANOxu1$r2xIe;OXk;vd$@? F2>@zac~<}c diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_word.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_word.png deleted file mode 100644 index ae8ecbf47672a874c0958d0d113a56162c2bd364..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 651 zcmV;60(AX}P)hkjP zNW|QGv-YFNLN^qH@tJycPNG5ti6B7;r4mEr#lr@*T8*M85D`{ZR^BWwF23T<%MYIh zdC)S*p=|xk^!~H=+HSZ183~y8v4|mYmZxt&)5{{~>J`>E223Q5>T$=~mtA71q-jdG z+eJhOAyBW^0k9Gk1+rX8)zFx((CG^&tDY>6XaS~Fy!WJON|Gdujg5^~Vzt@o%BcYLiNiTQSD`zL^ociBz_>bDlpw3kriQ@Z`bVsGz-_6N>$&gTDiKDTKR^ z-hB*tHa^>!oD~5TK^0UK5rZ}RBm50Bv}S-yA%s=Ha5RYb{)!z2N&$&64gfhybBu8p lh~_|?8^bu;BRYt{<}Yrwd83Y=s?Goa002ovPDHLkV1l%3CP4rI diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_world.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_world.png deleted file mode 100644 index 6ed2490ed1432d5d667a76235360824a1088e928..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 734 zcmV<40wMj0P)JT{hN;C#tgf#9krG=I>5!<*aE1_(spcgF}<`n4i zJi-}^6UUeU4jUFwdCiVPDm%`Zx^UBa8J(mnR6wEgz^}o8;)M*Y(@l_!Kfv)}4+NuM zaPXE50z)r)9=D=SR|RIqfQ^j}Hu!fzMeQBo+@PZk1G8hOw|vBTvkx`HM)Xe9q3xao z@`p0`NO!2904FHSLA6E@Y-O6zH$DQzvq@aHsz}}<(!v(Z_+EodX%R&NZW75g+nENo zV0020rxE^;7d!067AN>6*+&YLp$9uH6F-=In`XC{Cn%+o|5)b&boEPr02w@|P*oGm QmjD0&07*qoM6N<$g78X0Q~&?~ diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_wrench.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_wrench.png deleted file mode 100644 index fecadd08afed92536be91ab12d8e37b6bf410d5d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 613 zcmV-r0-F7aP)wK%m(L+9IV|s|#(WRl-O^4GvaQsnHq|OstfO zIJ3}3<01}YGARE4m!7=)QisvlHUo!Qymx-@-t*p_129Ko-#pVI)6#!*kLj-AGXWNR zyA_{wKii_amK7^YT-v z6#plaNm#8`-kz@OvjIt^4%IN{@J3bR zRI}ME1Mv85p|%;RK>ViR>APPLB4;;BpCtqE@P+*7!G>I4UjNx~e>r3HA^tWCQ@S)l z{BslcSwL-CxQ&_ZZSv_g0Tu{yi*X){Mt|W7)lbE`SQxFP00000NkvXXu0mjf;)M*S diff --git a/node_modules/express/node_modules/connect/lib/public/icons/page_white_zip.png b/node_modules/express/node_modules/connect/lib/public/icons/page_white_zip.png deleted file mode 100644 index fd4bbccdf1643f4ff5022fbc59b82546e259317e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 386 zcmV-|0e$|7P)_QM!1S$Bhw4w+iRuFWf;tfR6D%SMJrb+tx zC9R6{2>Ou6#juIy6u(I?|;&Owi$sRB4^20apB5xE2 z#B9XekY66S6lzfCL!eEQRgo0LokTA55@Y#%_wN!TXPw^Q4IIXsG~v#u_4t;x_HM16EQ@QRY+rut&97&UefsPmLrQ5P zBC2kcbux9L%2bJz$P$XV$*zSxb2e@6_3O#;&!FD<&hLjGn%~%en;7)djE^d6!t$lW7GyIOKlQ46hr`Z zjLNuRDP_53dNoN?wd&HMgL^m1DXFU<5dQsrceN>fSz00000)O9XRTN^$%%`*Fg>ryDtc(lF@?b>dE!20r+y z#Q*>(wbV5H`-E4Do={CJp7=ERhw15hgZi)?jRG88 zzVz(5;g?Td1izJyO33bhjg2Qc7FVY@f9!o)Gu?DII~vm-Dc?}3M!fsgjP?F(7`rgg z+xOk8XD)e?Zl=5+un`5!7kr?F=eq)K-5uqr%yU$1hLv){Vlm=)*5~`lwMciiXFu*g z)*Jkz6AF>#zb(Vx`Iv{bdGZHtlW)v(y5k^|xgSUc9%0}S20nrYrO}78ofk?bV!5)4 z=Ngz@+$9N1>>mA%IWx`Fqa240bWkiW;2TZgd8CZS0U}@mknC;!2;wi$eI@`h0y2JS`Eae0CW}q(2(%!m8 zWq$`PDU>LT1_y*bBv#P5<@q0@ttz$hIH}YMDvAigCc=y*)jY-VOpTd;A8@3t7Xh4r z0KTWOk;N2Ox4!&&^4B*no$WtTX!BXB)rg!y8dvGgKBQKLJNXRRp0}Bsjd1|LNQX~c zbC~fjrk2iL@4dYF*vt;}dFn(%h)n_-vzEIHMOKRkdF%3Lq|zBgKm_h>TEq!))nWjq zzn;B!?!(dQcHu$#=JF`cS&W~C`WHFW^B!~MI#k)>1Vk&eQy8P1O`J6V04{D@|7d6^ zyBABnh-d^H0FX&L07M||E0n_dp4v&Q%PSE9p#R#Hq)`5I_(B5CE#q dxjPz0{s-<+c#AC!i7@~G002ovPDHLkV1iPlpuqqD diff --git a/node_modules/express/node_modules/connect/lib/public/style.css b/node_modules/express/node_modules/connect/lib/public/style.css deleted file mode 100644 index 32b6507..0000000 --- a/node_modules/express/node_modules/connect/lib/public/style.css +++ /dev/null @@ -1,141 +0,0 @@ -body { - margin: 0; - padding: 80px 100px; - font: 13px "Helvetica Neue", "Lucida Grande", "Arial"; - background: #ECE9E9 -webkit-gradient(linear, 0% 0%, 0% 100%, from(#fff), to(#ECE9E9)); - background: #ECE9E9 -moz-linear-gradient(top, #fff, #ECE9E9); - background-repeat: no-repeat; - color: #555; - -webkit-font-smoothing: antialiased; -} -h1, h2, h3 { - margin: 0; - font-size: 22px; - color: #343434; -} -h1 em, h2 em { - padding: 0 5px; - font-weight: normal; -} -h1 { - font-size: 60px; -} -h2 { - margin-top: 10px; -} -h3 { - margin: 5px 0 10px 0; - padding-bottom: 5px; - border-bottom: 1px solid #eee; - font-size: 18px; -} -ul { - margin: 0; - padding: 0; -} -ul li { - margin: 5px 0; - padding: 3px 8px; - list-style: none; -} -ul li:hover { - cursor: pointer; - color: #2e2e2e; -} -ul li .path { - padding-left: 5px; - font-weight: bold; -} -ul li .line { - padding-right: 5px; - font-style: italic; -} -ul li:first-child .path { - padding-left: 0; -} -p { - line-height: 1.5; -} -a { - color: #555; - text-decoration: none; -} -a:hover { - color: #303030; -} -#stacktrace { - margin-top: 15px; -} -.directory h1 { - margin-bottom: 15px; - font-size: 18px; -} -ul#files { - width: 100%; - height: 500px; -} -ul#files li { - padding: 0; -} -ul#files li img { - position: absolute; - top: 5px; - left: 5px; -} -ul#files li a { - position: relative; - display: block; - margin: 1px; - width: 30%; - height: 25px; - line-height: 25px; - text-indent: 8px; - float: left; - border: 1px solid transparent; - -webkit-border-radius: 5px; - -moz-border-radius: 5px; - border-radius: 5px; - overflow: hidden; - text-overflow: ellipsis; -} -ul#files li a.icon { - text-indent: 25px; -} -ul#files li a:focus, -ul#files li a:hover { - outline: none; - background: rgba(255,255,255,0.65); - border: 1px solid #ececec; -} -ul#files li a.highlight { - -webkit-transition: background .4s ease-in-out; - background: #ffff4f; - border-color: #E9DC51; -} -#search { - display: block; - position: fixed; - top: 20px; - right: 20px; - width: 90px; - -webkit-transition: width ease 0.2s, opacity ease 0.4s; - -moz-transition: width ease 0.2s, opacity ease 0.4s; - -webkit-border-radius: 32px; - -moz-border-radius: 32px; - -webkit-box-shadow: inset 0px 0px 3px rgba(0, 0, 0, 0.25), inset 0px 1px 3px rgba(0, 0, 0, 0.7), 0px 1px 0px rgba(255, 255, 255, 0.03); - -moz-box-shadow: inset 0px 0px 3px rgba(0, 0, 0, 0.25), inset 0px 1px 3px rgba(0, 0, 0, 0.7), 0px 1px 0px rgba(255, 255, 255, 0.03); - -webkit-font-smoothing: antialiased; - text-align: left; - font: 13px "Helvetica Neue", Arial, sans-serif; - padding: 4px 10px; - border: none; - background: transparent; - margin-bottom: 0; - outline: none; - opacity: 0.7; - color: #888; -} -#search:focus { - width: 120px; - opacity: 1.0; -} diff --git a/node_modules/express/node_modules/connect/lib/utils.js b/node_modules/express/node_modules/connect/lib/utils.js deleted file mode 100644 index 35738b8..0000000 --- a/node_modules/express/node_modules/connect/lib/utils.js +++ /dev/null @@ -1,404 +0,0 @@ - -/*! - * Connect - utils - * Copyright(c) 2010 Sencha Inc. - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var http = require('http') - , crypto = require('crypto') - , parse = require('url').parse - , signature = require('cookie-signature') - , nodeVersion = process.versions.node.split('.'); - -// pause is broken in node < 0.10 -exports.brokenPause = parseInt(nodeVersion[0], 10) === 0 - && parseInt(nodeVersion[1], 10) < 10; - -/** - * Return `true` if the request has a body, otherwise return `false`. - * - * @param {IncomingMessage} req - * @return {Boolean} - * @api private - */ - -exports.hasBody = function(req) { - return 'transfer-encoding' in req.headers || 'content-length' in req.headers; -}; - -/** - * Extract the mime type from the given request's - * _Content-Type_ header. - * - * @param {IncomingMessage} req - * @return {String} - * @api private - */ - -exports.mime = function(req) { - var str = req.headers['content-type'] || ''; - return str.split(';')[0]; -}; - -/** - * Generate an `Error` from the given status `code` - * and optional `msg`. - * - * @param {Number} code - * @param {String} msg - * @return {Error} - * @api private - */ - -exports.error = function(code, msg){ - var err = new Error(msg || http.STATUS_CODES[code]); - err.status = code; - return err; -}; - -/** - * Return md5 hash of the given string and optional encoding, - * defaulting to hex. - * - * utils.md5('wahoo'); - * // => "e493298061761236c96b02ea6aa8a2ad" - * - * @param {String} str - * @param {String} encoding - * @return {String} - * @api private - */ - -exports.md5 = function(str, encoding){ - return crypto - .createHash('md5') - .update(str) - .digest(encoding || 'hex'); -}; - -/** - * Merge object b with object a. - * - * var a = { foo: 'bar' } - * , b = { bar: 'baz' }; - * - * utils.merge(a, b); - * // => { foo: 'bar', bar: 'baz' } - * - * @param {Object} a - * @param {Object} b - * @return {Object} - * @api private - */ - -exports.merge = function(a, b){ - if (a && b) { - for (var key in b) { - a[key] = b[key]; - } - } - return a; -}; - -/** - * Escape the given string of `html`. - * - * @param {String} html - * @return {String} - * @api private - */ - -exports.escape = function(html){ - return String(html) - .replace(/&(?!\w+;)/g, '&') - .replace(//g, '>') - .replace(/"/g, '"'); -}; - - -/** - * Return a unique identifier with the given `len`. - * - * utils.uid(10); - * // => "FDaS435D2z" - * - * @param {Number} len - * @return {String} - * @api private - */ - -exports.uid = function(len) { - return crypto.randomBytes(Math.ceil(len * 3 / 4)) - .toString('base64') - .slice(0, len) - .replace(/\//g, '-') - .replace(/\+/g, '_'); -}; - -/** - * Sign the given `val` with `secret`. - * - * @param {String} val - * @param {String} secret - * @return {String} - * @api private - */ - -exports.sign = function(val, secret){ - console.warn('do not use utils.sign(), use https://github.com/visionmedia/node-cookie-signature') - return val + '.' + crypto - .createHmac('sha256', secret) - .update(val) - .digest('base64') - .replace(/=+$/, ''); -}; - -/** - * Unsign and decode the given `val` with `secret`, - * returning `false` if the signature is invalid. - * - * @param {String} val - * @param {String} secret - * @return {String|Boolean} - * @api private - */ - -exports.unsign = function(val, secret){ - console.warn('do not use utils.unsign(), use https://github.com/visionmedia/node-cookie-signature') - var str = val.slice(0, val.lastIndexOf('.')); - return exports.sign(str, secret) == val - ? str - : false; -}; - -/** - * Parse signed cookies, returning an object - * containing the decoded key/value pairs, - * while removing the signed key from `obj`. - * - * @param {Object} obj - * @return {Object} - * @api private - */ - -exports.parseSignedCookies = function(obj, secret){ - var ret = {}; - Object.keys(obj).forEach(function(key){ - var val = obj[key]; - if (0 == val.indexOf('s:')) { - val = signature.unsign(val.slice(2), secret); - if (val) { - ret[key] = val; - delete obj[key]; - } - } - }); - return ret; -}; - -/** - * Parse a signed cookie string, return the decoded value - * - * @param {String} str signed cookie string - * @param {String} secret - * @return {String} decoded value - * @api private - */ - -exports.parseSignedCookie = function(str, secret){ - return 0 == str.indexOf('s:') - ? signature.unsign(str.slice(2), secret) - : str; -}; - -/** - * Parse JSON cookies. - * - * @param {Object} obj - * @return {Object} - * @api private - */ - -exports.parseJSONCookies = function(obj){ - Object.keys(obj).forEach(function(key){ - var val = obj[key]; - var res = exports.parseJSONCookie(val); - if (res) obj[key] = res; - }); - return obj; -}; - -/** - * Parse JSON cookie string - * - * @param {String} str - * @return {Object} Parsed object or null if not json cookie - * @api private - */ - -exports.parseJSONCookie = function(str) { - if (0 == str.indexOf('j:')) { - try { - return JSON.parse(str.slice(2)); - } catch (err) { - // no op - } - } -}; - -/** - * Pause `data` and `end` events on the given `obj`. - * Middleware performing async tasks _should_ utilize - * this utility (or similar), to re-emit data once - * the async operation has completed, otherwise these - * events may be lost. Pause is only required for - * node versions less than 10, and is replaced with - * noop's otherwise. - * - * var pause = utils.pause(req); - * fs.readFile(path, function(){ - * next(); - * pause.resume(); - * }); - * - * @param {Object} obj - * @return {Object} - * @api private - */ - -exports.pause = exports.brokenPause - ? require('pause') - : function () { - return { - end: noop, - resume: noop - } - } - -/** - * Strip `Content-*` headers from `res`. - * - * @param {ServerResponse} res - * @api private - */ - -exports.removeContentHeaders = function(res){ - Object.keys(res._headers).forEach(function(field){ - if (0 == field.indexOf('content')) { - res.removeHeader(field); - } - }); -}; - -/** - * Check if `req` is a conditional GET request. - * - * @param {IncomingMessage} req - * @return {Boolean} - * @api private - */ - -exports.conditionalGET = function(req) { - return req.headers['if-modified-since'] - || req.headers['if-none-match']; -}; - -/** - * Respond with 401 "Unauthorized". - * - * @param {ServerResponse} res - * @param {String} realm - * @api private - */ - -exports.unauthorized = function(res, realm) { - res.statusCode = 401; - res.setHeader('WWW-Authenticate', 'Basic realm="' + realm + '"'); - res.end('Unauthorized'); -}; - -/** - * Respond with 304 "Not Modified". - * - * @param {ServerResponse} res - * @param {Object} headers - * @api private - */ - -exports.notModified = function(res) { - exports.removeContentHeaders(res); - res.statusCode = 304; - res.end(); -}; - -/** - * Return an ETag in the form of `"-"` - * from the given `stat`. - * - * @param {Object} stat - * @return {String} - * @api private - */ - -exports.etag = function(stat) { - return '"' + stat.size + '-' + Number(stat.mtime) + '"'; -}; - -/** - * Parse the given Cache-Control `str`. - * - * @param {String} str - * @return {Object} - * @api private - */ - -exports.parseCacheControl = function(str){ - var directives = str.split(',') - , obj = {}; - - for(var i = 0, len = directives.length; i < len; i++) { - var parts = directives[i].split('=') - , key = parts.shift().trim() - , val = parseInt(parts.shift(), 10); - - obj[key] = isNaN(val) ? true : val; - } - - return obj; -}; - -/** - * Parse the `req` url with memoization. - * - * @param {ServerRequest} req - * @return {Object} - * @api private - */ - -exports.parseUrl = function(req){ - var parsed = req._parsedUrl; - if (parsed && parsed.href == req.url) { - return parsed; - } else { - return req._parsedUrl = parse(req.url); - } -}; - -/** - * Parse byte `size` string. - * - * @param {String} size - * @return {Number} - * @api private - */ - -exports.parseBytes = require('bytes'); - -function noop() {} \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/.npmignore b/node_modules/express/node_modules/connect/node_modules/buffer-crc32/.npmignore deleted file mode 100644 index b512c09..0000000 --- a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/.npmignore +++ /dev/null @@ -1 +0,0 @@ -node_modules \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/.travis.yml b/node_modules/express/node_modules/connect/node_modules/buffer-crc32/.travis.yml deleted file mode 100644 index 7a902e8..0000000 --- a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/.travis.yml +++ /dev/null @@ -1,8 +0,0 @@ -language: node_js -node_js: - - 0.6 - - 0.8 -notifications: - email: - recipients: - - brianloveswords@gmail.com \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/README.md b/node_modules/express/node_modules/connect/node_modules/buffer-crc32/README.md deleted file mode 100644 index 4ad5d64..0000000 --- a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/README.md +++ /dev/null @@ -1,33 +0,0 @@ -# buffer-crc32 - -[![Build Status](https://secure.travis-ci.org/brianloveswords/buffer-crc32.png?branch=master)](http://travis-ci.org/brianloveswords/buffer-crc32) - -crc32 that works with binary data and fancy character sets, outputs -buffer, signed or unsigned data and has tests. - -Derived from the sample CRC implementation in the PNG specification: http://www.w3.org/TR/PNG/#D-CRCAppendix - -# install -``` -npm install buffer-crc32 -``` - -# example -```js -var crc32 = require('buffer-crc32'); -// works with buffers -var buf = Buffer([[0x00, 0x73, 0x75, 0x70, 0x20, 0x62, 0x72, 0x6f, 0x00]) -crc32(buf) // -> - -// has convenience methods for getting signed or unsigned ints -crc32.signed(buf) // -> -1805997238 -crc32.unsigned(buf) // -> 2488970058 - -// will cast to buffer if given a string, so you can -// directly use foreign characters safely -crc32('自動販売機') // -> -``` - -# tests -This was tested against the output of zlib's crc32 method. You can run -the tests with`npm test` (requires tap) diff --git a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/index.js b/node_modules/express/node_modules/connect/node_modules/buffer-crc32/index.js deleted file mode 100644 index ab0e19e..0000000 --- a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/index.js +++ /dev/null @@ -1,84 +0,0 @@ -var Buffer = require('buffer').Buffer; - -var CRC_TABLE = [ - 0x00000000, 0x77073096, 0xee0e612c, 0x990951ba, 0x076dc419, - 0x706af48f, 0xe963a535, 0x9e6495a3, 0x0edb8832, 0x79dcb8a4, - 0xe0d5e91e, 0x97d2d988, 0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, - 0x90bf1d91, 0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de, - 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7, 0x136c9856, - 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec, 0x14015c4f, 0x63066cd9, - 0xfa0f3d63, 0x8d080df5, 0x3b6e20c8, 0x4c69105e, 0xd56041e4, - 0xa2677172, 0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b, - 0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940, 0x32d86ce3, - 0x45df5c75, 0xdcd60dcf, 0xabd13d59, 0x26d930ac, 0x51de003a, - 0xc8d75180, 0xbfd06116, 0x21b4f4b5, 0x56b3c423, 0xcfba9599, - 0xb8bda50f, 0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924, - 0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d, 0x76dc4190, - 0x01db7106, 0x98d220bc, 0xefd5102a, 0x71b18589, 0x06b6b51f, - 0x9fbfe4a5, 0xe8b8d433, 0x7807c9a2, 0x0f00f934, 0x9609a88e, - 0xe10e9818, 0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01, - 0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e, 0x6c0695ed, - 0x1b01a57b, 0x8208f4c1, 0xf50fc457, 0x65b0d9c6, 0x12b7e950, - 0x8bbeb8ea, 0xfcb9887c, 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, - 0xfbd44c65, 0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2, - 0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb, 0x4369e96a, - 0x346ed9fc, 0xad678846, 0xda60b8d0, 0x44042d73, 0x33031de5, - 0xaa0a4c5f, 0xdd0d7cc9, 0x5005713c, 0x270241aa, 0xbe0b1010, - 0xc90c2086, 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f, - 0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4, 0x59b33d17, - 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad, 0xedb88320, 0x9abfb3b6, - 0x03b6e20c, 0x74b1d29a, 0xead54739, 0x9dd277af, 0x04db2615, - 0x73dc1683, 0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8, - 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1, 0xf00f9344, - 0x8708a3d2, 0x1e01f268, 0x6906c2fe, 0xf762575d, 0x806567cb, - 0x196c3671, 0x6e6b06e7, 0xfed41b76, 0x89d32be0, 0x10da7a5a, - 0x67dd4acc, 0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5, - 0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252, 0xd1bb67f1, - 0xa6bc5767, 0x3fb506dd, 0x48b2364b, 0xd80d2bda, 0xaf0a1b4c, - 0x36034af6, 0x41047a60, 0xdf60efc3, 0xa867df55, 0x316e8eef, - 0x4669be79, 0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236, - 0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f, 0xc5ba3bbe, - 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04, 0xc2d7ffa7, 0xb5d0cf31, - 0x2cd99e8b, 0x5bdeae1d, 0x9b64c2b0, 0xec63f226, 0x756aa39c, - 0x026d930a, 0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713, - 0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38, 0x92d28e9b, - 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21, 0x86d3d2d4, 0xf1d4e242, - 0x68ddb3f8, 0x1fda836e, 0x81be16cd, 0xf6b9265b, 0x6fb077e1, - 0x18b74777, 0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c, - 0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45, 0xa00ae278, - 0xd70dd2ee, 0x4e048354, 0x3903b3c2, 0xa7672661, 0xd06016f7, - 0x4969474d, 0x3e6e77db, 0xaed16a4a, 0xd9d65adc, 0x40df0b66, - 0x37d83bf0, 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9, - 0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6, 0xbad03605, - 0xcdd70693, 0x54de5729, 0x23d967bf, 0xb3667a2e, 0xc4614ab8, - 0x5d681b02, 0x2a6f2b94, 0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, - 0x2d02ef8d -]; - -function bufferizeInt(num) { - var tmp = Buffer(4); - tmp.writeInt32BE(num, 0); - return tmp; -} - -function _crc32(buf) { - if (!Buffer.isBuffer(buf)) - buf = Buffer(buf); - var crc = 0xffffffff; - for (var n = 0; n < buf.length; n++) { - crc = CRC_TABLE[(crc ^ buf[n]) & 0xff] ^ (crc >>> 8); - } - return (crc ^ 0xffffffff); -} - -function crc32() { - return bufferizeInt(_crc32.apply(null, arguments)); -} -crc32.signed = function () { - return _crc32.apply(null, arguments); -}; -crc32.unsigned = function () { - return crc32.apply(null, arguments).readUInt32BE(0); -}; - -module.exports = crc32; diff --git a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/package.json b/node_modules/express/node_modules/connect/node_modules/buffer-crc32/package.json deleted file mode 100644 index cdcaeb7..0000000 --- a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/package.json +++ /dev/null @@ -1,34 +0,0 @@ -{ - "author": { - "name": "Brian J. Brennan", - "email": "brianloveswords@gmail.com", - "url": "http://bjb.io" - }, - "name": "buffer-crc32", - "description": "A pure javascript CRC32 algorithm that plays nice with binary data", - "version": "0.1.1", - "homepage": "https://github.com/brianloveswords/buffer-crc32", - "repository": { - "type": "git", - "url": "git://github.com/brianloveswords/buffer-crc32.git" - }, - "main": "index.js", - "scripts": { - "test": "./node_modules/.bin/tap tests/*.test.js" - }, - "dependencies": {}, - "devDependencies": { - "tap": "~0.2.5" - }, - "optionalDependencies": {}, - "engines": { - "node": "*" - }, - "readme": "# buffer-crc32\n\n[![Build Status](https://secure.travis-ci.org/brianloveswords/buffer-crc32.png?branch=master)](http://travis-ci.org/brianloveswords/buffer-crc32)\n\ncrc32 that works with binary data and fancy character sets, outputs\nbuffer, signed or unsigned data and has tests.\n\nDerived from the sample CRC implementation in the PNG specification: http://www.w3.org/TR/PNG/#D-CRCAppendix\n\n# install\n```\nnpm install buffer-crc32\n```\n\n# example\n```js\nvar crc32 = require('buffer-crc32');\n// works with buffers\nvar buf = Buffer([[0x00, 0x73, 0x75, 0x70, 0x20, 0x62, 0x72, 0x6f, 0x00])\ncrc32(buf) // -> \n\n// has convenience methods for getting signed or unsigned ints\ncrc32.signed(buf) // -> -1805997238\ncrc32.unsigned(buf) // -> 2488970058\n\n// will cast to buffer if given a string, so you can\n// directly use foreign characters safely\ncrc32('自動販売機') // -> \n```\n\n# tests\nThis was tested against the output of zlib's crc32 method. You can run\nthe tests with`npm test` (requires tap)\n", - "readmeFilename": "README.md", - "bugs": { - "url": "https://github.com/brianloveswords/buffer-crc32/issues" - }, - "_id": "buffer-crc32@0.1.1", - "_from": "buffer-crc32@0.1.1" -} diff --git a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/tests/crc.test.js b/node_modules/express/node_modules/connect/node_modules/buffer-crc32/tests/crc.test.js deleted file mode 100644 index d4767e3..0000000 --- a/node_modules/express/node_modules/connect/node_modules/buffer-crc32/tests/crc.test.js +++ /dev/null @@ -1,52 +0,0 @@ -var crc32 = require('..'); -var test = require('tap').test; - -test('simple crc32 is no problem', function (t) { - var input = Buffer('hey sup bros'); - var expected = Buffer([0x47, 0xfa, 0x55, 0x70]); - t.same(crc32(input), expected); - t.end(); -}); - -test('another simple one', function (t) { - var input = Buffer('IEND'); - var expected = Buffer([0xae, 0x42, 0x60, 0x82]); - t.same(crc32(input), expected); - t.end(); -}); - -test('slightly more complex', function (t) { - var input = Buffer([0x00, 0x00, 0x00]); - var expected = Buffer([0xff, 0x41, 0xd9, 0x12]); - t.same(crc32(input), expected); - t.end(); -}); - -test('complex crc32 gets calculated like a champ', function (t) { - var input = Buffer('शीर्षक'); - var expected = Buffer([0x17, 0xb8, 0xaf, 0xf1]); - t.same(crc32(input), expected); - t.end(); -}); - -test('casts to buffer if necessary', function (t) { - var input = 'शीर्षक'; - var expected = Buffer([0x17, 0xb8, 0xaf, 0xf1]); - t.same(crc32(input), expected); - t.end(); -}); - -test('can do unsigned', function (t) { - var input = 'ham sandwich'; - var expected = -1891873021; - t.same(crc32.signed(input), expected); - t.end(); -}); - -test('can do signed', function (t) { - var input = 'bear sandwich'; - var expected = 3711466352; - t.same(crc32.unsigned(input), expected); - t.end(); -}); - diff --git a/node_modules/express/node_modules/connect/node_modules/bytes/.npmignore b/node_modules/express/node_modules/connect/node_modules/bytes/.npmignore deleted file mode 100644 index 9daeafb..0000000 --- a/node_modules/express/node_modules/connect/node_modules/bytes/.npmignore +++ /dev/null @@ -1 +0,0 @@ -test diff --git a/node_modules/express/node_modules/connect/node_modules/bytes/History.md b/node_modules/express/node_modules/connect/node_modules/bytes/History.md deleted file mode 100644 index 1332808..0000000 --- a/node_modules/express/node_modules/connect/node_modules/bytes/History.md +++ /dev/null @@ -1,10 +0,0 @@ - -0.2.0 / 2012-10-28 -================== - - * bytes(200).should.eql('200b') - -0.1.0 / 2012-07-04 -================== - - * add bytes to string conversion [yields] diff --git a/node_modules/express/node_modules/connect/node_modules/bytes/Makefile b/node_modules/express/node_modules/connect/node_modules/bytes/Makefile deleted file mode 100644 index 8e8640f..0000000 --- a/node_modules/express/node_modules/connect/node_modules/bytes/Makefile +++ /dev/null @@ -1,7 +0,0 @@ - -test: - @./node_modules/.bin/mocha \ - --reporter spec \ - --require should - -.PHONY: test \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/bytes/Readme.md b/node_modules/express/node_modules/connect/node_modules/bytes/Readme.md deleted file mode 100644 index 9325d5b..0000000 --- a/node_modules/express/node_modules/connect/node_modules/bytes/Readme.md +++ /dev/null @@ -1,51 +0,0 @@ -# node-bytes - - Byte string parser / formatter. - -## Example: - -```js -bytes('1kb') -// => 1024 - -bytes('2mb') -// => 2097152 - -bytes('1gb') -// => 1073741824 - -bytes(1073741824) -// => 1gb -``` - -## Installation - -``` -$ npm install bytes -$ component install visionmedia/bytes.js -``` - -## License - -(The MIT License) - -Copyright (c) 2012 TJ Holowaychuk <tj@vision-media.ca> - -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/node_modules/express/node_modules/connect/node_modules/bytes/component.json b/node_modules/express/node_modules/connect/node_modules/bytes/component.json deleted file mode 100644 index 76a6057..0000000 --- a/node_modules/express/node_modules/connect/node_modules/bytes/component.json +++ /dev/null @@ -1,7 +0,0 @@ -{ - "name": "bytes", - "description": "byte size string parser / serializer", - "keywords": ["bytes", "utility"], - "version": "0.1.0", - "scripts": ["index.js"] -} diff --git a/node_modules/express/node_modules/connect/node_modules/bytes/index.js b/node_modules/express/node_modules/connect/node_modules/bytes/index.js deleted file mode 100644 index 70b2e01..0000000 --- a/node_modules/express/node_modules/connect/node_modules/bytes/index.js +++ /dev/null @@ -1,39 +0,0 @@ - -/** - * Parse byte `size` string. - * - * @param {String} size - * @return {Number} - * @api public - */ - -module.exports = function(size) { - if ('number' == typeof size) return convert(size); - var parts = size.match(/^(\d+(?:\.\d+)?) *(kb|mb|gb)$/) - , n = parseFloat(parts[1]) - , type = parts[2]; - - var map = { - kb: 1 << 10 - , mb: 1 << 20 - , gb: 1 << 30 - }; - - return map[type] * n; -}; - -/** - * convert bytes into string. - * - * @param {Number} b - bytes to convert - * @return {String} - * @api public - */ - -function convert (b) { - var gb = 1 << 30, mb = 1 << 20, kb = 1 << 10; - if (b >= gb) return (Math.round(b / gb * 100) / 100) + 'gb'; - if (b >= mb) return (Math.round(b / mb * 100) / 100) + 'mb'; - if (b >= kb) return (Math.round(b / kb * 100) / 100) + 'kb'; - return b + 'b'; -} diff --git a/node_modules/express/node_modules/connect/node_modules/bytes/package.json b/node_modules/express/node_modules/connect/node_modules/bytes/package.json deleted file mode 100644 index 0db9bc9..0000000 --- a/node_modules/express/node_modules/connect/node_modules/bytes/package.json +++ /dev/null @@ -1,20 +0,0 @@ -{ - "name": "bytes", - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca", - "url": "http://tjholowaychuk.com" - }, - "description": "byte size string parser / serializer", - "version": "0.2.0", - "main": "index.js", - "dependencies": {}, - "devDependencies": { - "mocha": "*", - "should": "*" - }, - "readme": "# node-bytes\n\n Byte string parser / formatter.\n\n## Example:\n\n```js\nbytes('1kb')\n// => 1024\n\nbytes('2mb')\n// => 2097152\n\nbytes('1gb')\n// => 1073741824\n\nbytes(1073741824)\n// => 1gb\n```\n\n## Installation\n\n```\n$ npm install bytes\n$ component install visionmedia/bytes.js\n```\n\n## License \n\n(The MIT License)\n\nCopyright (c) 2012 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n", - "readmeFilename": "Readme.md", - "_id": "bytes@0.2.0", - "_from": "bytes@0.2.0" -} diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/.npmignore b/node_modules/express/node_modules/connect/node_modules/formidable/.npmignore deleted file mode 100644 index 4fbabb3..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/.npmignore +++ /dev/null @@ -1,4 +0,0 @@ -/test/tmp/ -*.upload -*.un~ -*.http diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/.travis.yml b/node_modules/express/node_modules/connect/node_modules/formidable/.travis.yml deleted file mode 100644 index f1d0f13..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/.travis.yml +++ /dev/null @@ -1,4 +0,0 @@ -language: node_js -node_js: - - 0.4 - - 0.6 diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/Makefile b/node_modules/express/node_modules/connect/node_modules/formidable/Makefile deleted file mode 100644 index 8945872..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/Makefile +++ /dev/null @@ -1,14 +0,0 @@ -SHELL := /bin/bash - -test: - @./test/run.js - -build: npm test - -npm: - npm install . - -clean: - rm test/tmp/* - -.PHONY: test clean build diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/Readme.md b/node_modules/express/node_modules/connect/node_modules/formidable/Readme.md deleted file mode 100644 index a5ca104..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/Readme.md +++ /dev/null @@ -1,311 +0,0 @@ -# Formidable - -[![Build Status](https://secure.travis-ci.org/felixge/node-formidable.png?branch=master)](http://travis-ci.org/felixge/node-formidable) - -## Purpose - -A node.js module for parsing form data, especially file uploads. - -## Current status - -This module was developed for [Transloadit](http://transloadit.com/), a service focused on uploading -and encoding images and videos. It has been battle-tested against hundreds of GB of file uploads from -a large variety of clients and is considered production-ready. - -## Features - -* Fast (~500mb/sec), non-buffering multipart parser -* Automatically writing file uploads to disk -* Low memory footprint -* Graceful error handling -* Very high test coverage - -## Changelog - -### v1.0.9 - -* Emit progress when content length header parsed (Tim Koschützki) -* Fix Readme syntax due to GitHub changes (goob) -* Replace references to old 'sys' module in Readme with 'util' (Peter Sugihara) - -### v1.0.8 - -* Strip potentially unsafe characters when using `keepExtensions: true`. -* Switch to utest / urun for testing -* Add travis build - -### v1.0.7 - -* Remove file from package that was causing problems when installing on windows. (#102) -* Fix typos in Readme (Jason Davies). - -### v1.0.6 - -* Do not default to the default to the field name for file uploads where - filename="". - -### v1.0.5 - -* Support filename="" in multipart parts -* Explain unexpected end() errors in parser better - -**Note:** Starting with this version, formidable emits 'file' events for empty -file input fields. Previously those were incorrectly emitted as regular file -input fields with value = "". - -### v1.0.4 - -* Detect a good default tmp directory regardless of platform. (#88) - -### v1.0.3 - -* Fix problems with utf8 characters (#84) / semicolons in filenames (#58) -* Small performance improvements -* New test suite and fixture system - -### v1.0.2 - -* Exclude node\_modules folder from git -* Implement new `'aborted'` event -* Fix files in example folder to work with recent node versions -* Make gently a devDependency - -[See Commits](https://github.com/felixge/node-formidable/compare/v1.0.1...v1.0.2) - -### v1.0.1 - -* Fix package.json to refer to proper main directory. (#68, Dean Landolt) - -[See Commits](https://github.com/felixge/node-formidable/compare/v1.0.0...v1.0.1) - -### v1.0.0 - -* Add support for multipart boundaries that are quoted strings. (Jeff Craig) - -This marks the beginning of development on version 2.0 which will include -several architectural improvements. - -[See Commits](https://github.com/felixge/node-formidable/compare/v0.9.11...v1.0.0) - -### v0.9.11 - -* Emit `'progress'` event when receiving data, regardless of parsing it. (Tim Koschützki) -* Use [W3C FileAPI Draft](http://dev.w3.org/2006/webapi/FileAPI/) properties for File class - -**Important:** The old property names of the File class will be removed in a -future release. - -[See Commits](https://github.com/felixge/node-formidable/compare/v0.9.10...v0.9.11) - -### Older releases - -These releases were done before starting to maintain the above Changelog: - -* [v0.9.10](https://github.com/felixge/node-formidable/compare/v0.9.9...v0.9.10) -* [v0.9.9](https://github.com/felixge/node-formidable/compare/v0.9.8...v0.9.9) -* [v0.9.8](https://github.com/felixge/node-formidable/compare/v0.9.7...v0.9.8) -* [v0.9.7](https://github.com/felixge/node-formidable/compare/v0.9.6...v0.9.7) -* [v0.9.6](https://github.com/felixge/node-formidable/compare/v0.9.5...v0.9.6) -* [v0.9.5](https://github.com/felixge/node-formidable/compare/v0.9.4...v0.9.5) -* [v0.9.4](https://github.com/felixge/node-formidable/compare/v0.9.3...v0.9.4) -* [v0.9.3](https://github.com/felixge/node-formidable/compare/v0.9.2...v0.9.3) -* [v0.9.2](https://github.com/felixge/node-formidable/compare/v0.9.1...v0.9.2) -* [v0.9.1](https://github.com/felixge/node-formidable/compare/v0.9.0...v0.9.1) -* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0) -* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0) -* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0) -* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0) -* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0) -* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0) -* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0) -* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0) -* [v0.1.0](https://github.com/felixge/node-formidable/commits/v0.1.0) - -## Installation - -Via [npm](http://github.com/isaacs/npm): - - npm install formidable@latest - -Manually: - - git clone git://github.com/felixge/node-formidable.git formidable - vim my.js - # var formidable = require('./formidable'); - -Note: Formidable requires [gently](http://github.com/felixge/node-gently) to run the unit tests, but you won't need it for just using the library. - -## Example - -Parse an incoming file upload. - - var formidable = require('formidable'), - http = require('http'), - - util = require('util'); - - http.createServer(function(req, res) { - if (req.url == '/upload' && req.method.toLowerCase() == 'post') { - // parse a file upload - var form = new formidable.IncomingForm(); - form.parse(req, function(err, fields, files) { - res.writeHead(200, {'content-type': 'text/plain'}); - res.write('received upload:\n\n'); - res.end(util.inspect({fields: fields, files: files})); - }); - return; - } - - // show a file upload form - res.writeHead(200, {'content-type': 'text/html'}); - res.end( - '
    '+ - '
    '+ - '
    '+ - ''+ - '
    ' - ); - }).listen(80); - -## API - -### formidable.IncomingForm - -__new formidable.IncomingForm()__ - -Creates a new incoming form. - -__incomingForm.encoding = 'utf-8'__ - -The encoding to use for incoming form fields. - -__incomingForm.uploadDir = process.env.TMP || '/tmp' || process.cwd()__ - -The directory for placing file uploads in. You can move them later on using -`fs.rename()`. The default directory is picked at module load time depending on -the first existing directory from those listed above. - -__incomingForm.keepExtensions = false__ - -If you want the files written to `incomingForm.uploadDir` to include the extensions of the original files, set this property to `true`. - -__incomingForm.type__ - -Either 'multipart' or 'urlencoded' depending on the incoming request. - -__incomingForm.maxFieldsSize = 2 * 1024 * 1024__ - -Limits the amount of memory a field (not file) can allocate in bytes. -If this value is exceeded, an `'error'` event is emitted. The default -size is 2MB. - -__incomingForm.hash = false__ - -If you want checksums calculated for incoming files, set this to either `'sha1'` or `'md5'`. - -__incomingForm.bytesReceived__ - -The amount of bytes received for this form so far. - -__incomingForm.bytesExpected__ - -The expected number of bytes in this form. - -__incomingForm.parse(request, [cb])__ - -Parses an incoming node.js `request` containing form data. If `cb` is provided, all fields an files are collected and passed to the callback: - - incomingForm.parse(req, function(err, fields, files) { - // ... - }); - -__incomingForm.onPart(part)__ - -You may overwrite this method if you are interested in directly accessing the multipart stream. Doing so will disable any `'field'` / `'file'` events processing which would occur otherwise, making you fully responsible for handling the processing. - - incomingForm.onPart = function(part) { - part.addListener('data', function() { - // ... - }); - } - -If you want to use formidable to only handle certain parts for you, you can do so: - - incomingForm.onPart = function(part) { - if (!part.filename) { - // let formidable handle all non-file parts - incomingForm.handlePart(part); - } - } - -Check the code in this method for further inspiration. - -__Event: 'progress' (bytesReceived, bytesExpected)__ - -Emitted after each incoming chunk of data that has been parsed. Can be used to roll your own progress bar. - -__Event: 'field' (name, value)__ - -Emitted whenever a field / value pair has been received. - -__Event: 'fileBegin' (name, file)__ - -Emitted whenever a new file is detected in the upload stream. Use this even if -you want to stream the file to somewhere else while buffering the upload on -the file system. - -__Event: 'file' (name, file)__ - -Emitted whenever a field / file pair has been received. `file` is an instance of `File`. - -__Event: 'error' (err)__ - -Emitted when there is an error processing the incoming form. A request that experiences an error is automatically paused, you will have to manually call `request.resume()` if you want the request to continue firing `'data'` events. - -__Event: 'aborted'__ - -Emitted when the request was aborted by the user. Right now this can be due to a 'timeout' or 'close' event on the socket. In the future there will be a separate 'timeout' event (needs a change in the node core). - -__Event: 'end' ()__ - -Emitted when the entire request has been received, and all contained files have finished flushing to disk. This is a great place for you to send your response. - -### formidable.File - -__file.size = 0__ - -The size of the uploaded file in bytes. If the file is still being uploaded (see `'fileBegin'` event), this property says how many bytes of the file have been written to disk yet. - -__file.path = null__ - -The path this file is being written to. You can modify this in the `'fileBegin'` event in -case you are unhappy with the way formidable generates a temporary path for your files. - -__file.name = null__ - -The name this file had according to the uploading client. - -__file.type = null__ - -The mime type of this file, according to the uploading client. - -__file.lastModifiedDate = null__ - -A date object (or `null`) containing the time this file was last written to. Mostly -here for compatibility with the [W3C File API Draft](http://dev.w3.org/2006/webapi/FileAPI/). - -__file.hash = null__ - -If hash calculation was set, you can read the hex digest out of this var. - -## License - -Formidable is licensed under the MIT license. - -## Ports - -* [multipart-parser](http://github.com/FooBarWidget/multipart-parser): a C++ parser based on formidable - -## Credits - -* [Ryan Dahl](http://twitter.com/ryah) for his work on [http-parser](http://github.com/ry/http-parser) which heavily inspired multipart_parser.js diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/TODO b/node_modules/express/node_modules/connect/node_modules/formidable/TODO deleted file mode 100644 index e1107f2..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/TODO +++ /dev/null @@ -1,3 +0,0 @@ -- Better bufferMaxSize handling approach -- Add tests for JSON parser pull request and merge it -- Implement QuerystringParser the same way as MultipartParser diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/benchmark/bench-multipart-parser.js b/node_modules/express/node_modules/connect/node_modules/formidable/benchmark/bench-multipart-parser.js deleted file mode 100644 index bff41f1..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/benchmark/bench-multipart-parser.js +++ /dev/null @@ -1,70 +0,0 @@ -require('../test/common'); -var multipartParser = require('../lib/multipart_parser'), - MultipartParser = multipartParser.MultipartParser, - parser = new MultipartParser(), - Buffer = require('buffer').Buffer, - boundary = '-----------------------------168072824752491622650073', - mb = 100, - buffer = createMultipartBuffer(boundary, mb * 1024 * 1024), - callbacks = - { partBegin: -1, - partEnd: -1, - headerField: -1, - headerValue: -1, - partData: -1, - end: -1, - }; - - -parser.initWithBoundary(boundary); -parser.onHeaderField = function() { - callbacks.headerField++; -}; - -parser.onHeaderValue = function() { - callbacks.headerValue++; -}; - -parser.onPartBegin = function() { - callbacks.partBegin++; -}; - -parser.onPartData = function() { - callbacks.partData++; -}; - -parser.onPartEnd = function() { - callbacks.partEnd++; -}; - -parser.onEnd = function() { - callbacks.end++; -}; - -var start = +new Date(), - nparsed = parser.write(buffer), - duration = +new Date - start, - mbPerSec = (mb / (duration / 1000)).toFixed(2); - -console.log(mbPerSec+' mb/sec'); - -assert.equal(nparsed, buffer.length); - -function createMultipartBuffer(boundary, size) { - var head = - '--'+boundary+'\r\n' - + 'content-disposition: form-data; name="field1"\r\n' - + '\r\n' - , tail = '\r\n--'+boundary+'--\r\n' - , buffer = new Buffer(size); - - buffer.write(head, 'ascii', 0); - buffer.write(tail, 'ascii', buffer.length - tail.length); - return buffer; -} - -process.on('exit', function() { - for (var k in callbacks) { - assert.equal(0, callbacks[k], k+' count off by '+callbacks[k]); - } -}); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/example/post.js b/node_modules/express/node_modules/connect/node_modules/formidable/example/post.js deleted file mode 100644 index f6c15a6..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/example/post.js +++ /dev/null @@ -1,43 +0,0 @@ -require('../test/common'); -var http = require('http'), - util = require('util'), - formidable = require('formidable'), - server; - -server = http.createServer(function(req, res) { - if (req.url == '/') { - res.writeHead(200, {'content-type': 'text/html'}); - res.end( - '
    '+ - '
    '+ - '
    '+ - ''+ - '
    ' - ); - } else if (req.url == '/post') { - var form = new formidable.IncomingForm(), - fields = []; - - form - .on('error', function(err) { - res.writeHead(200, {'content-type': 'text/plain'}); - res.end('error:\n\n'+util.inspect(err)); - }) - .on('field', function(field, value) { - console.log(field, value); - fields.push([field, value]); - }) - .on('end', function() { - console.log('-> post done'); - res.writeHead(200, {'content-type': 'text/plain'}); - res.end('received fields:\n\n '+util.inspect(fields)); - }); - form.parse(req); - } else { - res.writeHead(404, {'content-type': 'text/plain'}); - res.end('404'); - } -}); -server.listen(TEST_PORT); - -console.log('listening on http://localhost:'+TEST_PORT+'/'); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/example/upload.js b/node_modules/express/node_modules/connect/node_modules/formidable/example/upload.js deleted file mode 100644 index 050cdd9..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/example/upload.js +++ /dev/null @@ -1,48 +0,0 @@ -require('../test/common'); -var http = require('http'), - util = require('util'), - formidable = require('formidable'), - server; - -server = http.createServer(function(req, res) { - if (req.url == '/') { - res.writeHead(200, {'content-type': 'text/html'}); - res.end( - '
    '+ - '
    '+ - '
    '+ - ''+ - '
    ' - ); - } else if (req.url == '/upload') { - var form = new formidable.IncomingForm(), - files = [], - fields = []; - - form.uploadDir = TEST_TMP; - - form - .on('field', function(field, value) { - console.log(field, value); - fields.push([field, value]); - }) - .on('file', function(field, file) { - console.log(field, file); - files.push([field, file]); - }) - .on('end', function() { - console.log('-> upload done'); - res.writeHead(200, {'content-type': 'text/plain'}); - res.write('received fields:\n\n '+util.inspect(fields)); - res.write('\n\n'); - res.end('received files:\n\n '+util.inspect(files)); - }); - form.parse(req); - } else { - res.writeHead(404, {'content-type': 'text/plain'}); - res.end('404'); - } -}); -server.listen(TEST_PORT); - -console.log('listening on http://localhost:'+TEST_PORT+'/'); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/index.js b/node_modules/express/node_modules/connect/node_modules/formidable/index.js deleted file mode 100644 index be41032..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/index.js +++ /dev/null @@ -1 +0,0 @@ -module.exports = require('./lib/formidable'); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/lib/file.js b/node_modules/express/node_modules/connect/node_modules/formidable/lib/file.js deleted file mode 100644 index dad8d5f..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/lib/file.js +++ /dev/null @@ -1,73 +0,0 @@ -if (global.GENTLY) require = GENTLY.hijack(require); - -var util = require('./util'), - WriteStream = require('fs').WriteStream, - EventEmitter = require('events').EventEmitter, - crypto = require('crypto'); - -function File(properties) { - EventEmitter.call(this); - - this.size = 0; - this.path = null; - this.name = null; - this.type = null; - this.hash = null; - this.lastModifiedDate = null; - - this._writeStream = null; - - for (var key in properties) { - this[key] = properties[key]; - } - - if(typeof this.hash === 'string') { - this.hash = crypto.createHash(properties.hash); - } - - this._backwardsCompatibility(); -} -module.exports = File; -util.inherits(File, EventEmitter); - -// @todo Next release: Show error messages when accessing these -File.prototype._backwardsCompatibility = function() { - var self = this; - this.__defineGetter__('length', function() { - return self.size; - }); - this.__defineGetter__('filename', function() { - return self.name; - }); - this.__defineGetter__('mime', function() { - return self.type; - }); -}; - -File.prototype.open = function() { - this._writeStream = new WriteStream(this.path); -}; - -File.prototype.write = function(buffer, cb) { - var self = this; - this._writeStream.write(buffer, function() { - if(self.hash) { - self.hash.update(buffer); - } - self.lastModifiedDate = new Date(); - self.size += buffer.length; - self.emit('progress', self.size); - cb(); - }); -}; - -File.prototype.end = function(cb) { - var self = this; - this._writeStream.end(function() { - if(self.hash) { - self.hash = self.hash.digest('hex'); - } - self.emit('end'); - cb(); - }); -}; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/lib/incoming_form.js b/node_modules/express/node_modules/connect/node_modules/formidable/lib/incoming_form.js deleted file mode 100644 index 060eac2..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/lib/incoming_form.js +++ /dev/null @@ -1,384 +0,0 @@ -if (global.GENTLY) require = GENTLY.hijack(require); - -var fs = require('fs'); -var util = require('./util'), - path = require('path'), - File = require('./file'), - MultipartParser = require('./multipart_parser').MultipartParser, - QuerystringParser = require('./querystring_parser').QuerystringParser, - StringDecoder = require('string_decoder').StringDecoder, - EventEmitter = require('events').EventEmitter, - Stream = require('stream').Stream; - -function IncomingForm(opts) { - if (!(this instanceof IncomingForm)) return new IncomingForm; - EventEmitter.call(this); - - opts=opts||{}; - - this.error = null; - this.ended = false; - - this.maxFieldsSize = opts.maxFieldsSize || 2 * 1024 * 1024; - this.keepExtensions = opts.keepExtensions || false; - this.uploadDir = opts.uploadDir || IncomingForm.UPLOAD_DIR; - this.encoding = opts.encoding || 'utf-8'; - this.headers = null; - this.type = null; - this.hash = false; - - this.bytesReceived = null; - this.bytesExpected = null; - - this._parser = null; - this._flushing = 0; - this._fieldsSize = 0; -}; -util.inherits(IncomingForm, EventEmitter); -exports.IncomingForm = IncomingForm; - -IncomingForm.UPLOAD_DIR = (function() { - var dirs = [process.env.TMP, '/tmp', process.cwd()]; - for (var i = 0; i < dirs.length; i++) { - var dir = dirs[i]; - var isDirectory = false; - - try { - isDirectory = fs.statSync(dir).isDirectory(); - } catch (e) {} - - if (isDirectory) return dir; - } -})(); - -IncomingForm.prototype.parse = function(req, cb) { - this.pause = function() { - try { - req.pause(); - } catch (err) { - // the stream was destroyed - if (!this.ended) { - // before it was completed, crash & burn - this._error(err); - } - return false; - } - return true; - }; - - this.resume = function() { - try { - req.resume(); - } catch (err) { - // the stream was destroyed - if (!this.ended) { - // before it was completed, crash & burn - this._error(err); - } - return false; - } - - return true; - }; - - this.writeHeaders(req.headers); - - var self = this; - req - .on('error', function(err) { - self._error(err); - }) - .on('aborted', function() { - self.emit('aborted'); - }) - .on('data', function(buffer) { - self.write(buffer); - }) - .on('end', function() { - if (self.error) { - return; - } - - var err = self._parser.end(); - if (err) { - self._error(err); - } - }); - - if (cb) { - var fields = {}, files = {}; - this - .on('field', function(name, value) { - fields[name] = value; - }) - .on('file', function(name, file) { - files[name] = file; - }) - .on('error', function(err) { - cb(err, fields, files); - }) - .on('end', function() { - cb(null, fields, files); - }); - } - - return this; -}; - -IncomingForm.prototype.writeHeaders = function(headers) { - this.headers = headers; - this._parseContentLength(); - this._parseContentType(); -}; - -IncomingForm.prototype.write = function(buffer) { - if (!this._parser) { - this._error(new Error('unintialized parser')); - return; - } - - this.bytesReceived += buffer.length; - this.emit('progress', this.bytesReceived, this.bytesExpected); - - var bytesParsed = this._parser.write(buffer); - if (bytesParsed !== buffer.length) { - this._error(new Error('parser error, '+bytesParsed+' of '+buffer.length+' bytes parsed')); - } - - return bytesParsed; -}; - -IncomingForm.prototype.pause = function() { - // this does nothing, unless overwritten in IncomingForm.parse - return false; -}; - -IncomingForm.prototype.resume = function() { - // this does nothing, unless overwritten in IncomingForm.parse - return false; -}; - -IncomingForm.prototype.onPart = function(part) { - // this method can be overwritten by the user - this.handlePart(part); -}; - -IncomingForm.prototype.handlePart = function(part) { - var self = this; - - if (part.filename === undefined) { - var value = '' - , decoder = new StringDecoder(this.encoding); - - part.on('data', function(buffer) { - self._fieldsSize += buffer.length; - if (self._fieldsSize > self.maxFieldsSize) { - self._error(new Error('maxFieldsSize exceeded, received '+self._fieldsSize+' bytes of field data')); - return; - } - value += decoder.write(buffer); - }); - - part.on('end', function() { - self.emit('field', part.name, value); - }); - return; - } - - this._flushing++; - - var file = new File({ - path: this._uploadPath(part.filename), - name: part.filename, - type: part.mime, - hash: self.hash - }); - - this.emit('fileBegin', part.name, file); - - file.open(); - - part.on('data', function(buffer) { - self.pause(); - file.write(buffer, function() { - self.resume(); - }); - }); - - part.on('end', function() { - file.end(function() { - self._flushing--; - self.emit('file', part.name, file); - self._maybeEnd(); - }); - }); -}; - -IncomingForm.prototype._parseContentType = function() { - if (!this.headers['content-type']) { - this._error(new Error('bad content-type header, no content-type')); - return; - } - - if (this.headers['content-type'].match(/urlencoded/i)) { - this._initUrlencoded(); - return; - } - - if (this.headers['content-type'].match(/multipart/i)) { - var m; - if (m = this.headers['content-type'].match(/boundary=(?:"([^"]+)"|([^;]+))/i)) { - this._initMultipart(m[1] || m[2]); - } else { - this._error(new Error('bad content-type header, no multipart boundary')); - } - return; - } - - this._error(new Error('bad content-type header, unknown content-type: '+this.headers['content-type'])); -}; - -IncomingForm.prototype._error = function(err) { - if (this.error) { - return; - } - - this.error = err; - this.pause(); - this.emit('error', err); -}; - -IncomingForm.prototype._parseContentLength = function() { - if (this.headers['content-length']) { - this.bytesReceived = 0; - this.bytesExpected = parseInt(this.headers['content-length'], 10); - this.emit('progress', this.bytesReceived, this.bytesExpected); - } -}; - -IncomingForm.prototype._newParser = function() { - return new MultipartParser(); -}; - -IncomingForm.prototype._initMultipart = function(boundary) { - this.type = 'multipart'; - - var parser = new MultipartParser(), - self = this, - headerField, - headerValue, - part; - - parser.initWithBoundary(boundary); - - parser.onPartBegin = function() { - part = new Stream(); - part.readable = true; - part.headers = {}; - part.name = null; - part.filename = null; - part.mime = null; - headerField = ''; - headerValue = ''; - }; - - parser.onHeaderField = function(b, start, end) { - headerField += b.toString(self.encoding, start, end); - }; - - parser.onHeaderValue = function(b, start, end) { - headerValue += b.toString(self.encoding, start, end); - }; - - parser.onHeaderEnd = function() { - headerField = headerField.toLowerCase(); - part.headers[headerField] = headerValue; - - var m; - if (headerField == 'content-disposition') { - if (m = headerValue.match(/name="([^"]+)"/i)) { - part.name = m[1]; - } - - part.filename = self._fileName(headerValue); - } else if (headerField == 'content-type') { - part.mime = headerValue; - } - - headerField = ''; - headerValue = ''; - }; - - parser.onHeadersEnd = function() { - self.onPart(part); - }; - - parser.onPartData = function(b, start, end) { - part.emit('data', b.slice(start, end)); - }; - - parser.onPartEnd = function() { - part.emit('end'); - }; - - parser.onEnd = function() { - self.ended = true; - self._maybeEnd(); - }; - - this._parser = parser; -}; - -IncomingForm.prototype._fileName = function(headerValue) { - var m = headerValue.match(/filename="(.*?)"($|; )/i) - if (!m) return; - - var filename = m[1].substr(m[1].lastIndexOf('\\') + 1); - filename = filename.replace(/%22/g, '"'); - filename = filename.replace(/&#([\d]{4});/g, function(m, code) { - return String.fromCharCode(code); - }); - return filename; -}; - -IncomingForm.prototype._initUrlencoded = function() { - this.type = 'urlencoded'; - - var parser = new QuerystringParser() - , self = this; - - parser.onField = function(key, val) { - self.emit('field', key, val); - }; - - parser.onEnd = function() { - self.ended = true; - self._maybeEnd(); - }; - - this._parser = parser; -}; - -IncomingForm.prototype._uploadPath = function(filename) { - var name = ''; - for (var i = 0; i < 32; i++) { - name += Math.floor(Math.random() * 16).toString(16); - } - - if (this.keepExtensions) { - var ext = path.extname(filename); - ext = ext.replace(/(\.[a-z0-9]+).*/, '$1') - - name += ext; - } - - return path.join(this.uploadDir, name); -}; - -IncomingForm.prototype._maybeEnd = function() { - if (!this.ended || this._flushing) { - return; - } - - this.emit('end'); -}; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/lib/index.js b/node_modules/express/node_modules/connect/node_modules/formidable/lib/index.js deleted file mode 100644 index 7a6e3e1..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/lib/index.js +++ /dev/null @@ -1,3 +0,0 @@ -var IncomingForm = require('./incoming_form').IncomingForm; -IncomingForm.IncomingForm = IncomingForm; -module.exports = IncomingForm; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/lib/multipart_parser.js b/node_modules/express/node_modules/connect/node_modules/formidable/lib/multipart_parser.js deleted file mode 100644 index 9ca567c..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/lib/multipart_parser.js +++ /dev/null @@ -1,312 +0,0 @@ -var Buffer = require('buffer').Buffer, - s = 0, - S = - { PARSER_UNINITIALIZED: s++, - START: s++, - START_BOUNDARY: s++, - HEADER_FIELD_START: s++, - HEADER_FIELD: s++, - HEADER_VALUE_START: s++, - HEADER_VALUE: s++, - HEADER_VALUE_ALMOST_DONE: s++, - HEADERS_ALMOST_DONE: s++, - PART_DATA_START: s++, - PART_DATA: s++, - PART_END: s++, - END: s++, - }, - - f = 1, - F = - { PART_BOUNDARY: f, - LAST_BOUNDARY: f *= 2, - }, - - LF = 10, - CR = 13, - SPACE = 32, - HYPHEN = 45, - COLON = 58, - A = 97, - Z = 122, - - lower = function(c) { - return c | 0x20; - }; - -for (var s in S) { - exports[s] = S[s]; -} - -function MultipartParser() { - this.boundary = null; - this.boundaryChars = null; - this.lookbehind = null; - this.state = S.PARSER_UNINITIALIZED; - - this.index = null; - this.flags = 0; -}; -exports.MultipartParser = MultipartParser; - -MultipartParser.stateToString = function(stateNumber) { - for (var state in S) { - var number = S[state]; - if (number === stateNumber) return state; - } -}; - -MultipartParser.prototype.initWithBoundary = function(str) { - this.boundary = new Buffer(str.length+4); - this.boundary.write('\r\n--', 'ascii', 0); - this.boundary.write(str, 'ascii', 4); - this.lookbehind = new Buffer(this.boundary.length+8); - this.state = S.START; - - this.boundaryChars = {}; - for (var i = 0; i < this.boundary.length; i++) { - this.boundaryChars[this.boundary[i]] = true; - } -}; - -MultipartParser.prototype.write = function(buffer) { - var self = this, - i = 0, - len = buffer.length, - prevIndex = this.index, - index = this.index, - state = this.state, - flags = this.flags, - lookbehind = this.lookbehind, - boundary = this.boundary, - boundaryChars = this.boundaryChars, - boundaryLength = this.boundary.length, - boundaryEnd = boundaryLength - 1, - bufferLength = buffer.length, - c, - cl, - - mark = function(name) { - self[name+'Mark'] = i; - }, - clear = function(name) { - delete self[name+'Mark']; - }, - callback = function(name, buffer, start, end) { - if (start !== undefined && start === end) { - return; - } - - var callbackSymbol = 'on'+name.substr(0, 1).toUpperCase()+name.substr(1); - if (callbackSymbol in self) { - self[callbackSymbol](buffer, start, end); - } - }, - dataCallback = function(name, clear) { - var markSymbol = name+'Mark'; - if (!(markSymbol in self)) { - return; - } - - if (!clear) { - callback(name, buffer, self[markSymbol], buffer.length); - self[markSymbol] = 0; - } else { - callback(name, buffer, self[markSymbol], i); - delete self[markSymbol]; - } - }; - - for (i = 0; i < len; i++) { - c = buffer[i]; - switch (state) { - case S.PARSER_UNINITIALIZED: - return i; - case S.START: - index = 0; - state = S.START_BOUNDARY; - case S.START_BOUNDARY: - if (index == boundary.length - 2) { - if (c != CR) { - return i; - } - index++; - break; - } else if (index - 1 == boundary.length - 2) { - if (c != LF) { - return i; - } - index = 0; - callback('partBegin'); - state = S.HEADER_FIELD_START; - break; - } - - if (c != boundary[index+2]) { - return i; - } - index++; - break; - case S.HEADER_FIELD_START: - state = S.HEADER_FIELD; - mark('headerField'); - index = 0; - case S.HEADER_FIELD: - if (c == CR) { - clear('headerField'); - state = S.HEADERS_ALMOST_DONE; - break; - } - - index++; - if (c == HYPHEN) { - break; - } - - if (c == COLON) { - if (index == 1) { - // empty header field - return i; - } - dataCallback('headerField', true); - state = S.HEADER_VALUE_START; - break; - } - - cl = lower(c); - if (cl < A || cl > Z) { - return i; - } - break; - case S.HEADER_VALUE_START: - if (c == SPACE) { - break; - } - - mark('headerValue'); - state = S.HEADER_VALUE; - case S.HEADER_VALUE: - if (c == CR) { - dataCallback('headerValue', true); - callback('headerEnd'); - state = S.HEADER_VALUE_ALMOST_DONE; - } - break; - case S.HEADER_VALUE_ALMOST_DONE: - if (c != LF) { - return i; - } - state = S.HEADER_FIELD_START; - break; - case S.HEADERS_ALMOST_DONE: - if (c != LF) { - return i; - } - - callback('headersEnd'); - state = S.PART_DATA_START; - break; - case S.PART_DATA_START: - state = S.PART_DATA - mark('partData'); - case S.PART_DATA: - prevIndex = index; - - if (index == 0) { - // boyer-moore derrived algorithm to safely skip non-boundary data - i += boundaryEnd; - while (i < bufferLength && !(buffer[i] in boundaryChars)) { - i += boundaryLength; - } - i -= boundaryEnd; - c = buffer[i]; - } - - if (index < boundary.length) { - if (boundary[index] == c) { - if (index == 0) { - dataCallback('partData', true); - } - index++; - } else { - index = 0; - } - } else if (index == boundary.length) { - index++; - if (c == CR) { - // CR = part boundary - flags |= F.PART_BOUNDARY; - } else if (c == HYPHEN) { - // HYPHEN = end boundary - flags |= F.LAST_BOUNDARY; - } else { - index = 0; - } - } else if (index - 1 == boundary.length) { - if (flags & F.PART_BOUNDARY) { - index = 0; - if (c == LF) { - // unset the PART_BOUNDARY flag - flags &= ~F.PART_BOUNDARY; - callback('partEnd'); - callback('partBegin'); - state = S.HEADER_FIELD_START; - break; - } - } else if (flags & F.LAST_BOUNDARY) { - if (c == HYPHEN) { - callback('partEnd'); - callback('end'); - state = S.END; - } else { - index = 0; - } - } else { - index = 0; - } - } - - if (index > 0) { - // when matching a possible boundary, keep a lookbehind reference - // in case it turns out to be a false lead - lookbehind[index-1] = c; - } else if (prevIndex > 0) { - // if our boundary turned out to be rubbish, the captured lookbehind - // belongs to partData - callback('partData', lookbehind, 0, prevIndex); - prevIndex = 0; - mark('partData'); - - // reconsider the current character even so it interrupted the sequence - // it could be the beginning of a new sequence - i--; - } - - break; - case S.END: - break; - default: - return i; - } - } - - dataCallback('headerField'); - dataCallback('headerValue'); - dataCallback('partData'); - - this.index = index; - this.state = state; - this.flags = flags; - - return len; -}; - -MultipartParser.prototype.end = function() { - if (this.state != S.END) { - return new Error('MultipartParser.end(): stream ended unexpectedly: ' + this.explain()); - } -}; - -MultipartParser.prototype.explain = function() { - return 'state = ' + MultipartParser.stateToString(this.state); -}; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/lib/querystring_parser.js b/node_modules/express/node_modules/connect/node_modules/formidable/lib/querystring_parser.js deleted file mode 100644 index 63f109e..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/lib/querystring_parser.js +++ /dev/null @@ -1,25 +0,0 @@ -if (global.GENTLY) require = GENTLY.hijack(require); - -// This is a buffering parser, not quite as nice as the multipart one. -// If I find time I'll rewrite this to be fully streaming as well -var querystring = require('querystring'); - -function QuerystringParser() { - this.buffer = ''; -}; -exports.QuerystringParser = QuerystringParser; - -QuerystringParser.prototype.write = function(buffer) { - this.buffer += buffer.toString('ascii'); - return buffer.length; -}; - -QuerystringParser.prototype.end = function() { - var fields = querystring.parse(this.buffer); - for (var field in fields) { - this.onField(field, fields[field]); - } - this.buffer = ''; - - this.onEnd(); -}; \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/lib/util.js b/node_modules/express/node_modules/connect/node_modules/formidable/lib/util.js deleted file mode 100644 index e9493e9..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/lib/util.js +++ /dev/null @@ -1,6 +0,0 @@ -// Backwards compatibility ... -try { - module.exports = require('util'); -} catch (e) { - module.exports = require('sys'); -} diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/Makefile b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/Makefile deleted file mode 100644 index 01f7140..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/Makefile +++ /dev/null @@ -1,4 +0,0 @@ -test: - @find test/simple/test-*.js | xargs -n 1 -t node - -.PHONY: test \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/Readme.md b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/Readme.md deleted file mode 100644 index f8f0c66..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/Readme.md +++ /dev/null @@ -1,167 +0,0 @@ -# Gently - -## Purpose - -A node.js module that helps with stubbing and behavior verification. It allows you to test the most remote and nested corners of your code while keeping being fully unobtrusive. - -## Features - -* Overwrite and stub individual object functions -* Verify that all expected calls have been made in the expected order -* Restore stubbed functions to their original behavior -* Detect object / class names from obj.constructor.name and obj.toString() -* Hijack any required module function or class constructor - -## Installation - -Via [npm](http://github.com/isaacs/npm): - - npm install gently@latest - -## Example - -Make sure your dog is working properly: - - function Dog() {} - - Dog.prototype.seeCat = function() { - this.bark('whuf, whuf'); - this.run(); - } - - Dog.prototype.bark = function(bark) { - require('sys').puts(bark); - } - - var gently = new (require('gently')) - , assert = require('assert') - , dog = new Dog(); - - gently.expect(dog, 'bark', function(bark) { - assert.equal(bark, 'whuf, whuf'); - }); - gently.expect(dog, 'run'); - - dog.seeCat(); - -You can also easily test event emitters with this, for example a simple sequence of 2 events emitted by `fs.WriteStream`: - - var gently = new (require('gently')) - , stream = new (require('fs').WriteStream)('my_file.txt'); - - gently.expect(stream, 'emit', function(event) { - assert.equal(event, 'open'); - }); - - gently.expect(stream, 'emit', function(event) { - assert.equal(event, 'drain'); - }); - -For a full read world example, check out this test case: [test-incoming-form.js](http://github.com/felixge/node-formidable/blob/master/test/simple/test-incoming-form.js) (in [node-formdiable](http://github.com/felixge/node-formidable)). - -## API - -### Gently - -#### new Gently() - -Creates a new gently instance. It listens to the process `'exit'` event to make sure all expectations have been verified. - -#### gently.expect(obj, method, [[count], stubFn]) - -Creates an expectation for an objects method to be called. You can optionally specify the call `count` you are expecting, as well as `stubFn` function that will run instead of the original function. - -Returns a reference to the function that is getting overwritten. - -#### gently.expect([count], stubFn) - -Returns a function that is supposed to be executed `count` times, delegating any calls to the provided `stubFn` function. Naming your stubFn closure will help to properly diagnose errors that are being thrown: - - childProcess.exec('ls', gently.expect(function lsCallback(code) { - assert.equal(0, code); - })); - -#### gently.restore(obj, method) - -Restores an object method that has been previously overwritten using `gently.expect()`. - -#### gently.hijack(realRequire) - -Returns a new require functions that catches a reference to all required modules into `gently.hijacked`. - -To use this function, include a line like this in your `'my-module.js'`. - - if (global.GENTLY) require = GENTLY.hijack(require); - - var sys = require('sys'); - exports.hello = function() { - sys.log('world'); - }; - -Now you can write a test for the module above: - - var gently = global.GENTLY = new (require('gently')) - , myModule = require('./my-module'); - - gently.expect(gently.hijacked.sys, 'log', function(str) { - assert.equal(str, 'world'); - }); - - myModule.hello(); - -#### gently.stub(location, [exportsName]) - -Returns a stub class that will be used instead of the real class from the module at `location` with the given `exportsName`. - -This allows to test an OOP version of the previous example, where `'my-module.js'`. - - if (global.GENTLY) require = GENTLY.hijack(require); - - var World = require('./world'); - - exports.hello = function() { - var world = new World(); - world.hello(); - } - -And `world.js` looks like this: - - var sys = require('sys'); - - function World() { - - } - module.exports = World; - - World.prototype.hello = function() { - sys.log('world'); - }; - -Testing `'my-module.js'` can now easily be accomplished: - - var gently = global.GENTLY = new (require('gently')) - , WorldStub = gently.stub('./world') - , myModule = require('./my-module') - , WORLD; - - gently.expect(WorldStub, 'new', function() { - WORLD = this; - }); - - gently.expect(WORLD, 'hello'); - - myModule.hello(); - -#### gently.hijacked - -An object that holds the references to all hijacked modules. - -#### gently.verify([msg]) - -Verifies that all expectations of this gently instance have been satisfied. If not called manually, this method is called when the process `'exit'` event is fired. - -If `msg` is given, it will appear in any error that might be thrown. - -## License - -Gently is licensed under the MIT license. \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/example/dog.js b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/example/dog.js deleted file mode 100644 index 022fae0..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/example/dog.js +++ /dev/null @@ -1,22 +0,0 @@ -require('../test/common'); -function Dog() {} - -Dog.prototype.seeCat = function() { - this.bark('whuf, whuf'); - this.run(); -} - -Dog.prototype.bark = function(bark) { - require('sys').puts(bark); -} - -var gently = new (require('gently')) - , assert = require('assert') - , dog = new Dog(); - -gently.expect(dog, 'bark', function(bark) { - assert.equal(bark, 'whuf, whuf'); -}); -gently.expect(dog, 'run'); - -dog.seeCat(); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/example/event_emitter.js b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/example/event_emitter.js deleted file mode 100644 index 7def134..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/example/event_emitter.js +++ /dev/null @@ -1,11 +0,0 @@ -require('../test/common'); -var gently = new (require('gently')) - , stream = new (require('fs').WriteStream)('my_file.txt'); - -gently.expect(stream, 'emit', function(event) { - assert.equal(event, 'open'); -}); - -gently.expect(stream, 'emit', function(event) { - assert.equal(event, 'drain'); -}); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/index.js b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/index.js deleted file mode 100644 index 69122bd..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/index.js +++ /dev/null @@ -1 +0,0 @@ -module.exports = require('./lib/gently'); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/lib/gently/gently.js b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/lib/gently/gently.js deleted file mode 100644 index 8af0e1e..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/lib/gently/gently.js +++ /dev/null @@ -1,184 +0,0 @@ -var path = require('path'); - -function Gently() { - this.expectations = []; - this.hijacked = {}; - - var self = this; - process.addListener('exit', function() { - self.verify('process exit'); - }); -}; -module.exports = Gently; - -Gently.prototype.stub = function(location, exportsName) { - function Stub() { - return Stub['new'].apply(this, arguments); - }; - - Stub['new'] = function () {}; - - var stubName = 'require('+JSON.stringify(location)+')'; - if (exportsName) { - stubName += '.'+exportsName; - } - - Stub.prototype.toString = Stub.toString = function() { - return stubName; - }; - - var exports = this.hijacked[location] || {}; - if (exportsName) { - exports[exportsName] = Stub; - } else { - exports = Stub; - } - - this.hijacked[location] = exports; - return Stub; -}; - -Gently.prototype.hijack = function(realRequire) { - var self = this; - return function(location) { - return self.hijacked[location] = (self.hijacked[location]) - ? self.hijacked[location] - : realRequire(location); - }; -}; - -Gently.prototype.expect = function(obj, method, count, stubFn) { - if (typeof obj != 'function' && typeof obj != 'object' && typeof obj != 'number') { - throw new Error - ( 'Bad 1st argument for gently.expect(), ' - + 'object, function, or number expected, got: '+(typeof obj) - ); - } else if (typeof obj == 'function' && (typeof method != 'string')) { - // expect(stubFn) interface - stubFn = obj; - obj = null; - method = null; - count = 1; - } else if (typeof method == 'function') { - // expect(count, stubFn) interface - count = obj; - stubFn = method; - obj = null; - method = null; - } else if (typeof count == 'function') { - // expect(obj, method, stubFn) interface - stubFn = count; - count = 1; - } else if (count === undefined) { - // expect(obj, method) interface - count = 1; - } - - var name = this._name(obj, method, stubFn); - this.expectations.push({obj: obj, method: method, stubFn: stubFn, name: name, count: count}); - - var self = this; - function delegate() { - return self._stubFn(this, obj, method, name, Array.prototype.slice.call(arguments)); - } - - if (!obj) { - return delegate; - } - - var original = (obj[method]) - ? obj[method]._original || obj[method] - : undefined; - - obj[method] = delegate; - return obj[method]._original = original; -}; - -Gently.prototype.restore = function(obj, method) { - if (!obj[method] || !obj[method]._original) { - throw new Error(this._name(obj, method)+' is not gently stubbed'); - } - obj[method] = obj[method]._original; -}; - -Gently.prototype.verify = function(msg) { - if (!this.expectations.length) { - return; - } - - var validExpectations = []; - for (var i = 0, l = this.expectations.length; i < l; i++) { - var expectation = this.expectations[i]; - - if (expectation.count > 0) { - validExpectations.push(expectation); - } - } - - this.expectations = []; // reset so that no duplicate verification attempts are made - - if (!validExpectations.length) { - return; - } - - var expectation = validExpectations[0]; - - throw new Error - ( 'Expected call to '+expectation.name+' did not happen' - + ( (msg) - ? ' ('+msg+')' - : '' - ) - ); -}; - -Gently.prototype._stubFn = function(self, obj, method, name, args) { - var expectation = this.expectations[0], obj, method; - - if (!expectation) { - throw new Error('Unexpected call to '+name+', no call was expected'); - } - - if (expectation.obj !== obj || expectation.method !== method) { - throw new Error('Unexpected call to '+name+', expected call to '+ expectation.name); - } - - expectation.count -= 1; - if (expectation.count === 0) { - this.expectations.shift(); - - // autorestore original if its not a closure - // and no more expectations on that object - var has_more_expectations = this.expectations.reduce(function (memo, expectation) { - return memo || (expectation.obj === obj && expectation.method === method); - }, false); - if (obj !== null && method !== null && !has_more_expectations) { - if (typeof obj[method]._original !== 'undefined') { - obj[method] = obj[method]._original; - delete obj[method]._original; - } else { - delete obj[method]; - } - } - } - - if (expectation.stubFn) { - return expectation.stubFn.apply(self, args); - } -}; - -Gently.prototype._name = function(obj, method, stubFn) { - if (obj) { - var objectName = obj.toString(); - if (objectName == '[object Object]' && obj.constructor.name) { - objectName = '['+obj.constructor.name+']'; - } - return (objectName)+'.'+method+'()'; - } - - if (stubFn.name) { - return stubFn.name+'()'; - } - - return '>> '+stubFn.toString()+' <<'; -}; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/lib/gently/index.js b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/lib/gently/index.js deleted file mode 100644 index 64c1977..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/lib/gently/index.js +++ /dev/null @@ -1 +0,0 @@ -module.exports = require('./gently'); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/package.json b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/package.json deleted file mode 100644 index 9c1b7a0..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/package.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "name": "gently", - "version": "0.9.2", - "directories": { - "lib": "./lib/gently" - }, - "main": "./lib/gently/index", - "dependencies": {}, - "devDependencies": {}, - "engines": { - "node": "*" - }, - "optionalDependencies": {} -} diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/test/common.js b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/test/common.js deleted file mode 100644 index 978b5c5..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/test/common.js +++ /dev/null @@ -1,8 +0,0 @@ -var path = require('path') - , sys = require('sys'); - -require.paths.unshift(path.dirname(__dirname)+'/lib'); - -global.puts = sys.puts; -global.p = function() {sys.error(sys.inspect.apply(null, arguments))};; -global.assert = require('assert'); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/test/simple/test-gently.js b/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/test/simple/test-gently.js deleted file mode 100644 index 4f8fe2d..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/node-gently/test/simple/test-gently.js +++ /dev/null @@ -1,348 +0,0 @@ -require('../common'); -var Gently = require('gently') - , gently; - -function test(test) { - process.removeAllListeners('exit'); - gently = new Gently(); - test(); -} - -test(function constructor() { - assert.deepEqual(gently.expectations, []); - assert.deepEqual(gently.hijacked, {}); - assert.equal(gently.constructor.name, 'Gently'); -}); - -test(function expectBadArgs() { - var BAD_ARG = 'oh no'; - try { - gently.expect(BAD_ARG); - assert.ok(false, 'throw needs to happen'); - } catch (e) { - assert.equal(e.message, 'Bad 1st argument for gently.expect(), object, function, or number expected, got: '+(typeof BAD_ARG)); - } -}); - -test(function expectObjMethod() { - var OBJ = {}, NAME = 'foobar'; - OBJ.foo = function(x) { - return x; - }; - - gently._name = function() { - return NAME; - }; - - var original = OBJ.foo - , stubFn = function() {}; - - (function testAddOne() { - assert.strictEqual(gently.expect(OBJ, 'foo', stubFn), original); - - assert.equal(gently.expectations.length, 1); - var expectation = gently.expectations[0]; - assert.strictEqual(expectation.obj, OBJ); - assert.strictEqual(expectation.method, 'foo'); - assert.strictEqual(expectation.stubFn, stubFn); - assert.strictEqual(expectation.name, NAME); - assert.strictEqual(OBJ.foo._original, original); - })(); - - (function testAddTwo() { - gently.expect(OBJ, 'foo', 2, stubFn); - assert.equal(gently.expectations.length, 2); - assert.strictEqual(OBJ.foo._original, original); - })(); - - (function testAddOneWithoutMock() { - gently.expect(OBJ, 'foo'); - assert.equal(gently.expectations.length, 3); - })(); - - var stubFnCalled = 0, SELF = {}; - gently._stubFn = function(self, obj, method, name, args) { - stubFnCalled++; - assert.strictEqual(self, SELF); - assert.strictEqual(obj, OBJ); - assert.strictEqual(method, 'foo'); - assert.strictEqual(name, NAME); - assert.deepEqual(args, [1, 2]); - return 23; - }; - assert.equal(OBJ.foo.apply(SELF, [1, 2]), 23); - assert.equal(stubFnCalled, 1); -}); - -test(function expectClosure() { - var NAME = 'MY CLOSURE'; - function closureFn() {}; - - gently._name = function() { - return NAME; - }; - - var fn = gently.expect(closureFn); - assert.equal(gently.expectations.length, 1); - var expectation = gently.expectations[0]; - assert.strictEqual(expectation.obj, null); - assert.strictEqual(expectation.method, null); - assert.strictEqual(expectation.stubFn, closureFn); - assert.strictEqual(expectation.name, NAME); - - var stubFnCalled = 0, SELF = {}; - gently._stubFn = function(self, obj, method, name, args) { - stubFnCalled++; - assert.strictEqual(self, SELF); - assert.strictEqual(obj, null); - assert.strictEqual(method, null); - assert.strictEqual(name, NAME); - assert.deepEqual(args, [1, 2]); - return 23; - }; - assert.equal(fn.apply(SELF, [1, 2]), 23); - assert.equal(stubFnCalled, 1); -}); - -test(function expectClosureCount() { - var stubFnCalled = 0; - function closureFn() {stubFnCalled++}; - - var fn = gently.expect(2, closureFn); - assert.equal(gently.expectations.length, 1); - fn(); - assert.equal(gently.expectations.length, 1); - fn(); - assert.equal(stubFnCalled, 2); -}); - -test(function restore() { - var OBJ = {}, NAME = '[my object].myFn()'; - OBJ.foo = function(x) { - return x; - }; - - gently._name = function() { - return NAME; - }; - - var original = OBJ.foo; - gently.expect(OBJ, 'foo'); - gently.restore(OBJ, 'foo'); - assert.strictEqual(OBJ.foo, original); - - (function testError() { - try { - gently.restore(OBJ, 'foo'); - assert.ok(false, 'throw needs to happen'); - } catch (e) { - assert.equal(e.message, NAME+' is not gently stubbed'); - } - })(); -}); - -test(function _stubFn() { - var OBJ1 = {toString: function() {return '[OBJ 1]'}} - , OBJ2 = {toString: function() {return '[OBJ 2]'}, foo: function () {return 'bar';}} - , SELF = {}; - - gently.expect(OBJ1, 'foo', function(x) { - assert.strictEqual(this, SELF); - return x * 2; - }); - - assert.equal(gently._stubFn(SELF, OBJ1, 'foo', 'dummy_name', [5]), 10); - - (function testAutorestore() { - assert.equal(OBJ2.foo(), 'bar'); - - gently.expect(OBJ2, 'foo', function() { - return 'stubbed foo'; - }); - - gently.expect(OBJ2, 'foo', function() { - return "didn't restore yet"; - }); - - assert.equal(gently._stubFn(SELF, OBJ2, 'foo', 'dummy_name', []), 'stubbed foo'); - assert.equal(gently._stubFn(SELF, OBJ2, 'foo', 'dummy_name', []), "didn't restore yet"); - assert.equal(OBJ2.foo(), 'bar'); - assert.deepEqual(gently.expectations, []); - })(); - - (function testNoMoreCallExpected() { - try { - gently._stubFn(SELF, OBJ1, 'foo', 'dummy_name', [5]); - assert.ok(false, 'throw needs to happen'); - } catch (e) { - assert.equal(e.message, 'Unexpected call to dummy_name, no call was expected'); - } - })(); - - (function testDifferentCallExpected() { - gently.expect(OBJ2, 'bar'); - try { - gently._stubFn(SELF, OBJ1, 'foo', 'dummy_name', [5]); - assert.ok(false, 'throw needs to happen'); - } catch (e) { - assert.equal(e.message, 'Unexpected call to dummy_name, expected call to '+gently._name(OBJ2, 'bar')); - } - - assert.equal(gently.expectations.length, 1); - })(); - - (function testNoMockCallback() { - OBJ2.bar(); - assert.equal(gently.expectations.length, 0); - })(); -}); - -test(function stub() { - var LOCATION = './my_class'; - - (function testRegular() { - var Stub = gently.stub(LOCATION); - assert.ok(Stub instanceof Function); - assert.strictEqual(gently.hijacked[LOCATION], Stub); - assert.ok(Stub['new'] instanceof Function); - assert.equal(Stub.toString(), 'require('+JSON.stringify(LOCATION)+')'); - - (function testConstructor() { - var newCalled = 0 - , STUB - , ARGS = ['foo', 'bar']; - - Stub['new'] = function(a, b) { - assert.equal(a, ARGS[0]); - assert.equal(b, ARGS[1]); - newCalled++; - STUB = this; - }; - - var stub = new Stub(ARGS[0], ARGS[1]); - assert.strictEqual(stub, STUB); - assert.equal(newCalled, 1); - assert.equal(stub.toString(), 'require('+JSON.stringify(LOCATION)+')'); - })(); - - (function testUseReturnValueAsInstance() { - var R = {}; - - Stub['new'] = function() { - return R; - }; - - var stub = new Stub(); - assert.strictEqual(stub, R); - - })(); - })(); - - var EXPORTS_NAME = 'MyClass'; - test(function testExportsName() { - var Stub = gently.stub(LOCATION, EXPORTS_NAME); - assert.strictEqual(gently.hijacked[LOCATION][EXPORTS_NAME], Stub); - assert.equal(Stub.toString(), 'require('+JSON.stringify(LOCATION)+').'+EXPORTS_NAME); - - (function testConstructor() { - var stub = new Stub(); - assert.equal(Stub.toString(), 'require('+JSON.stringify(LOCATION)+').'+EXPORTS_NAME); - })(); - }); -}); - -test(function hijack() { - var LOCATION = './foo' - , REQUIRE_CALLS = 0 - , EXPORTS = {} - , REQUIRE = function() { - REQUIRE_CALLS++; - return EXPORTS; - }; - - var hijackedRequire = gently.hijack(REQUIRE); - hijackedRequire(LOCATION); - assert.strictEqual(gently.hijacked[LOCATION], EXPORTS); - - assert.equal(REQUIRE_CALLS, 1); - - // make sure we are caching the hijacked module - hijackedRequire(LOCATION); - assert.equal(REQUIRE_CALLS, 1); -}); - -test(function verify() { - var OBJ = {toString: function() {return '[OBJ]'}}; - gently.verify(); - - gently.expect(OBJ, 'foo'); - try { - gently.verify(); - assert.ok(false, 'throw needs to happen'); - } catch (e) { - assert.equal(e.message, 'Expected call to [OBJ].foo() did not happen'); - } - - try { - gently.verify('foo'); - assert.ok(false, 'throw needs to happen'); - } catch (e) { - assert.equal(e.message, 'Expected call to [OBJ].foo() did not happen (foo)'); - } -}); - -test(function processExit() { - var verifyCalled = 0; - gently.verify = function(msg) { - verifyCalled++; - assert.equal(msg, 'process exit'); - }; - - process.emit('exit'); - assert.equal(verifyCalled, 1); -}); - -test(function _name() { - (function testNamedClass() { - function Foo() {}; - var foo = new Foo(); - assert.equal(gently._name(foo, 'bar'), '[Foo].bar()'); - })(); - - (function testToStringPreference() { - function Foo() {}; - Foo.prototype.toString = function() { - return '[Superman 123]'; - }; - var foo = new Foo(); - assert.equal(gently._name(foo, 'bar'), '[Superman 123].bar()'); - })(); - - (function testUnamedClass() { - var Foo = function() {}; - var foo = new Foo(); - assert.equal(gently._name(foo, 'bar'), foo.toString()+'.bar()'); - })(); - - (function testNamedClosure() { - function myClosure() {}; - assert.equal(gently._name(null, null, myClosure), myClosure.name+'()'); - })(); - - (function testUnamedClosure() { - var myClosure = function() {2+2 == 5}; - assert.equal(gently._name(null, null, myClosure), '>> '+myClosure.toString()+' <<'); - })(); -}); - -test(function verifyExpectNone() { - var OBJ = {toString: function() {return '[OBJ]'}}; - gently.verify(); - - gently.expect(OBJ, 'foo', 0); - try { - gently.verify(); - } catch (e) { - assert.fail('Exception should not have been thrown'); - } -}); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/package.json b/node_modules/express/node_modules/connect/node_modules/formidable/package.json deleted file mode 100644 index 177f84f..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/package.json +++ /dev/null @@ -1,28 +0,0 @@ -{ - "name": "formidable", - "version": "1.0.11", - "dependencies": {}, - "devDependencies": { - "gently": "0.8.0", - "findit": "0.1.1", - "hashish": "0.0.4", - "urun": "0.0.4", - "utest": "0.0.3" - }, - "directories": { - "lib": "./lib" - }, - "main": "./lib/index", - "scripts": { - "test": "make test" - }, - "engines": { - "node": "*" - }, - "optionalDependencies": {}, - "readme": "# Formidable\n\n[![Build Status](https://secure.travis-ci.org/felixge/node-formidable.png?branch=master)](http://travis-ci.org/felixge/node-formidable)\n\n## Purpose\n\nA node.js module for parsing form data, especially file uploads.\n\n## Current status\n\nThis module was developed for [Transloadit](http://transloadit.com/), a service focused on uploading\nand encoding images and videos. It has been battle-tested against hundreds of GB of file uploads from\na large variety of clients and is considered production-ready.\n\n## Features\n\n* Fast (~500mb/sec), non-buffering multipart parser\n* Automatically writing file uploads to disk\n* Low memory footprint\n* Graceful error handling\n* Very high test coverage\n\n## Changelog\n\n### v1.0.9\n\n* Emit progress when content length header parsed (Tim Koschützki)\n* Fix Readme syntax due to GitHub changes (goob)\n* Replace references to old 'sys' module in Readme with 'util' (Peter Sugihara)\n\n### v1.0.8\n\n* Strip potentially unsafe characters when using `keepExtensions: true`.\n* Switch to utest / urun for testing\n* Add travis build\n\n### v1.0.7\n\n* Remove file from package that was causing problems when installing on windows. (#102)\n* Fix typos in Readme (Jason Davies).\n\n### v1.0.6\n\n* Do not default to the default to the field name for file uploads where\n filename=\"\".\n\n### v1.0.5\n\n* Support filename=\"\" in multipart parts\n* Explain unexpected end() errors in parser better\n\n**Note:** Starting with this version, formidable emits 'file' events for empty\nfile input fields. Previously those were incorrectly emitted as regular file\ninput fields with value = \"\".\n\n### v1.0.4\n\n* Detect a good default tmp directory regardless of platform. (#88)\n\n### v1.0.3\n\n* Fix problems with utf8 characters (#84) / semicolons in filenames (#58)\n* Small performance improvements\n* New test suite and fixture system\n\n### v1.0.2\n\n* Exclude node\\_modules folder from git\n* Implement new `'aborted'` event\n* Fix files in example folder to work with recent node versions\n* Make gently a devDependency\n\n[See Commits](https://github.com/felixge/node-formidable/compare/v1.0.1...v1.0.2)\n\n### v1.0.1\n\n* Fix package.json to refer to proper main directory. (#68, Dean Landolt)\n\n[See Commits](https://github.com/felixge/node-formidable/compare/v1.0.0...v1.0.1)\n\n### v1.0.0\n\n* Add support for multipart boundaries that are quoted strings. (Jeff Craig)\n\nThis marks the beginning of development on version 2.0 which will include\nseveral architectural improvements.\n\n[See Commits](https://github.com/felixge/node-formidable/compare/v0.9.11...v1.0.0)\n\n### v0.9.11\n\n* Emit `'progress'` event when receiving data, regardless of parsing it. (Tim Koschützki)\n* Use [W3C FileAPI Draft](http://dev.w3.org/2006/webapi/FileAPI/) properties for File class\n\n**Important:** The old property names of the File class will be removed in a\nfuture release.\n\n[See Commits](https://github.com/felixge/node-formidable/compare/v0.9.10...v0.9.11)\n\n### Older releases\n\nThese releases were done before starting to maintain the above Changelog:\n\n* [v0.9.10](https://github.com/felixge/node-formidable/compare/v0.9.9...v0.9.10)\n* [v0.9.9](https://github.com/felixge/node-formidable/compare/v0.9.8...v0.9.9)\n* [v0.9.8](https://github.com/felixge/node-formidable/compare/v0.9.7...v0.9.8)\n* [v0.9.7](https://github.com/felixge/node-formidable/compare/v0.9.6...v0.9.7)\n* [v0.9.6](https://github.com/felixge/node-formidable/compare/v0.9.5...v0.9.6)\n* [v0.9.5](https://github.com/felixge/node-formidable/compare/v0.9.4...v0.9.5)\n* [v0.9.4](https://github.com/felixge/node-formidable/compare/v0.9.3...v0.9.4)\n* [v0.9.3](https://github.com/felixge/node-formidable/compare/v0.9.2...v0.9.3)\n* [v0.9.2](https://github.com/felixge/node-formidable/compare/v0.9.1...v0.9.2)\n* [v0.9.1](https://github.com/felixge/node-formidable/compare/v0.9.0...v0.9.1)\n* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0)\n* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0)\n* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0)\n* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0)\n* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0)\n* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0)\n* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0)\n* [v0.9.0](https://github.com/felixge/node-formidable/compare/v0.8.0...v0.9.0)\n* [v0.1.0](https://github.com/felixge/node-formidable/commits/v0.1.0)\n\n## Installation\n\nVia [npm](http://github.com/isaacs/npm):\n\n npm install formidable@latest\n\nManually:\n\n git clone git://github.com/felixge/node-formidable.git formidable\n vim my.js\n # var formidable = require('./formidable');\n\nNote: Formidable requires [gently](http://github.com/felixge/node-gently) to run the unit tests, but you won't need it for just using the library.\n\n## Example\n\nParse an incoming file upload.\n\n var formidable = require('formidable'),\n http = require('http'),\n\n util = require('util');\n\n http.createServer(function(req, res) {\n if (req.url == '/upload' && req.method.toLowerCase() == 'post') {\n // parse a file upload\n var form = new formidable.IncomingForm();\n form.parse(req, function(err, fields, files) {\n res.writeHead(200, {'content-type': 'text/plain'});\n res.write('received upload:\\n\\n');\n res.end(util.inspect({fields: fields, files: files}));\n });\n return;\n }\n\n // show a file upload form\n res.writeHead(200, {'content-type': 'text/html'});\n res.end(\n '
    '+\n '
    '+\n '
    '+\n ''+\n '
    '\n );\n }).listen(80);\n\n## API\n\n### formidable.IncomingForm\n\n__new formidable.IncomingForm()__\n\nCreates a new incoming form.\n\n__incomingForm.encoding = 'utf-8'__\n\nThe encoding to use for incoming form fields.\n\n__incomingForm.uploadDir = process.env.TMP || '/tmp' || process.cwd()__\n\nThe directory for placing file uploads in. You can move them later on using\n`fs.rename()`. The default directory is picked at module load time depending on\nthe first existing directory from those listed above.\n\n__incomingForm.keepExtensions = false__\n\nIf you want the files written to `incomingForm.uploadDir` to include the extensions of the original files, set this property to `true`.\n\n__incomingForm.type__\n\nEither 'multipart' or 'urlencoded' depending on the incoming request.\n\n__incomingForm.maxFieldsSize = 2 * 1024 * 1024__\n\nLimits the amount of memory a field (not file) can allocate in bytes.\nIf this value is exceeded, an `'error'` event is emitted. The default\nsize is 2MB.\n\n__incomingForm.hash = false__\n\nIf you want checksums calculated for incoming files, set this to either `'sha1'` or `'md5'`.\n\n__incomingForm.bytesReceived__\n\nThe amount of bytes received for this form so far.\n\n__incomingForm.bytesExpected__\n\nThe expected number of bytes in this form.\n\n__incomingForm.parse(request, [cb])__\n\nParses an incoming node.js `request` containing form data. If `cb` is provided, all fields an files are collected and passed to the callback:\n\n incomingForm.parse(req, function(err, fields, files) {\n // ...\n });\n\n__incomingForm.onPart(part)__\n\nYou may overwrite this method if you are interested in directly accessing the multipart stream. Doing so will disable any `'field'` / `'file'` events processing which would occur otherwise, making you fully responsible for handling the processing.\n\n incomingForm.onPart = function(part) {\n part.addListener('data', function() {\n // ...\n });\n }\n\nIf you want to use formidable to only handle certain parts for you, you can do so:\n\n incomingForm.onPart = function(part) {\n if (!part.filename) {\n // let formidable handle all non-file parts\n incomingForm.handlePart(part);\n }\n }\n\nCheck the code in this method for further inspiration.\n\n__Event: 'progress' (bytesReceived, bytesExpected)__\n\nEmitted after each incoming chunk of data that has been parsed. Can be used to roll your own progress bar.\n\n__Event: 'field' (name, value)__\n\nEmitted whenever a field / value pair has been received.\n\n__Event: 'fileBegin' (name, file)__\n\nEmitted whenever a new file is detected in the upload stream. Use this even if\nyou want to stream the file to somewhere else while buffering the upload on\nthe file system.\n\n__Event: 'file' (name, file)__\n\nEmitted whenever a field / file pair has been received. `file` is an instance of `File`.\n\n__Event: 'error' (err)__\n\nEmitted when there is an error processing the incoming form. A request that experiences an error is automatically paused, you will have to manually call `request.resume()` if you want the request to continue firing `'data'` events.\n\n__Event: 'aborted'__\n\nEmitted when the request was aborted by the user. Right now this can be due to a 'timeout' or 'close' event on the socket. In the future there will be a separate 'timeout' event (needs a change in the node core).\n\n__Event: 'end' ()__\n\nEmitted when the entire request has been received, and all contained files have finished flushing to disk. This is a great place for you to send your response.\n\n### formidable.File\n\n__file.size = 0__\n\nThe size of the uploaded file in bytes. If the file is still being uploaded (see `'fileBegin'` event), this property says how many bytes of the file have been written to disk yet.\n\n__file.path = null__\n\nThe path this file is being written to. You can modify this in the `'fileBegin'` event in\ncase you are unhappy with the way formidable generates a temporary path for your files.\n\n__file.name = null__\n\nThe name this file had according to the uploading client.\n\n__file.type = null__\n\nThe mime type of this file, according to the uploading client.\n\n__file.lastModifiedDate = null__\n\nA date object (or `null`) containing the time this file was last written to. Mostly\nhere for compatibility with the [W3C File API Draft](http://dev.w3.org/2006/webapi/FileAPI/).\n\n__file.hash = null__\n\nIf hash calculation was set, you can read the hex digest out of this var.\n\n## License\n\nFormidable is licensed under the MIT license.\n\n## Ports\n\n* [multipart-parser](http://github.com/FooBarWidget/multipart-parser): a C++ parser based on formidable\n\n## Credits\n\n* [Ryan Dahl](http://twitter.com/ryah) for his work on [http-parser](http://github.com/ry/http-parser) which heavily inspired multipart_parser.js\n", - "readmeFilename": "Readme.md", - "description": "[![Build Status](https://secure.travis-ci.org/felixge/node-formidable.png?branch=master)](http://travis-ci.org/felixge/node-formidable)", - "_id": "formidable@1.0.11", - "_from": "formidable@1.0.11" -} diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/common.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/common.js deleted file mode 100644 index eb432ad..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/common.js +++ /dev/null @@ -1,19 +0,0 @@ -var mysql = require('..'); -var path = require('path'); - -var root = path.join(__dirname, '../'); -exports.dir = { - root : root, - lib : root + '/lib', - fixture : root + '/test/fixture', - tmp : root + '/test/tmp', -}; - -exports.port = 13532; - -exports.formidable = require('..'); -exports.assert = require('assert'); - -exports.require = function(lib) { - return require(exports.dir.lib + '/' + lib); -}; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/file/funkyfilename.txt b/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/file/funkyfilename.txt deleted file mode 100644 index e7a4785..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/file/funkyfilename.txt +++ /dev/null @@ -1 +0,0 @@ -I am a text file with a funky name! diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/file/plain.txt b/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/file/plain.txt deleted file mode 100644 index 9b6903e..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/file/plain.txt +++ /dev/null @@ -1 +0,0 @@ -I am a plain text file diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/http/special-chars-in-filename/info.md b/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/http/special-chars-in-filename/info.md deleted file mode 100644 index 3c9dbe3..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/http/special-chars-in-filename/info.md +++ /dev/null @@ -1,3 +0,0 @@ -* Opera does not allow submitting this file, it shows a warning to the - user that the file could not be found instead. Tested in 9.8, 11.51 on OSX. - Reported to Opera on 08.09.2011 (tracking email DSK-346009@bugs.opera.com). diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/js/no-filename.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/js/no-filename.js deleted file mode 100644 index 0bae449..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/js/no-filename.js +++ /dev/null @@ -1,3 +0,0 @@ -module.exports['generic.http'] = [ - {type: 'file', name: 'upload', filename: '', fixture: 'plain.txt'}, -]; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/js/special-chars-in-filename.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/js/special-chars-in-filename.js deleted file mode 100644 index eb76fdc..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/js/special-chars-in-filename.js +++ /dev/null @@ -1,21 +0,0 @@ -var properFilename = 'funkyfilename.txt'; - -function expect(filename) { - return [ - {type: 'field', name: 'title', value: 'Weird filename'}, - {type: 'file', name: 'upload', filename: filename, fixture: properFilename}, - ]; -}; - -var webkit = " ? % * | \" < > . ? ; ' @ # $ ^ & ( ) - _ = + { } [ ] ` ~.txt"; -var ffOrIe = " ? % * | \" < > . ☃ ; ' @ # $ ^ & ( ) - _ = + { } [ ] ` ~.txt"; - -module.exports = { - 'osx-chrome-13.http' : expect(webkit), - 'osx-firefox-3.6.http' : expect(ffOrIe), - 'osx-safari-5.http' : expect(webkit), - 'xp-chrome-12.http' : expect(webkit), - 'xp-ie-7.http' : expect(ffOrIe), - 'xp-ie-8.http' : expect(ffOrIe), - 'xp-safari-5.http' : expect(webkit), -}; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/multipart.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/multipart.js deleted file mode 100644 index a476169..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/fixture/multipart.js +++ /dev/null @@ -1,72 +0,0 @@ -exports['rfc1867'] = - { boundary: 'AaB03x', - raw: - '--AaB03x\r\n'+ - 'content-disposition: form-data; name="field1"\r\n'+ - '\r\n'+ - 'Joe Blow\r\nalmost tricked you!\r\n'+ - '--AaB03x\r\n'+ - 'content-disposition: form-data; name="pics"; filename="file1.txt"\r\n'+ - 'Content-Type: text/plain\r\n'+ - '\r\n'+ - '... contents of file1.txt ...\r\r\n'+ - '--AaB03x--\r\n', - parts: - [ { headers: { - 'content-disposition': 'form-data; name="field1"', - }, - data: 'Joe Blow\r\nalmost tricked you!', - }, - { headers: { - 'content-disposition': 'form-data; name="pics"; filename="file1.txt"', - 'Content-Type': 'text/plain', - }, - data: '... contents of file1.txt ...\r', - } - ] - }; - -exports['noTrailing\r\n'] = - { boundary: 'AaB03x', - raw: - '--AaB03x\r\n'+ - 'content-disposition: form-data; name="field1"\r\n'+ - '\r\n'+ - 'Joe Blow\r\nalmost tricked you!\r\n'+ - '--AaB03x\r\n'+ - 'content-disposition: form-data; name="pics"; filename="file1.txt"\r\n'+ - 'Content-Type: text/plain\r\n'+ - '\r\n'+ - '... contents of file1.txt ...\r\r\n'+ - '--AaB03x--', - parts: - [ { headers: { - 'content-disposition': 'form-data; name="field1"', - }, - data: 'Joe Blow\r\nalmost tricked you!', - }, - { headers: { - 'content-disposition': 'form-data; name="pics"; filename="file1.txt"', - 'Content-Type': 'text/plain', - }, - data: '... contents of file1.txt ...\r', - } - ] - }; - -exports['emptyHeader'] = - { boundary: 'AaB03x', - raw: - '--AaB03x\r\n'+ - 'content-disposition: form-data; name="field1"\r\n'+ - ': foo\r\n'+ - '\r\n'+ - 'Joe Blow\r\nalmost tricked you!\r\n'+ - '--AaB03x\r\n'+ - 'content-disposition: form-data; name="pics"; filename="file1.txt"\r\n'+ - 'Content-Type: text/plain\r\n'+ - '\r\n'+ - '... contents of file1.txt ...\r\r\n'+ - '--AaB03x--\r\n', - expectError: true, - }; diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/integration/test-fixtures.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/integration/test-fixtures.js deleted file mode 100644 index 66ad259..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/integration/test-fixtures.js +++ /dev/null @@ -1,89 +0,0 @@ -var hashish = require('hashish'); -var fs = require('fs'); -var findit = require('findit'); -var path = require('path'); -var http = require('http'); -var net = require('net'); -var assert = require('assert'); - -var common = require('../common'); -var formidable = common.formidable; - -var server = http.createServer(); -server.listen(common.port, findFixtures); - -function findFixtures() { - var fixtures = []; - findit - .sync(common.dir.fixture + '/js') - .forEach(function(jsPath) { - if (!/\.js$/.test(jsPath)) return; - - var group = path.basename(jsPath, '.js'); - hashish.forEach(require(jsPath), function(fixture, name) { - fixtures.push({ - name : group + '/' + name, - fixture : fixture, - }); - }); - }); - - testNext(fixtures); -} - -function testNext(fixtures) { - var fixture = fixtures.shift(); - if (!fixture) return server.close(); - - var name = fixture.name; - var fixture = fixture.fixture; - - uploadFixture(name, function(err, parts) { - if (err) throw err; - - fixture.forEach(function(expectedPart, i) { - var parsedPart = parts[i]; - assert.equal(parsedPart.type, expectedPart.type); - assert.equal(parsedPart.name, expectedPart.name); - - if (parsedPart.type === 'file') { - var filename = parsedPart.value.name; - assert.equal(filename, expectedPart.filename); - } - }); - - testNext(fixtures); - }); -}; - -function uploadFixture(name, cb) { - server.once('request', function(req, res) { - var form = new formidable.IncomingForm(); - form.uploadDir = common.dir.tmp; - form.parse(req); - - function callback() { - var realCallback = cb; - cb = function() {}; - realCallback.apply(null, arguments); - } - - var parts = []; - form - .on('error', callback) - .on('fileBegin', function(name, value) { - parts.push({type: 'file', name: name, value: value}); - }) - .on('field', function(name, value) { - parts.push({type: 'field', name: name, value: value}); - }) - .on('end', function() { - callback(null, parts); - }); - }); - - var socket = net.createConnection(common.port); - var file = fs.createReadStream(common.dir.fixture + '/http/' + name); - - file.pipe(socket); -} diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/common.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/common.js deleted file mode 100644 index 2b98598..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/common.js +++ /dev/null @@ -1,24 +0,0 @@ -var path = require('path'), - fs = require('fs'); - -try { - global.Gently = require('gently'); -} catch (e) { - throw new Error('this test suite requires node-gently'); -} - -exports.lib = path.join(__dirname, '../../lib'); - -global.GENTLY = new Gently(); - -global.assert = require('assert'); -global.TEST_PORT = 13532; -global.TEST_FIXTURES = path.join(__dirname, '../fixture'); -global.TEST_TMP = path.join(__dirname, '../tmp'); - -// Stupid new feature in node that complains about gently attaching too many -// listeners to process 'exit'. This is a workaround until I can think of a -// better way to deal with this. -if (process.setMaxListeners) { - process.setMaxListeners(10000); -} diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/integration/test-multipart-parser.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/integration/test-multipart-parser.js deleted file mode 100644 index 75232aa..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/integration/test-multipart-parser.js +++ /dev/null @@ -1,80 +0,0 @@ -var common = require('../common'); -var CHUNK_LENGTH = 10, - multipartParser = require(common.lib + '/multipart_parser'), - MultipartParser = multipartParser.MultipartParser, - parser = new MultipartParser(), - fixtures = require(TEST_FIXTURES + '/multipart'), - Buffer = require('buffer').Buffer; - -Object.keys(fixtures).forEach(function(name) { - var fixture = fixtures[name], - buffer = new Buffer(Buffer.byteLength(fixture.raw, 'binary')), - offset = 0, - chunk, - nparsed, - - parts = [], - part = null, - headerField, - headerValue, - endCalled = ''; - - parser.initWithBoundary(fixture.boundary); - parser.onPartBegin = function() { - part = {headers: {}, data: ''}; - parts.push(part); - headerField = ''; - headerValue = ''; - }; - - parser.onHeaderField = function(b, start, end) { - headerField += b.toString('ascii', start, end); - }; - - parser.onHeaderValue = function(b, start, end) { - headerValue += b.toString('ascii', start, end); - } - - parser.onHeaderEnd = function() { - part.headers[headerField] = headerValue; - headerField = ''; - headerValue = ''; - }; - - parser.onPartData = function(b, start, end) { - var str = b.toString('ascii', start, end); - part.data += b.slice(start, end); - } - - parser.onEnd = function() { - endCalled = true; - } - - buffer.write(fixture.raw, 'binary', 0); - - while (offset < buffer.length) { - if (offset + CHUNK_LENGTH < buffer.length) { - chunk = buffer.slice(offset, offset+CHUNK_LENGTH); - } else { - chunk = buffer.slice(offset, buffer.length); - } - offset = offset + CHUNK_LENGTH; - - nparsed = parser.write(chunk); - if (nparsed != chunk.length) { - if (fixture.expectError) { - return; - } - puts('-- ERROR --'); - p(chunk.toString('ascii')); - throw new Error(chunk.length+' bytes written, but only '+nparsed+' bytes parsed!'); - } - } - - if (fixture.expectError) { - throw new Error('expected parse error did not happen'); - } - - assert.ok(endCalled); - assert.deepEqual(parts, fixture.parts); -}); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-file.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-file.js deleted file mode 100644 index 52ceedb..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-file.js +++ /dev/null @@ -1,104 +0,0 @@ -var common = require('../common'); -var WriteStreamStub = GENTLY.stub('fs', 'WriteStream'); - -var File = require(common.lib + '/file'), - EventEmitter = require('events').EventEmitter, - file, - gently; - -function test(test) { - gently = new Gently(); - file = new File(); - test(); - gently.verify(test.name); -} - -test(function constructor() { - assert.ok(file instanceof EventEmitter); - assert.strictEqual(file.size, 0); - assert.strictEqual(file.path, null); - assert.strictEqual(file.name, null); - assert.strictEqual(file.type, null); - assert.strictEqual(file.lastModifiedDate, null); - - assert.strictEqual(file._writeStream, null); - - (function testSetProperties() { - var file2 = new File({foo: 'bar'}); - assert.equal(file2.foo, 'bar'); - })(); -}); - -test(function open() { - var WRITE_STREAM; - file.path = '/foo'; - - gently.expect(WriteStreamStub, 'new', function (path) { - WRITE_STREAM = this; - assert.strictEqual(path, file.path); - }); - - file.open(); - assert.strictEqual(file._writeStream, WRITE_STREAM); -}); - -test(function write() { - var BUFFER = {length: 10}, - CB_STUB, - CB = function() { - CB_STUB.apply(this, arguments); - }; - - file._writeStream = {}; - - gently.expect(file._writeStream, 'write', function (buffer, cb) { - assert.strictEqual(buffer, BUFFER); - - gently.expect(file, 'emit', function (event, bytesWritten) { - assert.ok(file.lastModifiedDate instanceof Date); - assert.equal(event, 'progress'); - assert.equal(bytesWritten, file.size); - }); - - CB_STUB = gently.expect(function writeCb() { - assert.equal(file.size, 10); - }); - - cb(); - - gently.expect(file, 'emit', function (event, bytesWritten) { - assert.equal(event, 'progress'); - assert.equal(bytesWritten, file.size); - }); - - CB_STUB = gently.expect(function writeCb() { - assert.equal(file.size, 20); - }); - - cb(); - }); - - file.write(BUFFER, CB); -}); - -test(function end() { - var CB_STUB, - CB = function() { - CB_STUB.apply(this, arguments); - }; - - file._writeStream = {}; - - gently.expect(file._writeStream, 'end', function (cb) { - gently.expect(file, 'emit', function (event) { - assert.equal(event, 'end'); - }); - - CB_STUB = gently.expect(function endCb() { - }); - - cb(); - }); - - file.end(CB); -}); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-incoming-form.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-incoming-form.js deleted file mode 100644 index 84de439..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-incoming-form.js +++ /dev/null @@ -1,727 +0,0 @@ -var common = require('../common'); -var MultipartParserStub = GENTLY.stub('./multipart_parser', 'MultipartParser'), - QuerystringParserStub = GENTLY.stub('./querystring_parser', 'QuerystringParser'), - EventEmitterStub = GENTLY.stub('events', 'EventEmitter'), - StreamStub = GENTLY.stub('stream', 'Stream'), - FileStub = GENTLY.stub('./file'); - -var formidable = require(common.lib + '/index'), - IncomingForm = formidable.IncomingForm, - events = require('events'), - fs = require('fs'), - path = require('path'), - Buffer = require('buffer').Buffer, - fixtures = require(TEST_FIXTURES + '/multipart'), - form, - gently; - -function test(test) { - gently = new Gently(); - gently.expect(EventEmitterStub, 'call'); - form = new IncomingForm(); - test(); - gently.verify(test.name); -} - -test(function constructor() { - assert.strictEqual(form.error, null); - assert.strictEqual(form.ended, false); - assert.strictEqual(form.type, null); - assert.strictEqual(form.headers, null); - assert.strictEqual(form.keepExtensions, false); - assert.strictEqual(form.uploadDir, '/tmp'); - assert.strictEqual(form.encoding, 'utf-8'); - assert.strictEqual(form.bytesReceived, null); - assert.strictEqual(form.bytesExpected, null); - assert.strictEqual(form.maxFieldsSize, 2 * 1024 * 1024); - assert.strictEqual(form._parser, null); - assert.strictEqual(form._flushing, 0); - assert.strictEqual(form._fieldsSize, 0); - assert.ok(form instanceof EventEmitterStub); - assert.equal(form.constructor.name, 'IncomingForm'); - - (function testSimpleConstructor() { - gently.expect(EventEmitterStub, 'call'); - var form = IncomingForm(); - assert.ok(form instanceof IncomingForm); - })(); - - (function testSimpleConstructorShortcut() { - gently.expect(EventEmitterStub, 'call'); - var form = formidable(); - assert.ok(form instanceof IncomingForm); - })(); -}); - -test(function parse() { - var REQ = {headers: {}} - , emit = {}; - - gently.expect(form, 'writeHeaders', function(headers) { - assert.strictEqual(headers, REQ.headers); - }); - - var events = ['error', 'aborted', 'data', 'end']; - gently.expect(REQ, 'on', events.length, function(event, fn) { - assert.equal(event, events.shift()); - emit[event] = fn; - return this; - }); - - form.parse(REQ); - - (function testPause() { - gently.expect(REQ, 'pause'); - assert.strictEqual(form.pause(), true); - })(); - - (function testPauseCriticalException() { - form.ended = false; - - var ERR = new Error('dasdsa'); - gently.expect(REQ, 'pause', function() { - throw ERR; - }); - - gently.expect(form, '_error', function(err) { - assert.strictEqual(err, ERR); - }); - - assert.strictEqual(form.pause(), false); - })(); - - (function testPauseHarmlessException() { - form.ended = true; - - var ERR = new Error('dasdsa'); - gently.expect(REQ, 'pause', function() { - throw ERR; - }); - - assert.strictEqual(form.pause(), false); - })(); - - (function testResume() { - gently.expect(REQ, 'resume'); - assert.strictEqual(form.resume(), true); - })(); - - (function testResumeCriticalException() { - form.ended = false; - - var ERR = new Error('dasdsa'); - gently.expect(REQ, 'resume', function() { - throw ERR; - }); - - gently.expect(form, '_error', function(err) { - assert.strictEqual(err, ERR); - }); - - assert.strictEqual(form.resume(), false); - })(); - - (function testResumeHarmlessException() { - form.ended = true; - - var ERR = new Error('dasdsa'); - gently.expect(REQ, 'resume', function() { - throw ERR; - }); - - assert.strictEqual(form.resume(), false); - })(); - - (function testEmitError() { - var ERR = new Error('something bad happened'); - gently.expect(form, '_error',function(err) { - assert.strictEqual(err, ERR); - }); - emit.error(ERR); - })(); - - (function testEmitAborted() { - gently.expect(form, 'emit',function(event) { - assert.equal(event, 'aborted'); - }); - - emit.aborted(); - })(); - - - (function testEmitData() { - var BUFFER = [1, 2, 3]; - gently.expect(form, 'write', function(buffer) { - assert.strictEqual(buffer, BUFFER); - }); - emit.data(BUFFER); - })(); - - (function testEmitEnd() { - form._parser = {}; - - (function testWithError() { - var ERR = new Error('haha'); - gently.expect(form._parser, 'end', function() { - return ERR; - }); - - gently.expect(form, '_error', function(err) { - assert.strictEqual(err, ERR); - }); - - emit.end(); - })(); - - (function testWithoutError() { - gently.expect(form._parser, 'end'); - emit.end(); - })(); - - (function testAfterError() { - form.error = true; - emit.end(); - })(); - })(); - - (function testWithCallback() { - gently.expect(EventEmitterStub, 'call'); - var form = new IncomingForm(), - REQ = {headers: {}}, - parseCalled = 0; - - gently.expect(form, 'writeHeaders'); - gently.expect(REQ, 'on', 4, function() { - return this; - }); - - gently.expect(form, 'on', 4, function(event, fn) { - if (event == 'field') { - fn('field1', 'foo'); - fn('field1', 'bar'); - fn('field2', 'nice'); - } - - if (event == 'file') { - fn('file1', '1'); - fn('file1', '2'); - fn('file2', '3'); - } - - if (event == 'end') { - fn(); - } - return this; - }); - - form.parse(REQ, gently.expect(function parseCbOk(err, fields, files) { - assert.deepEqual(fields, {field1: 'bar', field2: 'nice'}); - assert.deepEqual(files, {file1: '2', file2: '3'}); - })); - - gently.expect(form, 'writeHeaders'); - gently.expect(REQ, 'on', 4, function() { - return this; - }); - - var ERR = new Error('test'); - gently.expect(form, 'on', 3, function(event, fn) { - if (event == 'field') { - fn('foo', 'bar'); - } - - if (event == 'error') { - fn(ERR); - gently.expect(form, 'on'); - } - return this; - }); - - form.parse(REQ, gently.expect(function parseCbErr(err, fields, files) { - assert.strictEqual(err, ERR); - assert.deepEqual(fields, {foo: 'bar'}); - })); - })(); -}); - -test(function pause() { - assert.strictEqual(form.pause(), false); -}); - -test(function resume() { - assert.strictEqual(form.resume(), false); -}); - - -test(function writeHeaders() { - var HEADERS = {}; - gently.expect(form, '_parseContentLength'); - gently.expect(form, '_parseContentType'); - - form.writeHeaders(HEADERS); - assert.strictEqual(form.headers, HEADERS); -}); - -test(function write() { - var parser = {}, - BUFFER = [1, 2, 3]; - - form._parser = parser; - form.bytesExpected = 523423; - - (function testBasic() { - gently.expect(form, 'emit', function(event, bytesReceived, bytesExpected) { - assert.equal(event, 'progress'); - assert.equal(bytesReceived, BUFFER.length); - assert.equal(bytesExpected, form.bytesExpected); - }); - - gently.expect(parser, 'write', function(buffer) { - assert.strictEqual(buffer, BUFFER); - return buffer.length; - }); - - assert.equal(form.write(BUFFER), BUFFER.length); - assert.equal(form.bytesReceived, BUFFER.length); - })(); - - (function testParserError() { - gently.expect(form, 'emit'); - - gently.expect(parser, 'write', function(buffer) { - assert.strictEqual(buffer, BUFFER); - return buffer.length - 1; - }); - - gently.expect(form, '_error', function(err) { - assert.ok(err.message.match(/parser error/i)); - }); - - assert.equal(form.write(BUFFER), BUFFER.length - 1); - assert.equal(form.bytesReceived, BUFFER.length + BUFFER.length); - })(); - - (function testUninitialized() { - delete form._parser; - - gently.expect(form, '_error', function(err) { - assert.ok(err.message.match(/unintialized parser/i)); - }); - form.write(BUFFER); - })(); -}); - -test(function parseContentType() { - var HEADERS = {}; - - form.headers = {'content-type': 'application/x-www-form-urlencoded'}; - gently.expect(form, '_initUrlencoded'); - form._parseContentType(); - - // accept anything that has 'urlencoded' in it - form.headers = {'content-type': 'broken-client/urlencoded-stupid'}; - gently.expect(form, '_initUrlencoded'); - form._parseContentType(); - - var BOUNDARY = '---------------------------57814261102167618332366269'; - form.headers = {'content-type': 'multipart/form-data; boundary='+BOUNDARY}; - - gently.expect(form, '_initMultipart', function(boundary) { - assert.equal(boundary, BOUNDARY); - }); - form._parseContentType(); - - (function testQuotedBoundary() { - form.headers = {'content-type': 'multipart/form-data; boundary="' + BOUNDARY + '"'}; - - gently.expect(form, '_initMultipart', function(boundary) { - assert.equal(boundary, BOUNDARY); - }); - form._parseContentType(); - })(); - - (function testNoBoundary() { - form.headers = {'content-type': 'multipart/form-data'}; - - gently.expect(form, '_error', function(err) { - assert.ok(err.message.match(/no multipart boundary/i)); - }); - form._parseContentType(); - })(); - - (function testNoContentType() { - form.headers = {}; - - gently.expect(form, '_error', function(err) { - assert.ok(err.message.match(/no content-type/i)); - }); - form._parseContentType(); - })(); - - (function testUnknownContentType() { - form.headers = {'content-type': 'invalid'}; - - gently.expect(form, '_error', function(err) { - assert.ok(err.message.match(/unknown content-type/i)); - }); - form._parseContentType(); - })(); -}); - -test(function parseContentLength() { - var HEADERS = {}; - - form.headers = {}; - form._parseContentLength(); - assert.strictEqual(form.bytesReceived, null); - assert.strictEqual(form.bytesExpected, null); - - form.headers['content-length'] = '8'; - gently.expect(form, 'emit', function(event, bytesReceived, bytesExpected) { - assert.equal(event, 'progress'); - assert.equal(bytesReceived, 0); - assert.equal(bytesExpected, 8); - }); - form._parseContentLength(); - assert.strictEqual(form.bytesReceived, 0); - assert.strictEqual(form.bytesExpected, 8); - - // JS can be evil, lets make sure we are not - form.headers['content-length'] = '08'; - gently.expect(form, 'emit', function(event, bytesReceived, bytesExpected) { - assert.equal(event, 'progress'); - assert.equal(bytesReceived, 0); - assert.equal(bytesExpected, 8); - }); - form._parseContentLength(); - assert.strictEqual(form.bytesExpected, 8); -}); - -test(function _initMultipart() { - var BOUNDARY = '123', - PARSER; - - gently.expect(MultipartParserStub, 'new', function() { - PARSER = this; - }); - - gently.expect(MultipartParserStub.prototype, 'initWithBoundary', function(boundary) { - assert.equal(boundary, BOUNDARY); - }); - - form._initMultipart(BOUNDARY); - assert.equal(form.type, 'multipart'); - assert.strictEqual(form._parser, PARSER); - - (function testRegularField() { - var PART; - gently.expect(StreamStub, 'new', function() { - PART = this; - }); - - gently.expect(form, 'onPart', function(part) { - assert.strictEqual(part, PART); - assert.deepEqual - ( part.headers - , { 'content-disposition': 'form-data; name="field1"' - , 'foo': 'bar' - } - ); - assert.equal(part.name, 'field1'); - - var strings = ['hello', ' world']; - gently.expect(part, 'emit', 2, function(event, b) { - assert.equal(event, 'data'); - assert.equal(b.toString(), strings.shift()); - }); - - gently.expect(part, 'emit', function(event, b) { - assert.equal(event, 'end'); - }); - }); - - PARSER.onPartBegin(); - PARSER.onHeaderField(new Buffer('content-disposition'), 0, 10); - PARSER.onHeaderField(new Buffer('content-disposition'), 10, 19); - PARSER.onHeaderValue(new Buffer('form-data; name="field1"'), 0, 14); - PARSER.onHeaderValue(new Buffer('form-data; name="field1"'), 14, 24); - PARSER.onHeaderEnd(); - PARSER.onHeaderField(new Buffer('foo'), 0, 3); - PARSER.onHeaderValue(new Buffer('bar'), 0, 3); - PARSER.onHeaderEnd(); - PARSER.onHeadersEnd(); - PARSER.onPartData(new Buffer('hello world'), 0, 5); - PARSER.onPartData(new Buffer('hello world'), 5, 11); - PARSER.onPartEnd(); - })(); - - (function testFileField() { - var PART; - gently.expect(StreamStub, 'new', function() { - PART = this; - }); - - gently.expect(form, 'onPart', function(part) { - assert.deepEqual - ( part.headers - , { 'content-disposition': 'form-data; name="field2"; filename="C:\\Documents and Settings\\IE\\Must\\Die\\Sun"et.jpg"' - , 'content-type': 'text/plain' - } - ); - assert.equal(part.name, 'field2'); - assert.equal(part.filename, 'Sun"et.jpg'); - assert.equal(part.mime, 'text/plain'); - - gently.expect(part, 'emit', function(event, b) { - assert.equal(event, 'data'); - assert.equal(b.toString(), '... contents of file1.txt ...'); - }); - - gently.expect(part, 'emit', function(event, b) { - assert.equal(event, 'end'); - }); - }); - - PARSER.onPartBegin(); - PARSER.onHeaderField(new Buffer('content-disposition'), 0, 19); - PARSER.onHeaderValue(new Buffer('form-data; name="field2"; filename="C:\\Documents and Settings\\IE\\Must\\Die\\Sun"et.jpg"'), 0, 85); - PARSER.onHeaderEnd(); - PARSER.onHeaderField(new Buffer('Content-Type'), 0, 12); - PARSER.onHeaderValue(new Buffer('text/plain'), 0, 10); - PARSER.onHeaderEnd(); - PARSER.onHeadersEnd(); - PARSER.onPartData(new Buffer('... contents of file1.txt ...'), 0, 29); - PARSER.onPartEnd(); - })(); - - (function testEnd() { - gently.expect(form, '_maybeEnd'); - PARSER.onEnd(); - assert.ok(form.ended); - })(); -}); - -test(function _fileName() { - // TODO - return; -}); - -test(function _initUrlencoded() { - var PARSER; - - gently.expect(QuerystringParserStub, 'new', function() { - PARSER = this; - }); - - form._initUrlencoded(); - assert.equal(form.type, 'urlencoded'); - assert.strictEqual(form._parser, PARSER); - - (function testOnField() { - var KEY = 'KEY', VAL = 'VAL'; - gently.expect(form, 'emit', function(field, key, val) { - assert.equal(field, 'field'); - assert.equal(key, KEY); - assert.equal(val, VAL); - }); - - PARSER.onField(KEY, VAL); - })(); - - (function testOnEnd() { - gently.expect(form, '_maybeEnd'); - - PARSER.onEnd(); - assert.equal(form.ended, true); - })(); -}); - -test(function _error() { - var ERR = new Error('bla'); - - gently.expect(form, 'pause'); - gently.expect(form, 'emit', function(event, err) { - assert.equal(event, 'error'); - assert.strictEqual(err, ERR); - }); - - form._error(ERR); - assert.strictEqual(form.error, ERR); - - // make sure _error only does its thing once - form._error(ERR); -}); - -test(function onPart() { - var PART = {}; - gently.expect(form, 'handlePart', function(part) { - assert.strictEqual(part, PART); - }); - - form.onPart(PART); -}); - -test(function handlePart() { - (function testUtf8Field() { - var PART = new events.EventEmitter(); - PART.name = 'my_field'; - - gently.expect(form, 'emit', function(event, field, value) { - assert.equal(event, 'field'); - assert.equal(field, 'my_field'); - assert.equal(value, 'hello world: €'); - }); - - form.handlePart(PART); - PART.emit('data', new Buffer('hello')); - PART.emit('data', new Buffer(' world: ')); - PART.emit('data', new Buffer([0xE2])); - PART.emit('data', new Buffer([0x82, 0xAC])); - PART.emit('end'); - })(); - - (function testBinaryField() { - var PART = new events.EventEmitter(); - PART.name = 'my_field2'; - - gently.expect(form, 'emit', function(event, field, value) { - assert.equal(event, 'field'); - assert.equal(field, 'my_field2'); - assert.equal(value, 'hello world: '+new Buffer([0xE2, 0x82, 0xAC]).toString('binary')); - }); - - form.encoding = 'binary'; - form.handlePart(PART); - PART.emit('data', new Buffer('hello')); - PART.emit('data', new Buffer(' world: ')); - PART.emit('data', new Buffer([0xE2])); - PART.emit('data', new Buffer([0x82, 0xAC])); - PART.emit('end'); - })(); - - (function testFieldSize() { - form.maxFieldsSize = 8; - var PART = new events.EventEmitter(); - PART.name = 'my_field'; - - gently.expect(form, '_error', function(err) { - assert.equal(err.message, 'maxFieldsSize exceeded, received 9 bytes of field data'); - }); - - form.handlePart(PART); - form._fieldsSize = 1; - PART.emit('data', new Buffer(7)); - PART.emit('data', new Buffer(1)); - })(); - - (function testFilePart() { - var PART = new events.EventEmitter(), - FILE = new events.EventEmitter(), - PATH = '/foo/bar'; - - PART.name = 'my_file'; - PART.filename = 'sweet.txt'; - PART.mime = 'sweet.txt'; - - gently.expect(form, '_uploadPath', function(filename) { - assert.equal(filename, PART.filename); - return PATH; - }); - - gently.expect(FileStub, 'new', function(properties) { - assert.equal(properties.path, PATH); - assert.equal(properties.name, PART.filename); - assert.equal(properties.type, PART.mime); - FILE = this; - - gently.expect(form, 'emit', function (event, field, file) { - assert.equal(event, 'fileBegin'); - assert.strictEqual(field, PART.name); - assert.strictEqual(file, FILE); - }); - - gently.expect(FILE, 'open'); - }); - - form.handlePart(PART); - assert.equal(form._flushing, 1); - - var BUFFER; - gently.expect(form, 'pause'); - gently.expect(FILE, 'write', function(buffer, cb) { - assert.strictEqual(buffer, BUFFER); - gently.expect(form, 'resume'); - // @todo handle cb(new Err) - cb(); - }); - - PART.emit('data', BUFFER = new Buffer('test')); - - gently.expect(FILE, 'end', function(cb) { - gently.expect(form, 'emit', function(event, field, file) { - assert.equal(event, 'file'); - assert.strictEqual(file, FILE); - }); - - gently.expect(form, '_maybeEnd'); - - cb(); - assert.equal(form._flushing, 0); - }); - - PART.emit('end'); - })(); -}); - -test(function _uploadPath() { - (function testUniqueId() { - var UUID_A, UUID_B; - gently.expect(GENTLY.hijacked.path, 'join', function(uploadDir, uuid) { - assert.equal(uploadDir, form.uploadDir); - UUID_A = uuid; - }); - form._uploadPath(); - - gently.expect(GENTLY.hijacked.path, 'join', function(uploadDir, uuid) { - UUID_B = uuid; - }); - form._uploadPath(); - - assert.notEqual(UUID_A, UUID_B); - })(); - - (function testFileExtension() { - form.keepExtensions = true; - var FILENAME = 'foo.jpg', - EXT = '.bar'; - - gently.expect(GENTLY.hijacked.path, 'extname', function(filename) { - assert.equal(filename, FILENAME); - gently.restore(path, 'extname'); - - return EXT; - }); - - gently.expect(GENTLY.hijacked.path, 'join', function(uploadDir, name) { - assert.equal(path.extname(name), EXT); - }); - form._uploadPath(FILENAME); - })(); -}); - -test(function _maybeEnd() { - gently.expect(form, 'emit', 0); - form._maybeEnd(); - - form.ended = true; - form._flushing = 1; - form._maybeEnd(); - - gently.expect(form, 'emit', function(event) { - assert.equal(event, 'end'); - }); - - form.ended = true; - form._flushing = 0; - form._maybeEnd(); -}); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-multipart-parser.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-multipart-parser.js deleted file mode 100644 index d8dc968..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-multipart-parser.js +++ /dev/null @@ -1,50 +0,0 @@ -var common = require('../common'); -var multipartParser = require(common.lib + '/multipart_parser'), - MultipartParser = multipartParser.MultipartParser, - events = require('events'), - Buffer = require('buffer').Buffer, - parser; - -function test(test) { - parser = new MultipartParser(); - test(); -} - -test(function constructor() { - assert.equal(parser.boundary, null); - assert.equal(parser.state, 0); - assert.equal(parser.flags, 0); - assert.equal(parser.boundaryChars, null); - assert.equal(parser.index, null); - assert.equal(parser.lookbehind, null); - assert.equal(parser.constructor.name, 'MultipartParser'); -}); - -test(function initWithBoundary() { - var boundary = 'abc'; - parser.initWithBoundary(boundary); - assert.deepEqual(Array.prototype.slice.call(parser.boundary), [13, 10, 45, 45, 97, 98, 99]); - assert.equal(parser.state, multipartParser.START); - - assert.deepEqual(parser.boundaryChars, {10: true, 13: true, 45: true, 97: true, 98: true, 99: true}); -}); - -test(function parserError() { - var boundary = 'abc', - buffer = new Buffer(5); - - parser.initWithBoundary(boundary); - buffer.write('--ad', 'ascii', 0); - assert.equal(parser.write(buffer), 3); -}); - -test(function end() { - (function testError() { - assert.equal(parser.end().message, 'MultipartParser.end(): stream ended unexpectedly: ' + parser.explain()); - })(); - - (function testRegular() { - parser.state = multipartParser.END; - assert.strictEqual(parser.end(), undefined); - })(); -}); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-querystring-parser.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-querystring-parser.js deleted file mode 100644 index 54d3e2d..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/simple/test-querystring-parser.js +++ /dev/null @@ -1,45 +0,0 @@ -var common = require('../common'); -var QuerystringParser = require(common.lib + '/querystring_parser').QuerystringParser, - Buffer = require('buffer').Buffer, - gently, - parser; - -function test(test) { - gently = new Gently(); - parser = new QuerystringParser(); - test(); - gently.verify(test.name); -} - -test(function constructor() { - assert.equal(parser.buffer, ''); - assert.equal(parser.constructor.name, 'QuerystringParser'); -}); - -test(function write() { - var a = new Buffer('a=1'); - assert.equal(parser.write(a), a.length); - - var b = new Buffer('&b=2'); - parser.write(b); - assert.equal(parser.buffer, a + b); -}); - -test(function end() { - var FIELDS = {a: ['b', {c: 'd'}], e: 'f'}; - - gently.expect(GENTLY.hijacked.querystring, 'parse', function(str) { - assert.equal(str, parser.buffer); - return FIELDS; - }); - - gently.expect(parser, 'onField', Object.keys(FIELDS).length, function(key, val) { - assert.deepEqual(FIELDS[key], val); - }); - - gently.expect(parser, 'onEnd'); - - parser.buffer = 'my buffer'; - parser.end(); - assert.equal(parser.buffer, ''); -}); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/system/test-multi-video-upload.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/system/test-multi-video-upload.js deleted file mode 100644 index 479e46d..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/legacy/system/test-multi-video-upload.js +++ /dev/null @@ -1,75 +0,0 @@ -var common = require('../common'); -var BOUNDARY = '---------------------------10102754414578508781458777923', - FIXTURE = TEST_FIXTURES+'/multi_video.upload', - fs = require('fs'), - util = require(common.lib + '/util'), - http = require('http'), - formidable = require(common.lib + '/index'), - server = http.createServer(); - -server.on('request', function(req, res) { - var form = new formidable.IncomingForm(), - uploads = {}; - - form.uploadDir = TEST_TMP; - form.hash = 'sha1'; - form.parse(req); - - form - .on('fileBegin', function(field, file) { - assert.equal(field, 'upload'); - - var tracker = {file: file, progress: [], ended: false}; - uploads[file.filename] = tracker; - file - .on('progress', function(bytesReceived) { - tracker.progress.push(bytesReceived); - assert.equal(bytesReceived, file.length); - }) - .on('end', function() { - tracker.ended = true; - }); - }) - .on('field', function(field, value) { - assert.equal(field, 'title'); - assert.equal(value, ''); - }) - .on('file', function(field, file) { - assert.equal(field, 'upload'); - assert.strictEqual(uploads[file.filename].file, file); - }) - .on('end', function() { - assert.ok(uploads['shortest_video.flv']); - assert.ok(uploads['shortest_video.flv'].ended); - assert.ok(uploads['shortest_video.flv'].progress.length > 3); - assert.equal(uploads['shortest_video.flv'].file.hash, 'd6a17616c7143d1b1438ceeef6836d1a09186b3a'); - assert.equal(uploads['shortest_video.flv'].progress.slice(-1), uploads['shortest_video.flv'].file.length); - assert.ok(uploads['shortest_video.mp4']); - assert.ok(uploads['shortest_video.mp4'].ended); - assert.ok(uploads['shortest_video.mp4'].progress.length > 3); - assert.equal(uploads['shortest_video.mp4'].file.hash, '937dfd4db263f4887ceae19341dcc8d63bcd557f'); - - server.close(); - res.writeHead(200); - res.end('good'); - }); -}); - -server.listen(TEST_PORT, function() { - var client = http.createClient(TEST_PORT), - stat = fs.statSync(FIXTURE), - headers = { - 'content-type': 'multipart/form-data; boundary='+BOUNDARY, - 'content-length': stat.size, - } - request = client.request('POST', '/', headers), - fixture = new fs.ReadStream(FIXTURE); - - fixture - .on('data', function(b) { - request.write(b); - }) - .on('end', function() { - request.end(); - }); -}); diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/run.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/run.js deleted file mode 100755 index 50b2361..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/run.js +++ /dev/null @@ -1,2 +0,0 @@ -#!/usr/bin/env node -require('urun')(__dirname) diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/test/unit/test-incoming-form.js b/node_modules/express/node_modules/connect/node_modules/formidable/test/unit/test-incoming-form.js deleted file mode 100644 index fe2ac1c..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/test/unit/test-incoming-form.js +++ /dev/null @@ -1,63 +0,0 @@ -var common = require('../common'); -var test = require('utest'); -var assert = common.assert; -var IncomingForm = common.require('incoming_form').IncomingForm; -var path = require('path'); - -var form; -test('IncomingForm', { - before: function() { - form = new IncomingForm(); - }, - - '#_fileName with regular characters': function() { - var filename = 'foo.txt'; - assert.equal(form._fileName(makeHeader(filename)), 'foo.txt'); - }, - - '#_fileName with unescaped quote': function() { - var filename = 'my".txt'; - assert.equal(form._fileName(makeHeader(filename)), 'my".txt'); - }, - - '#_fileName with escaped quote': function() { - var filename = 'my%22.txt'; - assert.equal(form._fileName(makeHeader(filename)), 'my".txt'); - }, - - '#_fileName with bad quote and additional sub-header': function() { - var filename = 'my".txt'; - var header = makeHeader(filename) + '; foo="bar"'; - assert.equal(form._fileName(header), filename); - }, - - '#_fileName with semicolon': function() { - var filename = 'my;.txt'; - assert.equal(form._fileName(makeHeader(filename)), 'my;.txt'); - }, - - '#_fileName with utf8 character': function() { - var filename = 'my☃.txt'; - assert.equal(form._fileName(makeHeader(filename)), 'my☃.txt'); - }, - - '#_uploadPath strips harmful characters from extension when keepExtensions': function() { - form.keepExtensions = true; - - var ext = path.extname(form._uploadPath('fine.jpg?foo=bar')); - assert.equal(ext, '.jpg'); - - var ext = path.extname(form._uploadPath('fine?foo=bar')); - assert.equal(ext, ''); - - var ext = path.extname(form._uploadPath('super.cr2+dsad')); - assert.equal(ext, '.cr2'); - - var ext = path.extname(form._uploadPath('super.bar')); - assert.equal(ext, '.bar'); - }, -}); - -function makeHeader(filename) { - return 'Content-Disposition: form-data; name="upload"; filename="' + filename + '"'; -} diff --git a/node_modules/express/node_modules/connect/node_modules/formidable/tool/record.js b/node_modules/express/node_modules/connect/node_modules/formidable/tool/record.js deleted file mode 100644 index 9f1cef8..0000000 --- a/node_modules/express/node_modules/connect/node_modules/formidable/tool/record.js +++ /dev/null @@ -1,47 +0,0 @@ -var http = require('http'); -var fs = require('fs'); -var connections = 0; - -var server = http.createServer(function(req, res) { - var socket = req.socket; - console.log('Request: %s %s -> %s', req.method, req.url, socket.filename); - - req.on('end', function() { - if (req.url !== '/') { - res.end(JSON.stringify({ - method: req.method, - url: req.url, - filename: socket.filename, - })); - return; - } - - res.writeHead(200, {'content-type': 'text/html'}); - res.end( - '
    '+ - '
    '+ - '
    '+ - ''+ - '
    ' - ); - }); -}); - -server.on('connection', function(socket) { - connections++; - - socket.id = connections; - socket.filename = 'connection-' + socket.id + '.http'; - socket.file = fs.createWriteStream(socket.filename); - socket.pipe(socket.file); - - console.log('--> %s', socket.filename); - socket.on('close', function() { - console.log('<-- %s', socket.filename); - }); -}); - -var port = process.env.PORT || 8080; -server.listen(port, function() { - console.log('Recording connections on port %s', port); -}); diff --git a/node_modules/express/node_modules/connect/node_modules/pause/.npmignore b/node_modules/express/node_modules/connect/node_modules/pause/.npmignore deleted file mode 100644 index f1250e5..0000000 --- a/node_modules/express/node_modules/connect/node_modules/pause/.npmignore +++ /dev/null @@ -1,4 +0,0 @@ -support -test -examples -*.sock diff --git a/node_modules/express/node_modules/connect/node_modules/pause/History.md b/node_modules/express/node_modules/connect/node_modules/pause/History.md deleted file mode 100644 index c8aa68f..0000000 --- a/node_modules/express/node_modules/connect/node_modules/pause/History.md +++ /dev/null @@ -1,5 +0,0 @@ - -0.0.1 / 2010-01-03 -================== - - * Initial release diff --git a/node_modules/express/node_modules/connect/node_modules/pause/Makefile b/node_modules/express/node_modules/connect/node_modules/pause/Makefile deleted file mode 100644 index 4e9c8d3..0000000 --- a/node_modules/express/node_modules/connect/node_modules/pause/Makefile +++ /dev/null @@ -1,7 +0,0 @@ - -test: - @./node_modules/.bin/mocha \ - --require should \ - --reporter spec - -.PHONY: test \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/pause/Readme.md b/node_modules/express/node_modules/connect/node_modules/pause/Readme.md deleted file mode 100644 index 1cdd68a..0000000 --- a/node_modules/express/node_modules/connect/node_modules/pause/Readme.md +++ /dev/null @@ -1,29 +0,0 @@ - -# pause - - Pause streams... - -## License - -(The MIT License) - -Copyright (c) 2012 TJ Holowaychuk <tj@vision-media.ca> - -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. \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/pause/index.js b/node_modules/express/node_modules/connect/node_modules/pause/index.js deleted file mode 100644 index 1b7b379..0000000 --- a/node_modules/express/node_modules/connect/node_modules/pause/index.js +++ /dev/null @@ -1,29 +0,0 @@ - -module.exports = function(obj){ - var onData - , onEnd - , events = []; - - // buffer data - obj.on('data', onData = function(data, encoding){ - events.push(['data', data, encoding]); - }); - - // buffer end - obj.on('end', onEnd = function(data, encoding){ - events.push(['end', data, encoding]); - }); - - return { - end: function(){ - obj.removeListener('data', onData); - obj.removeListener('end', onEnd); - }, - resume: function(){ - this.end(); - for (var i = 0, len = events.length; i < len; ++i) { - obj.emit.apply(obj, events[i]); - } - } - }; -}; \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/pause/package.json b/node_modules/express/node_modules/connect/node_modules/pause/package.json deleted file mode 100644 index 73cfe40..0000000 --- a/node_modules/express/node_modules/connect/node_modules/pause/package.json +++ /dev/null @@ -1,20 +0,0 @@ -{ - "name": "pause", - "version": "0.0.1", - "description": "Pause streams...", - "keywords": [], - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca" - }, - "dependencies": {}, - "devDependencies": { - "mocha": "*", - "should": "*" - }, - "main": "index", - "readme": "\n# pause\n\n Pause streams...\n\n## License \n\n(The MIT License)\n\nCopyright (c) 2012 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.", - "readmeFilename": "Readme.md", - "_id": "pause@0.0.1", - "_from": "pause@0.0.1" -} diff --git a/node_modules/express/node_modules/connect/node_modules/qs/.gitmodules b/node_modules/express/node_modules/connect/node_modules/qs/.gitmodules deleted file mode 100644 index 49e31da..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/.gitmodules +++ /dev/null @@ -1,6 +0,0 @@ -[submodule "support/expresso"] - path = support/expresso - url = git://github.com/visionmedia/expresso.git -[submodule "support/should"] - path = support/should - url = git://github.com/visionmedia/should.js.git diff --git a/node_modules/express/node_modules/connect/node_modules/qs/.npmignore b/node_modules/express/node_modules/connect/node_modules/qs/.npmignore deleted file mode 100644 index 3c3629e..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/.npmignore +++ /dev/null @@ -1 +0,0 @@ -node_modules diff --git a/node_modules/express/node_modules/connect/node_modules/qs/.travis.yml b/node_modules/express/node_modules/connect/node_modules/qs/.travis.yml deleted file mode 100644 index 2c0a8f6..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/.travis.yml +++ /dev/null @@ -1,4 +0,0 @@ -language: node_js -node_js: - - 0.6 - - 0.4 \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/History.md b/node_modules/express/node_modules/connect/node_modules/qs/History.md deleted file mode 100644 index 1feef45..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/History.md +++ /dev/null @@ -1,83 +0,0 @@ - -0.5.1 / 2012-09-18 -================== - - * fix encoded `=`. Closes #43 - -0.5.0 / 2012-05-04 -================== - - * Added component support - -0.4.2 / 2012-02-08 -================== - - * Fixed: ensure objects are created when appropriate not arrays [aheckmann] - -0.4.1 / 2012-01-26 -================== - - * Fixed stringify()ing numbers. Closes #23 - -0.4.0 / 2011-11-21 -================== - - * Allow parsing of an existing object (for `bodyParser()`) [jackyz] - * Replaced expresso with mocha - -0.3.2 / 2011-11-08 -================== - - * Fixed global variable leak - -0.3.1 / 2011-08-17 -================== - - * Added `try/catch` around malformed uri components - * Add test coverage for Array native method bleed-though - -0.3.0 / 2011-07-19 -================== - - * Allow `array[index]` and `object[property]` syntaxes [Aria Stewart] - -0.2.0 / 2011-06-29 -================== - - * Added `qs.stringify()` [Cory Forsyth] - -0.1.0 / 2011-04-13 -================== - - * Added jQuery-ish array support - -0.0.7 / 2011-03-13 -================== - - * Fixed; handle empty string and `== null` in `qs.parse()` [dmit] - allows for convenient `qs.parse(url.parse(str).query)` - -0.0.6 / 2011-02-14 -================== - - * Fixed; support for implicit arrays - -0.0.4 / 2011-02-09 -================== - - * Fixed `+` as a space - -0.0.3 / 2011-02-08 -================== - - * Fixed case when right-hand value contains "]" - -0.0.2 / 2011-02-07 -================== - - * Fixed "=" presence in key - -0.0.1 / 2011-02-07 -================== - - * Initial release \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/Makefile b/node_modules/express/node_modules/connect/node_modules/qs/Makefile deleted file mode 100644 index 0a21cf7..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/Makefile +++ /dev/null @@ -1,12 +0,0 @@ - -test/browser/qs.js: querystring.js - component build package.json test/browser/qs - -querystring.js: lib/head.js lib/querystring.js lib/tail.js - cat $^ > $@ - -test: - @./node_modules/.bin/mocha \ - --ui bdd - -.PHONY: test \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/Readme.md b/node_modules/express/node_modules/connect/node_modules/qs/Readme.md deleted file mode 100644 index 27e54a4..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/Readme.md +++ /dev/null @@ -1,58 +0,0 @@ -# node-querystring - - query string parser for node and the browser supporting nesting, as it was removed from `0.3.x`, so this library provides the previous and commonly desired behaviour (and twice as fast). Used by [express](http://expressjs.com), [connect](http://senchalabs.github.com/connect) and others. - -## Installation - - $ npm install qs - -## Examples - -```js -var qs = require('qs'); - -qs.parse('user[name][first]=Tobi&user[email]=tobi@learnboost.com'); -// => { user: { name: { first: 'Tobi' }, email: 'tobi@learnboost.com' } } - -qs.stringify({ user: { name: 'Tobi', email: 'tobi@learnboost.com' }}) -// => user[name]=Tobi&user[email]=tobi%40learnboost.com -``` - -## Testing - -Install dev dependencies: - - $ npm install -d - -and execute: - - $ make test - -browser: - - $ open test/browser/index.html - -## License - -(The MIT License) - -Copyright (c) 2010 TJ Holowaychuk <tj@vision-media.ca> - -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. \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/benchmark.js b/node_modules/express/node_modules/connect/node_modules/qs/benchmark.js deleted file mode 100644 index 97e2c93..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/benchmark.js +++ /dev/null @@ -1,17 +0,0 @@ - -var qs = require('./'); - -var times = 100000 - , start = new Date - , n = times; - -console.log('times: %d', times); - -while (n--) qs.parse('foo=bar'); -console.log('simple: %dms', new Date - start); - -var start = new Date - , n = times; - -while (n--) qs.parse('user[name][first]=tj&user[name][last]=holowaychuk'); -console.log('nested: %dms', new Date - start); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/component.json b/node_modules/express/node_modules/connect/node_modules/qs/component.json deleted file mode 100644 index ba34ead..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/component.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "name": "querystring", - "description": "Querystring parser / stringifier with nesting support", - "keywords": ["querystring", "query", "parser"], - "main": "lib/querystring.js" -} \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/examples.js b/node_modules/express/node_modules/connect/node_modules/qs/examples.js deleted file mode 100644 index 27617b7..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/examples.js +++ /dev/null @@ -1,51 +0,0 @@ - -/** - * Module dependencies. - */ - -var qs = require('./'); - -var obj = qs.parse('foo'); -console.log(obj) - -var obj = qs.parse('foo=bar=baz'); -console.log(obj) - -var obj = qs.parse('users[]'); -console.log(obj) - -var obj = qs.parse('name=tj&email=tj@vision-media.ca'); -console.log(obj) - -var obj = qs.parse('users[]=tj&users[]=tobi&users[]=jane'); -console.log(obj) - -var obj = qs.parse('user[name][first]=tj&user[name][last]=holowaychuk'); -console.log(obj) - -var obj = qs.parse('users[][name][first]=tj&users[][name][last]=holowaychuk'); -console.log(obj) - -var obj = qs.parse('a=a&a=b&a=c'); -console.log(obj) - -var obj = qs.parse('user[tj]=tj&user[tj]=TJ'); -console.log(obj) - -var obj = qs.parse('user[names]=tj&user[names]=TJ&user[names]=Tyler'); -console.log(obj) - -var obj = qs.parse('user[name][first]=tj&user[name][first]=TJ'); -console.log(obj) - -var obj = qs.parse('user[0]=tj&user[1]=TJ'); -console.log(obj) - -var obj = qs.parse('user[0]=tj&user[]=TJ'); -console.log(obj) - -var obj = qs.parse('user[0]=tj&user[foo]=TJ'); -console.log(obj) - -var str = qs.stringify({ user: { name: 'Tobi', email: 'tobi@learnboost.com' }}); -console.log(str); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/index.js b/node_modules/express/node_modules/connect/node_modules/qs/index.js deleted file mode 100644 index d177d20..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/index.js +++ /dev/null @@ -1,2 +0,0 @@ - -module.exports = require('./lib/querystring'); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/lib/head.js b/node_modules/express/node_modules/connect/node_modules/qs/lib/head.js deleted file mode 100644 index 55d3817..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/lib/head.js +++ /dev/null @@ -1 +0,0 @@ -;(function(){ diff --git a/node_modules/express/node_modules/connect/node_modules/qs/lib/querystring.js b/node_modules/express/node_modules/connect/node_modules/qs/lib/querystring.js deleted file mode 100644 index d3689bb..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/lib/querystring.js +++ /dev/null @@ -1,262 +0,0 @@ - -/** - * Object#toString() ref for stringify(). - */ - -var toString = Object.prototype.toString; - -/** - * Cache non-integer test regexp. - */ - -var isint = /^[0-9]+$/; - -function promote(parent, key) { - if (parent[key].length == 0) return parent[key] = {}; - var t = {}; - for (var i in parent[key]) t[i] = parent[key][i]; - parent[key] = t; - return t; -} - -function parse(parts, parent, key, val) { - var part = parts.shift(); - // end - if (!part) { - if (Array.isArray(parent[key])) { - parent[key].push(val); - } else if ('object' == typeof parent[key]) { - parent[key] = val; - } else if ('undefined' == typeof parent[key]) { - parent[key] = val; - } else { - parent[key] = [parent[key], val]; - } - // array - } else { - var obj = parent[key] = parent[key] || []; - if (']' == part) { - if (Array.isArray(obj)) { - if ('' != val) obj.push(val); - } else if ('object' == typeof obj) { - obj[Object.keys(obj).length] = val; - } else { - obj = parent[key] = [parent[key], val]; - } - // prop - } else if (~part.indexOf(']')) { - part = part.substr(0, part.length - 1); - if (!isint.test(part) && Array.isArray(obj)) obj = promote(parent, key); - parse(parts, obj, part, val); - // key - } else { - if (!isint.test(part) && Array.isArray(obj)) obj = promote(parent, key); - parse(parts, obj, part, val); - } - } -} - -/** - * Merge parent key/val pair. - */ - -function merge(parent, key, val){ - if (~key.indexOf(']')) { - var parts = key.split('[') - , len = parts.length - , last = len - 1; - parse(parts, parent, 'base', val); - // optimize - } else { - if (!isint.test(key) && Array.isArray(parent.base)) { - var t = {}; - for (var k in parent.base) t[k] = parent.base[k]; - parent.base = t; - } - set(parent.base, key, val); - } - - return parent; -} - -/** - * Parse the given obj. - */ - -function parseObject(obj){ - var ret = { base: {} }; - Object.keys(obj).forEach(function(name){ - merge(ret, name, obj[name]); - }); - return ret.base; -} - -/** - * Parse the given str. - */ - -function parseString(str){ - return String(str) - .split('&') - .reduce(function(ret, pair){ - var eql = pair.indexOf('=') - , brace = lastBraceInKey(pair) - , key = pair.substr(0, brace || eql) - , val = pair.substr(brace || eql, pair.length) - , val = val.substr(val.indexOf('=') + 1, val.length); - - // ?foo - if ('' == key) key = pair, val = ''; - - return merge(ret, decode(key), decode(val)); - }, { base: {} }).base; -} - -/** - * Parse the given query `str` or `obj`, returning an object. - * - * @param {String} str | {Object} obj - * @return {Object} - * @api public - */ - -exports.parse = function(str){ - if (null == str || '' == str) return {}; - return 'object' == typeof str - ? parseObject(str) - : parseString(str); -}; - -/** - * Turn the given `obj` into a query string - * - * @param {Object} obj - * @return {String} - * @api public - */ - -var stringify = exports.stringify = function(obj, prefix) { - if (Array.isArray(obj)) { - return stringifyArray(obj, prefix); - } else if ('[object Object]' == toString.call(obj)) { - return stringifyObject(obj, prefix); - } else if ('string' == typeof obj) { - return stringifyString(obj, prefix); - } else { - return prefix + '=' + obj; - } -}; - -/** - * Stringify the given `str`. - * - * @param {String} str - * @param {String} prefix - * @return {String} - * @api private - */ - -function stringifyString(str, prefix) { - if (!prefix) throw new TypeError('stringify expects an object'); - return prefix + '=' + encodeURIComponent(str); -} - -/** - * Stringify the given `arr`. - * - * @param {Array} arr - * @param {String} prefix - * @return {String} - * @api private - */ - -function stringifyArray(arr, prefix) { - var ret = []; - if (!prefix) throw new TypeError('stringify expects an object'); - for (var i = 0; i < arr.length; i++) { - ret.push(stringify(arr[i], prefix + '['+i+']')); - } - return ret.join('&'); -} - -/** - * Stringify the given `obj`. - * - * @param {Object} obj - * @param {String} prefix - * @return {String} - * @api private - */ - -function stringifyObject(obj, prefix) { - var ret = [] - , keys = Object.keys(obj) - , key; - - for (var i = 0, len = keys.length; i < len; ++i) { - key = keys[i]; - ret.push(stringify(obj[key], prefix - ? prefix + '[' + encodeURIComponent(key) + ']' - : encodeURIComponent(key))); - } - - return ret.join('&'); -} - -/** - * Set `obj`'s `key` to `val` respecting - * the weird and wonderful syntax of a qs, - * where "foo=bar&foo=baz" becomes an array. - * - * @param {Object} obj - * @param {String} key - * @param {String} val - * @api private - */ - -function set(obj, key, val) { - var v = obj[key]; - if (undefined === v) { - obj[key] = val; - } else if (Array.isArray(v)) { - v.push(val); - } else { - obj[key] = [v, val]; - } -} - -/** - * Locate last brace in `str` within the key. - * - * @param {String} str - * @return {Number} - * @api private - */ - -function lastBraceInKey(str) { - var len = str.length - , brace - , c; - for (var i = 0; i < len; ++i) { - c = str[i]; - if (']' == c) brace = false; - if ('[' == c) brace = true; - if ('=' == c && !brace) return i; - } -} - -/** - * Decode `str`. - * - * @param {String} str - * @return {String} - * @api private - */ - -function decode(str) { - try { - return decodeURIComponent(str.replace(/\+/g, ' ')); - } catch (err) { - return str; - } -} diff --git a/node_modules/express/node_modules/connect/node_modules/qs/lib/tail.js b/node_modules/express/node_modules/connect/node_modules/qs/lib/tail.js deleted file mode 100644 index 158693a..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/lib/tail.js +++ /dev/null @@ -1 +0,0 @@ -})(); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/package.json b/node_modules/express/node_modules/connect/node_modules/qs/package.json deleted file mode 100644 index b0c394b..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/package.json +++ /dev/null @@ -1,39 +0,0 @@ -{ - "name": "qs", - "description": "querystring parser", - "version": "0.5.1", - "keywords": [ - "query string", - "parser", - "component" - ], - "repository": { - "type": "git", - "url": "git://github.com/visionmedia/node-querystring.git" - }, - "devDependencies": { - "mocha": "*", - "expect.js": "*" - }, - "component": { - "scripts": { - "querystring": "querystring.js" - } - }, - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca", - "url": "http://tjholowaychuk.com" - }, - "main": "index", - "engines": { - "node": "*" - }, - "readme": "# node-querystring\n\n query string parser for node and the browser supporting nesting, as it was removed from `0.3.x`, so this library provides the previous and commonly desired behaviour (and twice as fast). Used by [express](http://expressjs.com), [connect](http://senchalabs.github.com/connect) and others.\n\n## Installation\n\n $ npm install qs\n\n## Examples\n\n```js\nvar qs = require('qs');\n\nqs.parse('user[name][first]=Tobi&user[email]=tobi@learnboost.com');\n// => { user: { name: { first: 'Tobi' }, email: 'tobi@learnboost.com' } }\n\nqs.stringify({ user: { name: 'Tobi', email: 'tobi@learnboost.com' }})\n// => user[name]=Tobi&user[email]=tobi%40learnboost.com\n```\n\n## Testing\n\nInstall dev dependencies:\n\n $ npm install -d\n\nand execute:\n\n $ make test\n\nbrowser:\n\n $ open test/browser/index.html\n\n## License \n\n(The MIT License)\n\nCopyright (c) 2010 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.", - "readmeFilename": "Readme.md", - "bugs": { - "url": "https://github.com/visionmedia/node-querystring/issues" - }, - "_id": "qs@0.5.1", - "_from": "qs@0.5.1" -} diff --git a/node_modules/express/node_modules/connect/node_modules/qs/querystring.js b/node_modules/express/node_modules/connect/node_modules/qs/querystring.js deleted file mode 100644 index 7466b06..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/querystring.js +++ /dev/null @@ -1,254 +0,0 @@ -;(function(){ - -/** - * Object#toString() ref for stringify(). - */ - -var toString = Object.prototype.toString; - -/** - * Cache non-integer test regexp. - */ - -var isint = /^[0-9]+$/; - -function promote(parent, key) { - if (parent[key].length == 0) return parent[key] = {}; - var t = {}; - for (var i in parent[key]) t[i] = parent[key][i]; - parent[key] = t; - return t; -} - -function parse(parts, parent, key, val) { - var part = parts.shift(); - // end - if (!part) { - if (Array.isArray(parent[key])) { - parent[key].push(val); - } else if ('object' == typeof parent[key]) { - parent[key] = val; - } else if ('undefined' == typeof parent[key]) { - parent[key] = val; - } else { - parent[key] = [parent[key], val]; - } - // array - } else { - var obj = parent[key] = parent[key] || []; - if (']' == part) { - if (Array.isArray(obj)) { - if ('' != val) obj.push(val); - } else if ('object' == typeof obj) { - obj[Object.keys(obj).length] = val; - } else { - obj = parent[key] = [parent[key], val]; - } - // prop - } else if (~part.indexOf(']')) { - part = part.substr(0, part.length - 1); - if (!isint.test(part) && Array.isArray(obj)) obj = promote(parent, key); - parse(parts, obj, part, val); - // key - } else { - if (!isint.test(part) && Array.isArray(obj)) obj = promote(parent, key); - parse(parts, obj, part, val); - } - } -} - -/** - * Merge parent key/val pair. - */ - -function merge(parent, key, val){ - if (~key.indexOf(']')) { - var parts = key.split('[') - , len = parts.length - , last = len - 1; - parse(parts, parent, 'base', val); - // optimize - } else { - if (!isint.test(key) && Array.isArray(parent.base)) { - var t = {}; - for (var k in parent.base) t[k] = parent.base[k]; - parent.base = t; - } - set(parent.base, key, val); - } - - return parent; -} - -/** - * Parse the given obj. - */ - -function parseObject(obj){ - var ret = { base: {} }; - Object.keys(obj).forEach(function(name){ - merge(ret, name, obj[name]); - }); - return ret.base; -} - -/** - * Parse the given str. - */ - -function parseString(str){ - return String(str) - .split('&') - .reduce(function(ret, pair){ - try{ - pair = decodeURIComponent(pair.replace(/\+/g, ' ')); - } catch(e) { - // ignore - } - - var eql = pair.indexOf('=') - , brace = lastBraceInKey(pair) - , key = pair.substr(0, brace || eql) - , val = pair.substr(brace || eql, pair.length) - , val = val.substr(val.indexOf('=') + 1, val.length); - - // ?foo - if ('' == key) key = pair, val = ''; - - return merge(ret, key, val); - }, { base: {} }).base; -} - -/** - * Parse the given query `str` or `obj`, returning an object. - * - * @param {String} str | {Object} obj - * @return {Object} - * @api public - */ - -exports.parse = function(str){ - if (null == str || '' == str) return {}; - return 'object' == typeof str - ? parseObject(str) - : parseString(str); -}; - -/** - * Turn the given `obj` into a query string - * - * @param {Object} obj - * @return {String} - * @api public - */ - -var stringify = exports.stringify = function(obj, prefix) { - if (Array.isArray(obj)) { - return stringifyArray(obj, prefix); - } else if ('[object Object]' == toString.call(obj)) { - return stringifyObject(obj, prefix); - } else if ('string' == typeof obj) { - return stringifyString(obj, prefix); - } else { - return prefix + '=' + obj; - } -}; - -/** - * Stringify the given `str`. - * - * @param {String} str - * @param {String} prefix - * @return {String} - * @api private - */ - -function stringifyString(str, prefix) { - if (!prefix) throw new TypeError('stringify expects an object'); - return prefix + '=' + encodeURIComponent(str); -} - -/** - * Stringify the given `arr`. - * - * @param {Array} arr - * @param {String} prefix - * @return {String} - * @api private - */ - -function stringifyArray(arr, prefix) { - var ret = []; - if (!prefix) throw new TypeError('stringify expects an object'); - for (var i = 0; i < arr.length; i++) { - ret.push(stringify(arr[i], prefix + '['+i+']')); - } - return ret.join('&'); -} - -/** - * Stringify the given `obj`. - * - * @param {Object} obj - * @param {String} prefix - * @return {String} - * @api private - */ - -function stringifyObject(obj, prefix) { - var ret = [] - , keys = Object.keys(obj) - , key; - - for (var i = 0, len = keys.length; i < len; ++i) { - key = keys[i]; - ret.push(stringify(obj[key], prefix - ? prefix + '[' + encodeURIComponent(key) + ']' - : encodeURIComponent(key))); - } - - return ret.join('&'); -} - -/** - * Set `obj`'s `key` to `val` respecting - * the weird and wonderful syntax of a qs, - * where "foo=bar&foo=baz" becomes an array. - * - * @param {Object} obj - * @param {String} key - * @param {String} val - * @api private - */ - -function set(obj, key, val) { - var v = obj[key]; - if (undefined === v) { - obj[key] = val; - } else if (Array.isArray(v)) { - v.push(val); - } else { - obj[key] = [v, val]; - } -} - -/** - * Locate last brace in `str` within the key. - * - * @param {String} str - * @return {Number} - * @api private - */ - -function lastBraceInKey(str) { - var len = str.length - , brace - , c; - for (var i = 0; i < len; ++i) { - c = str[i]; - if (']' == c) brace = false; - if ('[' == c) brace = true; - if ('=' == c && !brace) return i; - } -} -})(); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/test/browser/expect.js b/node_modules/express/node_modules/connect/node_modules/qs/test/browser/expect.js deleted file mode 100644 index 76aa4e8..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/test/browser/expect.js +++ /dev/null @@ -1,1202 +0,0 @@ - -(function (global, module) { - - if ('undefined' == typeof module) { - var module = { exports: {} } - , exports = module.exports - } - - /** - * Exports. - */ - - module.exports = expect; - expect.Assertion = Assertion; - - /** - * Exports version. - */ - - expect.version = '0.1.2'; - - /** - * Possible assertion flags. - */ - - var flags = { - not: ['to', 'be', 'have', 'include', 'only'] - , to: ['be', 'have', 'include', 'only', 'not'] - , only: ['have'] - , have: ['own'] - , be: ['an'] - }; - - function expect (obj) { - return new Assertion(obj); - } - - /** - * Constructor - * - * @api private - */ - - function Assertion (obj, flag, parent) { - this.obj = obj; - this.flags = {}; - - if (undefined != parent) { - this.flags[flag] = true; - - for (var i in parent.flags) { - if (parent.flags.hasOwnProperty(i)) { - this.flags[i] = true; - } - } - } - - var $flags = flag ? flags[flag] : keys(flags) - , self = this - - if ($flags) { - for (var i = 0, l = $flags.length; i < l; i++) { - // avoid recursion - if (this.flags[$flags[i]]) continue; - - var name = $flags[i] - , assertion = new Assertion(this.obj, name, this) - - if ('function' == typeof Assertion.prototype[name]) { - // clone the function, make sure we dont touch the prot reference - var old = this[name]; - this[name] = function () { - return old.apply(self, arguments); - } - - for (var fn in Assertion.prototype) { - if (Assertion.prototype.hasOwnProperty(fn) && fn != name) { - this[name][fn] = bind(assertion[fn], assertion); - } - } - } else { - this[name] = assertion; - } - } - } - }; - - /** - * Performs an assertion - * - * @api private - */ - - Assertion.prototype.assert = function (truth, msg, error) { - var msg = this.flags.not ? error : msg - , ok = this.flags.not ? !truth : truth; - - if (!ok) { - throw new Error(msg); - } - - this.and = new Assertion(this.obj); - }; - - /** - * Check if the value is truthy - * - * @api public - */ - - Assertion.prototype.ok = function () { - this.assert( - !!this.obj - , 'expected ' + i(this.obj) + ' to be truthy' - , 'expected ' + i(this.obj) + ' to be falsy'); - }; - - /** - * Assert that the function throws. - * - * @param {Function|RegExp} callback, or regexp to match error string against - * @api public - */ - - Assertion.prototype.throwError = - Assertion.prototype.throwException = function (fn) { - expect(this.obj).to.be.a('function'); - - var thrown = false - , not = this.flags.not - - try { - this.obj(); - } catch (e) { - if ('function' == typeof fn) { - fn(e); - } else if ('object' == typeof fn) { - var subject = 'string' == typeof e ? e : e.message; - if (not) { - expect(subject).to.not.match(fn); - } else { - expect(subject).to.match(fn); - } - } - thrown = true; - } - - if ('object' == typeof fn && not) { - // in the presence of a matcher, ensure the `not` only applies to - // the matching. - this.flags.not = false; - } - - var name = this.obj.name || 'fn'; - this.assert( - thrown - , 'expected ' + name + ' to throw an exception' - , 'expected ' + name + ' not to throw an exception'); - }; - - /** - * Checks if the array is empty. - * - * @api public - */ - - Assertion.prototype.empty = function () { - var expectation; - - if ('object' == typeof this.obj && null !== this.obj && !isArray(this.obj)) { - if ('number' == typeof this.obj.length) { - expectation = !this.obj.length; - } else { - expectation = !keys(this.obj).length; - } - } else { - if ('string' != typeof this.obj) { - expect(this.obj).to.be.an('object'); - } - - expect(this.obj).to.have.property('length'); - expectation = !this.obj.length; - } - - this.assert( - expectation - , 'expected ' + i(this.obj) + ' to be empty' - , 'expected ' + i(this.obj) + ' to not be empty'); - return this; - }; - - /** - * Checks if the obj exactly equals another. - * - * @api public - */ - - Assertion.prototype.be = - Assertion.prototype.equal = function (obj) { - this.assert( - obj === this.obj - , 'expected ' + i(this.obj) + ' to equal ' + i(obj) - , 'expected ' + i(this.obj) + ' to not equal ' + i(obj)); - return this; - }; - - /** - * Checks if the obj sortof equals another. - * - * @api public - */ - - Assertion.prototype.eql = function (obj) { - this.assert( - expect.eql(obj, this.obj) - , 'expected ' + i(this.obj) + ' to sort of equal ' + i(obj) - , 'expected ' + i(this.obj) + ' to sort of not equal ' + i(obj)); - return this; - }; - - /** - * Assert within start to finish (inclusive). - * - * @param {Number} start - * @param {Number} finish - * @api public - */ - - Assertion.prototype.within = function (start, finish) { - var range = start + '..' + finish; - this.assert( - this.obj >= start && this.obj <= finish - , 'expected ' + i(this.obj) + ' to be within ' + range - , 'expected ' + i(this.obj) + ' to not be within ' + range); - return this; - }; - - /** - * Assert typeof / instance of - * - * @api public - */ - - Assertion.prototype.a = - Assertion.prototype.an = function (type) { - if ('string' == typeof type) { - // proper english in error msg - var n = /^[aeiou]/.test(type) ? 'n' : ''; - - // typeof with support for 'array' - this.assert( - 'array' == type ? isArray(this.obj) : - 'object' == type - ? 'object' == typeof this.obj && null !== this.obj - : type == typeof this.obj - , 'expected ' + i(this.obj) + ' to be a' + n + ' ' + type - , 'expected ' + i(this.obj) + ' not to be a' + n + ' ' + type); - } else { - // instanceof - var name = type.name || 'supplied constructor'; - this.assert( - this.obj instanceof type - , 'expected ' + i(this.obj) + ' to be an instance of ' + name - , 'expected ' + i(this.obj) + ' not to be an instance of ' + name); - } - - return this; - }; - - /** - * Assert numeric value above _n_. - * - * @param {Number} n - * @api public - */ - - Assertion.prototype.greaterThan = - Assertion.prototype.above = function (n) { - this.assert( - this.obj > n - , 'expected ' + i(this.obj) + ' to be above ' + n - , 'expected ' + i(this.obj) + ' to be below ' + n); - return this; - }; - - /** - * Assert numeric value below _n_. - * - * @param {Number} n - * @api public - */ - - Assertion.prototype.lessThan = - Assertion.prototype.below = function (n) { - this.assert( - this.obj < n - , 'expected ' + i(this.obj) + ' to be below ' + n - , 'expected ' + i(this.obj) + ' to be above ' + n); - return this; - }; - - /** - * Assert string value matches _regexp_. - * - * @param {RegExp} regexp - * @api public - */ - - Assertion.prototype.match = function (regexp) { - this.assert( - regexp.exec(this.obj) - , 'expected ' + i(this.obj) + ' to match ' + regexp - , 'expected ' + i(this.obj) + ' not to match ' + regexp); - return this; - }; - - /** - * Assert property "length" exists and has value of _n_. - * - * @param {Number} n - * @api public - */ - - Assertion.prototype.length = function (n) { - expect(this.obj).to.have.property('length'); - var len = this.obj.length; - this.assert( - n == len - , 'expected ' + i(this.obj) + ' to have a length of ' + n + ' but got ' + len - , 'expected ' + i(this.obj) + ' to not have a length of ' + len); - return this; - }; - - /** - * Assert property _name_ exists, with optional _val_. - * - * @param {String} name - * @param {Mixed} val - * @api public - */ - - Assertion.prototype.property = function (name, val) { - if (this.flags.own) { - this.assert( - Object.prototype.hasOwnProperty.call(this.obj, name) - , 'expected ' + i(this.obj) + ' to have own property ' + i(name) - , 'expected ' + i(this.obj) + ' to not have own property ' + i(name)); - return this; - } - - if (this.flags.not && undefined !== val) { - if (undefined === this.obj[name]) { - throw new Error(i(this.obj) + ' has no property ' + i(name)); - } - } else { - var hasProp; - try { - hasProp = name in this.obj - } catch (e) { - hasProp = undefined !== this.obj[name] - } - - this.assert( - hasProp - , 'expected ' + i(this.obj) + ' to have a property ' + i(name) - , 'expected ' + i(this.obj) + ' to not have a property ' + i(name)); - } - - if (undefined !== val) { - this.assert( - val === this.obj[name] - , 'expected ' + i(this.obj) + ' to have a property ' + i(name) - + ' of ' + i(val) + ', but got ' + i(this.obj[name]) - , 'expected ' + i(this.obj) + ' to not have a property ' + i(name) - + ' of ' + i(val)); - } - - this.obj = this.obj[name]; - return this; - }; - - /** - * Assert that the array contains _obj_ or string contains _obj_. - * - * @param {Mixed} obj|string - * @api public - */ - - Assertion.prototype.string = - Assertion.prototype.contain = function (obj) { - if ('string' == typeof this.obj) { - this.assert( - ~this.obj.indexOf(obj) - , 'expected ' + i(this.obj) + ' to contain ' + i(obj) - , 'expected ' + i(this.obj) + ' to not contain ' + i(obj)); - } else { - this.assert( - ~indexOf(this.obj, obj) - , 'expected ' + i(this.obj) + ' to contain ' + i(obj) - , 'expected ' + i(this.obj) + ' to not contain ' + i(obj)); - } - return this; - }; - - /** - * Assert exact keys or inclusion of keys by using - * the `.own` modifier. - * - * @param {Array|String ...} keys - * @api public - */ - - Assertion.prototype.key = - Assertion.prototype.keys = function ($keys) { - var str - , ok = true; - - $keys = isArray($keys) - ? $keys - : Array.prototype.slice.call(arguments); - - if (!$keys.length) throw new Error('keys required'); - - var actual = keys(this.obj) - , len = $keys.length; - - // Inclusion - ok = every($keys, function (key) { - return ~indexOf(actual, key); - }); - - // Strict - if (!this.flags.not && this.flags.only) { - ok = ok && $keys.length == actual.length; - } - - // Key string - if (len > 1) { - $keys = map($keys, function (key) { - return i(key); - }); - var last = $keys.pop(); - str = $keys.join(', ') + ', and ' + last; - } else { - str = i($keys[0]); - } - - // Form - str = (len > 1 ? 'keys ' : 'key ') + str; - - // Have / include - str = (!this.flags.only ? 'include ' : 'only have ') + str; - - // Assertion - this.assert( - ok - , 'expected ' + i(this.obj) + ' to ' + str - , 'expected ' + i(this.obj) + ' to not ' + str); - - return this; - }; - - /** - * Function bind implementation. - */ - - function bind (fn, scope) { - return function () { - return fn.apply(scope, arguments); - } - } - - /** - * Array every compatibility - * - * @see bit.ly/5Fq1N2 - * @api public - */ - - function every (arr, fn, thisObj) { - var scope = thisObj || global; - for (var i = 0, j = arr.length; i < j; ++i) { - if (!fn.call(scope, arr[i], i, arr)) { - return false; - } - } - return true; - }; - - /** - * Array indexOf compatibility. - * - * @see bit.ly/a5Dxa2 - * @api public - */ - - function indexOf (arr, o, i) { - if (Array.prototype.indexOf) { - return Array.prototype.indexOf.call(arr, o, i); - } - - if (arr.length === undefined) { - return -1; - } - - for (var j = arr.length, i = i < 0 ? i + j < 0 ? 0 : i + j : i || 0 - ; i < j && arr[i] !== o; i++); - - return j <= i ? -1 : i; - }; - - /** - * Inspects an object. - * - * @see taken from node.js `util` module (copyright Joyent, MIT license) - * @api private - */ - - function i (obj, showHidden, depth) { - var seen = []; - - function stylize (str) { - return str; - }; - - function format (value, recurseTimes) { - // Provide a hook for user-specified inspect functions. - // Check that value is an object with an inspect function on it - if (value && typeof value.inspect === 'function' && - // Filter out the util module, it's inspect function is special - value !== exports && - // Also filter out any prototype objects using the circular check. - !(value.constructor && value.constructor.prototype === value)) { - return value.inspect(recurseTimes); - } - - // Primitive types cannot have properties - switch (typeof value) { - case 'undefined': - return stylize('undefined', 'undefined'); - - case 'string': - var simple = '\'' + json.stringify(value).replace(/^"|"$/g, '') - .replace(/'/g, "\\'") - .replace(/\\"/g, '"') + '\''; - return stylize(simple, 'string'); - - case 'number': - return stylize('' + value, 'number'); - - case 'boolean': - return stylize('' + value, 'boolean'); - } - // For some reason typeof null is "object", so special case here. - if (value === null) { - return stylize('null', 'null'); - } - - // Look up the keys of the object. - var visible_keys = keys(value); - var $keys = showHidden ? Object.getOwnPropertyNames(value) : visible_keys; - - // Functions without properties can be shortcutted. - if (typeof value === 'function' && $keys.length === 0) { - if (isRegExp(value)) { - return stylize('' + value, 'regexp'); - } else { - var name = value.name ? ': ' + value.name : ''; - return stylize('[Function' + name + ']', 'special'); - } - } - - // Dates without properties can be shortcutted - if (isDate(value) && $keys.length === 0) { - return stylize(value.toUTCString(), 'date'); - } - - var base, type, braces; - // Determine the object type - if (isArray(value)) { - type = 'Array'; - braces = ['[', ']']; - } else { - type = 'Object'; - braces = ['{', '}']; - } - - // Make functions say that they are functions - if (typeof value === 'function') { - var n = value.name ? ': ' + value.name : ''; - base = (isRegExp(value)) ? ' ' + value : ' [Function' + n + ']'; - } else { - base = ''; - } - - // Make dates with properties first say the date - if (isDate(value)) { - base = ' ' + value.toUTCString(); - } - - if ($keys.length === 0) { - return braces[0] + base + braces[1]; - } - - if (recurseTimes < 0) { - if (isRegExp(value)) { - return stylize('' + value, 'regexp'); - } else { - return stylize('[Object]', 'special'); - } - } - - seen.push(value); - - var output = map($keys, function (key) { - var name, str; - if (value.__lookupGetter__) { - if (value.__lookupGetter__(key)) { - if (value.__lookupSetter__(key)) { - str = stylize('[Getter/Setter]', 'special'); - } else { - str = stylize('[Getter]', 'special'); - } - } else { - if (value.__lookupSetter__(key)) { - str = stylize('[Setter]', 'special'); - } - } - } - if (indexOf(visible_keys, key) < 0) { - name = '[' + key + ']'; - } - if (!str) { - if (indexOf(seen, value[key]) < 0) { - if (recurseTimes === null) { - str = format(value[key]); - } else { - str = format(value[key], recurseTimes - 1); - } - if (str.indexOf('\n') > -1) { - if (isArray(value)) { - str = map(str.split('\n'), function (line) { - return ' ' + line; - }).join('\n').substr(2); - } else { - str = '\n' + map(str.split('\n'), function (line) { - return ' ' + line; - }).join('\n'); - } - } - } else { - str = stylize('[Circular]', 'special'); - } - } - if (typeof name === 'undefined') { - if (type === 'Array' && key.match(/^\d+$/)) { - return str; - } - name = json.stringify('' + key); - if (name.match(/^"([a-zA-Z_][a-zA-Z_0-9]*)"$/)) { - name = name.substr(1, name.length - 2); - name = stylize(name, 'name'); - } else { - name = name.replace(/'/g, "\\'") - .replace(/\\"/g, '"') - .replace(/(^"|"$)/g, "'"); - name = stylize(name, 'string'); - } - } - - return name + ': ' + str; - }); - - seen.pop(); - - var numLinesEst = 0; - var length = reduce(output, function (prev, cur) { - numLinesEst++; - if (indexOf(cur, '\n') >= 0) numLinesEst++; - return prev + cur.length + 1; - }, 0); - - if (length > 50) { - output = braces[0] + - (base === '' ? '' : base + '\n ') + - ' ' + - output.join(',\n ') + - ' ' + - braces[1]; - - } else { - output = braces[0] + base + ' ' + output.join(', ') + ' ' + braces[1]; - } - - return output; - } - return format(obj, (typeof depth === 'undefined' ? 2 : depth)); - }; - - function isArray (ar) { - return Object.prototype.toString.call(ar) == '[object Array]'; - }; - - function isRegExp(re) { - var s = '' + re; - return re instanceof RegExp || // easy case - // duck-type for context-switching evalcx case - typeof(re) === 'function' && - re.constructor.name === 'RegExp' && - re.compile && - re.test && - re.exec && - s.match(/^\/.*\/[gim]{0,3}$/); - }; - - function isDate(d) { - if (d instanceof Date) return true; - return false; - }; - - function keys (obj) { - if (Object.keys) { - return Object.keys(obj); - } - - var keys = []; - - for (var i in obj) { - if (Object.prototype.hasOwnProperty.call(obj, i)) { - keys.push(i); - } - } - - return keys; - } - - function map (arr, mapper, that) { - if (Array.prototype.map) { - return Array.prototype.map.call(arr, mapper, that); - } - - var other= new Array(arr.length); - - for (var i= 0, n = arr.length; i= 2) { - var rv = arguments[1]; - } else { - do { - if (i in this) { - rv = this[i++]; - break; - } - - // if array contains no values, no initial value to return - if (++i >= len) - throw new TypeError(); - } while (true); - } - - for (; i < len; i++) { - if (i in this) - rv = fun.call(null, rv, this[i], i, this); - } - - return rv; - }; - - /** - * Asserts deep equality - * - * @see taken from node.js `assert` module (copyright Joyent, MIT license) - * @api private - */ - - expect.eql = function eql (actual, expected) { - // 7.1. All identical values are equivalent, as determined by ===. - if (actual === expected) { - return true; - } else if ('undefined' != typeof Buffer - && Buffer.isBuffer(actual) && Buffer.isBuffer(expected)) { - if (actual.length != expected.length) return false; - - for (var i = 0; i < actual.length; i++) { - if (actual[i] !== expected[i]) return false; - } - - return true; - - // 7.2. If the expected value is a Date object, the actual value is - // equivalent if it is also a Date object that refers to the same time. - } else if (actual instanceof Date && expected instanceof Date) { - return actual.getTime() === expected.getTime(); - - // 7.3. Other pairs that do not both pass typeof value == "object", - // equivalence is determined by ==. - } else if (typeof actual != 'object' && typeof expected != 'object') { - return actual == expected; - - // 7.4. For all other Object pairs, including Array objects, equivalence is - // determined by having the same number of owned properties (as verified - // with Object.prototype.hasOwnProperty.call), the same set of keys - // (although not necessarily the same order), equivalent values for every - // corresponding key, and an identical "prototype" property. Note: this - // accounts for both named and indexed properties on Arrays. - } else { - return objEquiv(actual, expected); - } - } - - function isUndefinedOrNull (value) { - return value === null || value === undefined; - } - - function isArguments (object) { - return Object.prototype.toString.call(object) == '[object Arguments]'; - } - - function objEquiv (a, b) { - if (isUndefinedOrNull(a) || isUndefinedOrNull(b)) - return false; - // an identical "prototype" property. - if (a.prototype !== b.prototype) return false; - //~~~I've managed to break Object.keys through screwy arguments passing. - // Converting to array solves the problem. - if (isArguments(a)) { - if (!isArguments(b)) { - return false; - } - a = pSlice.call(a); - b = pSlice.call(b); - return expect.eql(a, b); - } - try{ - var ka = keys(a), - kb = keys(b), - key, i; - } catch (e) {//happens when one is a string literal and the other isn't - return false; - } - // having the same number of owned properties (keys incorporates hasOwnProperty) - if (ka.length != kb.length) - return false; - //the same set of keys (although not necessarily the same order), - ka.sort(); - kb.sort(); - //~~~cheap key test - for (i = ka.length - 1; i >= 0; i--) { - if (ka[i] != kb[i]) - return false; - } - //equivalent values for every corresponding key, and - //~~~possibly expensive deep test - for (i = ka.length - 1; i >= 0; i--) { - key = ka[i]; - if (!expect.eql(a[key], b[key])) - return false; - } - return true; - } - - var json = (function () { - "use strict"; - - if ('object' == typeof JSON && JSON.parse && JSON.stringify) { - return { - parse: nativeJSON.parse - , stringify: nativeJSON.stringify - } - } - - var JSON = {}; - - function f(n) { - // Format integers to have at least two digits. - return n < 10 ? '0' + n : n; - } - - function date(d, key) { - return isFinite(d.valueOf()) ? - d.getUTCFullYear() + '-' + - f(d.getUTCMonth() + 1) + '-' + - f(d.getUTCDate()) + 'T' + - f(d.getUTCHours()) + ':' + - f(d.getUTCMinutes()) + ':' + - f(d.getUTCSeconds()) + 'Z' : null; - }; - - var cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g, - escapable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g, - gap, - indent, - meta = { // table of character substitutions - '\b': '\\b', - '\t': '\\t', - '\n': '\\n', - '\f': '\\f', - '\r': '\\r', - '"' : '\\"', - '\\': '\\\\' - }, - rep; - - - function quote(string) { - - // If the string contains no control characters, no quote characters, and no - // backslash characters, then we can safely slap some quotes around it. - // Otherwise we must also replace the offending characters with safe escape - // sequences. - - escapable.lastIndex = 0; - return escapable.test(string) ? '"' + string.replace(escapable, function (a) { - var c = meta[a]; - return typeof c === 'string' ? c : - '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4); - }) + '"' : '"' + string + '"'; - } - - - function str(key, holder) { - - // Produce a string from holder[key]. - - var i, // The loop counter. - k, // The member key. - v, // The member value. - length, - mind = gap, - partial, - value = holder[key]; - - // If the value has a toJSON method, call it to obtain a replacement value. - - if (value instanceof Date) { - value = date(key); - } - - // If we were called with a replacer function, then call the replacer to - // obtain a replacement value. - - if (typeof rep === 'function') { - value = rep.call(holder, key, value); - } - - // What happens next depends on the value's type. - - switch (typeof value) { - case 'string': - return quote(value); - - case 'number': - - // JSON numbers must be finite. Encode non-finite numbers as null. - - return isFinite(value) ? String(value) : 'null'; - - case 'boolean': - case 'null': - - // If the value is a boolean or null, convert it to a string. Note: - // typeof null does not produce 'null'. The case is included here in - // the remote chance that this gets fixed someday. - - return String(value); - - // If the type is 'object', we might be dealing with an object or an array or - // null. - - case 'object': - - // Due to a specification blunder in ECMAScript, typeof null is 'object', - // so watch out for that case. - - if (!value) { - return 'null'; - } - - // Make an array to hold the partial results of stringifying this object value. - - gap += indent; - partial = []; - - // Is the value an array? - - if (Object.prototype.toString.apply(value) === '[object Array]') { - - // The value is an array. Stringify every element. Use null as a placeholder - // for non-JSON values. - - length = value.length; - for (i = 0; i < length; i += 1) { - partial[i] = str(i, value) || 'null'; - } - - // Join all of the elements together, separated with commas, and wrap them in - // brackets. - - v = partial.length === 0 ? '[]' : gap ? - '[\n' + gap + partial.join(',\n' + gap) + '\n' + mind + ']' : - '[' + partial.join(',') + ']'; - gap = mind; - return v; - } - - // If the replacer is an array, use it to select the members to be stringified. - - if (rep && typeof rep === 'object') { - length = rep.length; - for (i = 0; i < length; i += 1) { - if (typeof rep[i] === 'string') { - k = rep[i]; - v = str(k, value); - if (v) { - partial.push(quote(k) + (gap ? ': ' : ':') + v); - } - } - } - } else { - - // Otherwise, iterate through all of the keys in the object. - - for (k in value) { - if (Object.prototype.hasOwnProperty.call(value, k)) { - v = str(k, value); - if (v) { - partial.push(quote(k) + (gap ? ': ' : ':') + v); - } - } - } - } - - // Join all of the member texts together, separated with commas, - // and wrap them in braces. - - v = partial.length === 0 ? '{}' : gap ? - '{\n' + gap + partial.join(',\n' + gap) + '\n' + mind + '}' : - '{' + partial.join(',') + '}'; - gap = mind; - return v; - } - } - - // If the JSON object does not yet have a stringify method, give it one. - - JSON.stringify = function (value, replacer, space) { - - // The stringify method takes a value and an optional replacer, and an optional - // space parameter, and returns a JSON text. The replacer can be a function - // that can replace values, or an array of strings that will select the keys. - // A default replacer method can be provided. Use of the space parameter can - // produce text that is more easily readable. - - var i; - gap = ''; - indent = ''; - - // If the space parameter is a number, make an indent string containing that - // many spaces. - - if (typeof space === 'number') { - for (i = 0; i < space; i += 1) { - indent += ' '; - } - - // If the space parameter is a string, it will be used as the indent string. - - } else if (typeof space === 'string') { - indent = space; - } - - // If there is a replacer, it must be a function or an array. - // Otherwise, throw an error. - - rep = replacer; - if (replacer && typeof replacer !== 'function' && - (typeof replacer !== 'object' || - typeof replacer.length !== 'number')) { - throw new Error('JSON.stringify'); - } - - // Make a fake root object containing our value under the key of ''. - // Return the result of stringifying the value. - - return str('', {'': value}); - }; - - // If the JSON object does not yet have a parse method, give it one. - - JSON.parse = function (text, reviver) { - // The parse method takes a text and an optional reviver function, and returns - // a JavaScript value if the text is a valid JSON text. - - var j; - - function walk(holder, key) { - - // The walk method is used to recursively walk the resulting structure so - // that modifications can be made. - - var k, v, value = holder[key]; - if (value && typeof value === 'object') { - for (k in value) { - if (Object.prototype.hasOwnProperty.call(value, k)) { - v = walk(value, k); - if (v !== undefined) { - value[k] = v; - } else { - delete value[k]; - } - } - } - } - return reviver.call(holder, key, value); - } - - - // Parsing happens in four stages. In the first stage, we replace certain - // Unicode characters with escape sequences. JavaScript handles many characters - // incorrectly, either silently deleting them, or treating them as line endings. - - text = String(text); - cx.lastIndex = 0; - if (cx.test(text)) { - text = text.replace(cx, function (a) { - return '\\u' + - ('0000' + a.charCodeAt(0).toString(16)).slice(-4); - }); - } - - // In the second stage, we run the text against regular expressions that look - // for non-JSON patterns. We are especially concerned with '()' and 'new' - // because they can cause invocation, and '=' because it can cause mutation. - // But just to be safe, we want to reject all unexpected forms. - - // We split the second stage into 4 regexp operations in order to work around - // crippling inefficiencies in IE's and Safari's regexp engines. First we - // replace the JSON backslash pairs with '@' (a non-JSON character). Second, we - // replace all simple value tokens with ']' characters. Third, we delete all - // open brackets that follow a colon or comma or that begin the text. Finally, - // we look to see that the remaining characters are only whitespace or ']' or - // ',' or ':' or '{' or '}'. If that is so, then the text is safe for eval. - - if (/^[\],:{}\s]*$/ - .test(text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@') - .replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']') - .replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) { - - // In the third stage we use the eval function to compile the text into a - // JavaScript structure. The '{' operator is subject to a syntactic ambiguity - // in JavaScript: it can begin a block or an object literal. We wrap the text - // in parens to eliminate the ambiguity. - - j = eval('(' + text + ')'); - - // In the optional fourth stage, we recursively walk the new structure, passing - // each name/value pair to a reviver function for possible transformation. - - return typeof reviver === 'function' ? - walk({'': j}, '') : j; - } - - // If the text is not JSON parseable, then a SyntaxError is thrown. - - throw new SyntaxError('JSON.parse'); - }; - - return JSON; - })(); - - if ('undefined' != typeof window) { - window.expect = module.exports; - } - -})( - this - , 'undefined' != typeof module ? module : {} - , 'undefined' != typeof exports ? exports : {} -); \ No newline at end of file diff --git a/node_modules/express/node_modules/connect/node_modules/qs/test/browser/index.html b/node_modules/express/node_modules/connect/node_modules/qs/test/browser/index.html deleted file mode 100644 index c73147a..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/test/browser/index.html +++ /dev/null @@ -1,18 +0,0 @@ - - - Mocha - - - - - - - - - - - - -
    - - diff --git a/node_modules/express/node_modules/connect/node_modules/qs/test/browser/jquery.js b/node_modules/express/node_modules/connect/node_modules/qs/test/browser/jquery.js deleted file mode 100644 index f3201aa..0000000 --- a/node_modules/express/node_modules/connect/node_modules/qs/test/browser/jquery.js +++ /dev/null @@ -1,8981 +0,0 @@ -/*! - * jQuery JavaScript Library v1.6.2 - * http://jquery.com/ - * - * Copyright 2011, John Resig - * Dual licensed under the MIT or GPL Version 2 licenses. - * http://jquery.org/license - * - * Includes Sizzle.js - * http://sizzlejs.com/ - * Copyright 2011, The Dojo Foundation - * Released under the MIT, BSD, and GPL Licenses. - * - * Date: Thu Jun 30 14:16:56 2011 -0400 - */ -(function( window, undefined ) { - -// Use the correct document accordingly with window argument (sandbox) -var document = window.document, - navigator = window.navigator, - location = window.location; -var jQuery = (function() { - -// Define a local copy of jQuery -var jQuery = function( selector, context ) { - // The jQuery object is actually just the init constructor 'enhanced' - return new jQuery.fn.init( selector, context, rootjQuery ); - }, - - // Map over jQuery in case of overwrite - _jQuery = window.jQuery, - - // Map over the $ in case of overwrite - _$ = window.$, - - // A central reference to the root jQuery(document) - rootjQuery, - - // A simple way to check for HTML strings or ID strings - // (both of which we optimize for) - quickExpr = /^(?:[^<]*(<[\w\W]+>)[^>]*$|#([\w\-]*)$)/, - - // Check if a string has a non-whitespace character in it - rnotwhite = /\S/, - - // Used for trimming whitespace - trimLeft = /^\s+/, - trimRight = /\s+$/, - - // Check for digits - rdigit = /\d/, - - // Match a standalone tag - rsingleTag = /^<(\w+)\s*\/?>(?:<\/\1>)?$/, - - // JSON RegExp - rvalidchars = /^[\],:{}\s]*$/, - rvalidescape = /\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, - rvalidtokens = /"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, - rvalidbraces = /(?:^|:|,)(?:\s*\[)+/g, - - // Useragent RegExp - rwebkit = /(webkit)[ \/]([\w.]+)/, - ropera = /(opera)(?:.*version)?[ \/]([\w.]+)/, - rmsie = /(msie) ([\w.]+)/, - rmozilla = /(mozilla)(?:.*? rv:([\w.]+))?/, - - // Matches dashed string for camelizing - rdashAlpha = /-([a-z])/ig, - - // Used by jQuery.camelCase as callback to replace() - fcamelCase = function( all, letter ) { - return letter.toUpperCase(); - }, - - // Keep a UserAgent string for use with jQuery.browser - userAgent = navigator.userAgent, - - // For matching the engine and version of the browser - browserMatch, - - // The deferred used on DOM ready - readyList, - - // The ready event handler - DOMContentLoaded, - - // Save a reference to some core methods - toString = Object.prototype.toString, - hasOwn = Object.prototype.hasOwnProperty, - push = Array.prototype.push, - slice = Array.prototype.slice, - trim = String.prototype.trim, - indexOf = Array.prototype.indexOf, - - // [[Class]] -> type pairs - class2type = {}; - -jQuery.fn = jQuery.prototype = { - constructor: jQuery, - init: function( selector, context, rootjQuery ) { - var match, elem, ret, doc; - - // Handle $(""), $(null), or $(undefined) - if ( !selector ) { - return this; - } - - // Handle $(DOMElement) - if ( selector.nodeType ) { - this.context = this[0] = selector; - this.length = 1; - return this; - } - - // The body element only exists once, optimize finding it - if ( selector === "body" && !context && document.body ) { - this.context = document; - this[0] = document.body; - this.selector = selector; - this.length = 1; - return this; - } - - // Handle HTML strings - if ( typeof selector === "string" ) { - // Are we dealing with HTML string or an ID? - if ( selector.charAt(0) === "<" && selector.charAt( selector.length - 1 ) === ">" && selector.length >= 3 ) { - // Assume that strings that start and end with <> are HTML and skip the regex check - match = [ null, selector, null ]; - - } else { - match = quickExpr.exec( selector ); - } - - // Verify a match, and that no context was specified for #id - if ( match && (match[1] || !context) ) { - - // HANDLE: $(html) -> $(array) - if ( match[1] ) { - context = context instanceof jQuery ? context[0] : context; - doc = (context ? context.ownerDocument || context : document); - - // If a single string is passed in and it's a single tag - // just do a createElement and skip the rest - ret = rsingleTag.exec( selector ); - - if ( ret ) { - if ( jQuery.isPlainObject( context ) ) { - selector = [ document.createElement( ret[1] ) ]; - jQuery.fn.attr.call( selector, context, true ); - - } else { - selector = [ doc.createElement( ret[1] ) ]; - } - - } else { - ret = jQuery.buildFragment( [ match[1] ], [ doc ] ); - selector = (ret.cacheable ? jQuery.clone(ret.fragment) : ret.fragment).childNodes; - } - - return jQuery.merge( this, selector ); - - // HANDLE: $("#id") - } else { - elem = document.getElementById( match[2] ); - - // Check parentNode to catch when Blackberry 4.6 returns - // nodes that are no longer in the document #6963 - if ( elem && elem.parentNode ) { - // Handle the case where IE and Opera return items - // by name instead of ID - if ( elem.id !== match[2] ) { - return rootjQuery.find( selector ); - } - - // Otherwise, we inject the element directly into the jQuery object - this.length = 1; - this[0] = elem; - } - - this.context = document; - this.selector = selector; - return this; - } - - // HANDLE: $(expr, $(...)) - } else if ( !context || context.jquery ) { - return (context || rootjQuery).find( selector ); - - // HANDLE: $(expr, context) - // (which is just equivalent to: $(context).find(expr) - } else { - return this.constructor( context ).find( selector ); - } - - // HANDLE: $(function) - // Shortcut for document ready - } else if ( jQuery.isFunction( selector ) ) { - return rootjQuery.ready( selector ); - } - - if (selector.selector !== undefined) { - this.selector = selector.selector; - this.context = selector.context; - } - - return jQuery.makeArray( selector, this ); - }, - - // Start with an empty selector - selector: "", - - // The current version of jQuery being used - jquery: "1.6.2", - - // The default length of a jQuery object is 0 - length: 0, - - // The number of elements contained in the matched element set - size: function() { - return this.length; - }, - - toArray: function() { - return slice.call( this, 0 ); - }, - - // Get the Nth element in the matched element set OR - // Get the whole matched element set as a clean array - get: function( num ) { - return num == null ? - - // Return a 'clean' array - this.toArray() : - - // Return just the object - ( num < 0 ? this[ this.length + num ] : this[ num ] ); - }, - - // Take an array of elements and push it onto the stack - // (returning the new matched element set) - pushStack: function( elems, name, selector ) { - // Build a new jQuery matched element set - var ret = this.constructor(); - - if ( jQuery.isArray( elems ) ) { - push.apply( ret, elems ); - - } else { - jQuery.merge( ret, elems ); - } - - // Add the old object onto the stack (as a reference) - ret.prevObject = this; - - ret.context = this.context; - - if ( name === "find" ) { - ret.selector = this.selector + (this.selector ? " " : "") + selector; - } else if ( name ) { - ret.selector = this.selector + "." + name + "(" + selector + ")"; - } - - // Return the newly-formed element set - return ret; - }, - - // Execute a callback for every element in the matched set. - // (You can seed the arguments with an array of args, but this is - // only used internally.) - each: function( callback, args ) { - return jQuery.each( this, callback, args ); - }, - - ready: function( fn ) { - // Attach the listeners - jQuery.bindReady(); - - // Add the callback - readyList.done( fn ); - - return this; - }, - - eq: function( i ) { - return i === -1 ? - this.slice( i ) : - this.slice( i, +i + 1 ); - }, - - first: function() { - return this.eq( 0 ); - }, - - last: function() { - return this.eq( -1 ); - }, - - slice: function() { - return this.pushStack( slice.apply( this, arguments ), - "slice", slice.call(arguments).join(",") ); - }, - - map: function( callback ) { - return this.pushStack( jQuery.map(this, function( elem, i ) { - return callback.call( elem, i, elem ); - })); - }, - - end: function() { - return this.prevObject || this.constructor(null); - }, - - // For internal use only. - // Behaves like an Array's method, not like a jQuery method. - push: push, - sort: [].sort, - splice: [].splice -}; - -// Give the init function the jQuery prototype for later instantiation -jQuery.fn.init.prototype = jQuery.fn; - -jQuery.extend = jQuery.fn.extend = function() { - var options, name, src, copy, copyIsArray, clone, - target = arguments[0] || {}, - i = 1, - length = arguments.length, - deep = false; - - // Handle a deep copy situation - if ( typeof target === "boolean" ) { - deep = target; - target = arguments[1] || {}; - // skip the boolean and the target - i = 2; - } - - // Handle case when target is a string or something (possible in deep copy) - if ( typeof target !== "object" && !jQuery.isFunction(target) ) { - target = {}; - } - - // extend jQuery itself if only one argument is passed - if ( length === i ) { - target = this; - --i; - } - - for ( ; i < length; i++ ) { - // Only deal with non-null/undefined values - if ( (options = arguments[ i ]) != null ) { - // Extend the base object - for ( name in options ) { - src = target[ name ]; - copy = options[ name ]; - - // Prevent never-ending loop - if ( target === copy ) { - continue; - } - - // Recurse if we're merging plain objects or arrays - if ( deep && copy && ( jQuery.isPlainObject(copy) || (copyIsArray = jQuery.isArray(copy)) ) ) { - if ( copyIsArray ) { - copyIsArray = false; - clone = src && jQuery.isArray(src) ? src : []; - - } else { - clone = src && jQuery.isPlainObject(src) ? src : {}; - } - - // Never move original objects, clone them - target[ name ] = jQuery.extend( deep, clone, copy ); - - // Don't bring in undefined values - } else if ( copy !== undefined ) { - target[ name ] = copy; - } - } - } - } - - // Return the modified object - return target; -}; - -jQuery.extend({ - noConflict: function( deep ) { - if ( window.$ === jQuery ) { - window.$ = _$; - } - - if ( deep && window.jQuery === jQuery ) { - window.jQuery = _jQuery; - } - - return jQuery; - }, - - // Is the DOM ready to be used? Set to true once it occurs. - isReady: false, - - // A counter to track how many items to wait for before - // the ready event fires. See #6781 - readyWait: 1, - - // Hold (or release) the ready event - holdReady: function( hold ) { - if ( hold ) { - jQuery.readyWait++; - } else { - jQuery.ready( true ); - } - }, - - // Handle when the DOM is ready - ready: function( wait ) { - // Either a released hold or an DOMready/load event and not yet ready - if ( (wait === true && !--jQuery.readyWait) || (wait !== true && !jQuery.isReady) ) { - // Make sure body exists, at least, in case IE gets a little overzealous (ticket #5443). - if ( !document.body ) { - return setTimeout( jQuery.ready, 1 ); - } - - // Remember that the DOM is ready - jQuery.isReady = true; - - // If a normal DOM Ready event fired, decrement, and wait if need be - if ( wait !== true && --jQuery.readyWait > 0 ) { - return; - } - - // If there are functions bound, to execute - readyList.resolveWith( document, [ jQuery ] ); - - // Trigger any bound ready events - if ( jQuery.fn.trigger ) { - jQuery( document ).trigger( "ready" ).unbind( "ready" ); - } - } - }, - - bindReady: function() { - if ( readyList ) { - return; - } - - readyList = jQuery._Deferred(); - - // Catch cases where $(document).ready() is called after the - // browser event has already occurred. - if ( document.readyState === "complete" ) { - // Handle it asynchronously to allow scripts the opportunity to delay ready - return setTimeout( jQuery.ready, 1 ); - } - - // Mozilla, Opera and webkit nightlies currently support this event - if ( document.addEventListener ) { - // Use the handy event callback - document.addEventListener( "DOMContentLoaded", DOMContentLoaded, false ); - - // A fallback to window.onload, that will always work - window.addEventListener( "load", jQuery.ready, false ); - - // If IE event model is used - } else if ( document.attachEvent ) { - // ensure firing before onload, - // maybe late but safe also for iframes - document.attachEvent( "onreadystatechange", DOMContentLoaded ); - - // A fallback to window.onload, that will always work - window.attachEvent( "onload", jQuery.ready ); - - // If IE and not a frame - // continually check to see if the document is ready - var toplevel = false; - - try { - toplevel = window.frameElement == null; - } catch(e) {} - - if ( document.documentElement.doScroll && toplevel ) { - doScrollCheck(); - } - } - }, - - // See test/unit/core.js for details concerning isFunction. - // Since version 1.3, DOM methods and functions like alert - // aren't supported. They return false on IE (#2968). - isFunction: function( obj ) { - return jQuery.type(obj) === "function"; - }, - - isArray: Array.isArray || function( obj ) { - return jQuery.type(obj) === "array"; - }, - - // A crude way of determining if an object is a window - isWindow: function( obj ) { - return obj && typeof obj === "object" && "setInterval" in obj; - }, - - isNaN: function( obj ) { - return obj == null || !rdigit.test( obj ) || isNaN( obj ); - }, - - type: function( obj ) { - return obj == null ? - String( obj ) : - class2type[ toString.call(obj) ] || "object"; - }, - - isPlainObject: function( obj ) { - // Must be an Object. - // Because of IE, we also have to check the presence of the constructor property. - // Make sure that DOM nodes and window objects don't pass through, as well - if ( !obj || jQuery.type(obj) !== "object" || obj.nodeType || jQuery.isWindow( obj ) ) { - return false; - } - - // Not own constructor property must be Object - if ( obj.constructor && - !hasOwn.call(obj, "constructor") && - !hasOwn.call(obj.constructor.prototype, "isPrototypeOf") ) { - return false; - } - - // Own properties are enumerated firstly, so to speed up, - // if last one is own, then all properties are own. - - var key; - for ( key in obj ) {} - - return key === undefined || hasOwn.call( obj, key ); - }, - - isEmptyObject: function( obj ) { - for ( var name in obj ) { - return false; - } - return true; - }, - - error: function( msg ) { - throw msg; - }, - - parseJSON: function( data ) { - if ( typeof data !== "string" || !data ) { - return null; - } - - // Make sure leading/trailing whitespace is removed (IE can't handle it) - data = jQuery.trim( data ); - - // Attempt to parse using the native JSON parser first - if ( window.JSON && window.JSON.parse ) { - return window.JSON.parse( data ); - } - - // Make sure the incoming data is actual JSON - // Logic borrowed from http://json.org/json2.js - if ( rvalidchars.test( data.replace( rvalidescape, "@" ) - .replace( rvalidtokens, "]" ) - .replace( rvalidbraces, "")) ) { - - return (new Function( "return " + data ))(); - - } - jQuery.error( "Invalid JSON: " + data ); - }, - - // Cross-browser xml parsing - // (xml & tmp used internally) - parseXML: function( data , xml , tmp ) { - - if ( window.DOMParser ) { // Standard - tmp = new DOMParser(); - xml = tmp.parseFromString( data , "text/xml" ); - } else { // IE - xml = new ActiveXObject( "Microsoft.XMLDOM" ); - xml.async = "false"; - xml.loadXML( data ); - } - - tmp = xml.documentElement; - - if ( ! tmp || ! tmp.nodeName || tmp.nodeName === "parsererror" ) { - jQuery.error( "Invalid XML: " + data ); - } - - return xml; - }, - - noop: function() {}, - - // Evaluates a script in a global context - // Workarounds based on findings by Jim Driscoll - // http://weblogs.java.net/blog/driscoll/archive/2009/09/08/eval-javascript-global-context - globalEval: function( data ) { - if ( data && rnotwhite.test( data ) ) { - // We use execScript on Internet Explorer - // We use an anonymous function so that context is window - // rather than jQuery in Firefox - ( window.execScript || function( data ) { - window[ "eval" ].call( window, data ); - } )( data ); - } - }, - - // Converts a dashed string to camelCased string; - // Used by both the css and data modules - camelCase: function( string ) { - return string.replace( rdashAlpha, fcamelCase ); - }, - - nodeName: function( elem, name ) { - return elem.nodeName && elem.nodeName.toUpperCase() === name.toUpperCase(); - }, - - // args is for internal usage only - each: function( object, callback, args ) { - var name, i = 0, - length = object.length, - isObj = length === undefined || jQuery.isFunction( object ); - - if ( args ) { - if ( isObj ) { - for ( name in object ) { - if ( callback.apply( object[ name ], args ) === false ) { - break; - } - } - } else { - for ( ; i < length; ) { - if ( callback.apply( object[ i++ ], args ) === false ) { - break; - } - } - } - - // A special, fast, case for the most common use of each - } else { - if ( isObj ) { - for ( name in object ) { - if ( callback.call( object[ name ], name, object[ name ] ) === false ) { - break; - } - } - } else { - for ( ; i < length; ) { - if ( callback.call( object[ i ], i, object[ i++ ] ) === false ) { - break; - } - } - } - } - - return object; - }, - - // Use native String.trim function wherever possible - trim: trim ? - function( text ) { - return text == null ? - "" : - trim.call( text ); - } : - - // Otherwise use our own trimming functionality - function( text ) { - return text == null ? - "" : - text.toString().replace( trimLeft, "" ).replace( trimRight, "" ); - }, - - // results is for internal usage only - makeArray: function( array, results ) { - var ret = results || []; - - if ( array != null ) { - // The window, strings (and functions) also have 'length' - // The extra typeof function check is to prevent crashes - // in Safari 2 (See: #3039) - // Tweaked logic slightly to handle Blackberry 4.7 RegExp issues #6930 - var type = jQuery.type( array ); - - if ( array.length == null || type === "string" || type === "function" || type === "regexp" || jQuery.isWindow( array ) ) { - push.call( ret, array ); - } else { - jQuery.merge( ret, array ); - } - } - - return ret; - }, - - inArray: function( elem, array ) { - - if ( indexOf ) { - return indexOf.call( array, elem ); - } - - for ( var i = 0, length = array.length; i < length; i++ ) { - if ( array[ i ] === elem ) { - return i; - } - } - - return -1; - }, - - merge: function( first, second ) { - var i = first.length, - j = 0; - - if ( typeof second.length === "number" ) { - for ( var l = second.length; j < l; j++ ) { - first[ i++ ] = second[ j ]; - } - - } else { - while ( second[j] !== undefined ) { - first[ i++ ] = second[ j++ ]; - } - } - - first.length = i; - - return first; - }, - - grep: function( elems, callback, inv ) { - var ret = [], retVal; - inv = !!inv; - - // Go through the array, only saving the items - // that pass the validator function - for ( var i = 0, length = elems.length; i < length; i++ ) { - retVal = !!callback( elems[ i ], i ); - if ( inv !== retVal ) { - ret.push( elems[ i ] ); - } - } - - return ret; - }, - - // arg is for internal usage only - map: function( elems, callback, arg ) { - var value, key, ret = [], - i = 0, - length = elems.length, - // jquery objects are treated as arrays - isArray = elems instanceof jQuery || length !== undefined && typeof length === "number" && ( ( length > 0 && elems[ 0 ] && elems[ length -1 ] ) || length === 0 || jQuery.isArray( elems ) ) ; - - // Go through the array, translating each of the items to their - if ( isArray ) { - for ( ; i < length; i++ ) { - value = callback( elems[ i ], i, arg ); - - if ( value != null ) { - ret[ ret.length ] = value; - } - } - - // Go through every key on the object, - } else { - for ( key in elems ) { - value = callback( elems[ key ], key, arg ); - - if ( value != null ) { - ret[ ret.length ] = value; - } - } - } - - // Flatten any nested arrays - return ret.concat.apply( [], ret ); - }, - - // A global GUID counter for objects - guid: 1, - - // Bind a function to a context, optionally partially applying any - // arguments. - proxy: function( fn, context ) { - if ( typeof context === "string" ) { - var tmp = fn[ context ]; - context = fn; - fn = tmp; - } - - // Quick check to determine if target is callable, in the spec - // this throws a TypeError, but we will just return undefined. - if ( !jQuery.isFunction( fn ) ) { - return undefined; - } - - // Simulated bind - var args = slice.call( arguments, 2 ), - proxy = function() { - return fn.apply( context, args.concat( slice.call( arguments ) ) ); - }; - - // Set the guid of unique handler to the same of original handler, so it can be removed - proxy.guid = fn.guid = fn.guid || proxy.guid || jQuery.guid++; - - return proxy; - }, - - // Mutifunctional method to get and set values to a collection - // The value/s can optionally be executed if it's a function - access: function( elems, key, value, exec, fn, pass ) { - var length = elems.length; - - // Setting many attributes - if ( typeof key === "object" ) { - for ( var k in key ) { - jQuery.access( elems, k, key[k], exec, fn, value ); - } - return elems; - } - - // Setting one attribute - if ( value !== undefined ) { - // Optionally, function values get executed if exec is true - exec = !pass && exec && jQuery.isFunction(value); - - for ( var i = 0; i < length; i++ ) { - fn( elems[i], key, exec ? value.call( elems[i], i, fn( elems[i], key ) ) : value, pass ); - } - - return elems; - } - - // Getting an attribute - return length ? fn( elems[0], key ) : undefined; - }, - - now: function() { - return (new Date()).getTime(); - }, - - // Use of jQuery.browser is frowned upon. - // More details: http://docs.jquery.com/Utilities/jQuery.browser - uaMatch: function( ua ) { - ua = ua.toLowerCase(); - - var match = rwebkit.exec( ua ) || - ropera.exec( ua ) || - rmsie.exec( ua ) || - ua.indexOf("compatible") < 0 && rmozilla.exec( ua ) || - []; - - return { browser: match[1] || "", version: match[2] || "0" }; - }, - - sub: function() { - function jQuerySub( selector, context ) { - return new jQuerySub.fn.init( selector, context ); - } - jQuery.extend( true, jQuerySub, this ); - jQuerySub.superclass = this; - jQuerySub.fn = jQuerySub.prototype = this(); - jQuerySub.fn.constructor = jQuerySub; - jQuerySub.sub = this.sub; - jQuerySub.fn.init = function init( selector, context ) { - if ( context && context instanceof jQuery && !(context instanceof jQuerySub) ) { - context = jQuerySub( context ); - } - - return jQuery.fn.init.call( this, selector, context, rootjQuerySub ); - }; - jQuerySub.fn.init.prototype = jQuerySub.fn; - var rootjQuerySub = jQuerySub(document); - return jQuerySub; - }, - - browser: {} -}); - -// Populate the class2type map -jQuery.each("Boolean Number String Function Array Date RegExp Object".split(" "), function(i, name) { - class2type[ "[object " + name + "]" ] = name.toLowerCase(); -}); - -browserMatch = jQuery.uaMatch( userAgent ); -if ( browserMatch.browser ) { - jQuery.browser[ browserMatch.browser ] = true; - jQuery.browser.version = browserMatch.version; -} - -// Deprecated, use jQuery.browser.webkit instead -if ( jQuery.browser.webkit ) { - jQuery.browser.safari = true; -} - -// IE doesn't match non-breaking spaces with \s -if ( rnotwhite.test( "\xA0" ) ) { - trimLeft = /^[\s\xA0]+/; - trimRight = /[\s\xA0]+$/; -} - -// All jQuery objects should point back to these -rootjQuery = jQuery(document); - -// Cleanup functions for the document ready method -if ( document.addEventListener ) { - DOMContentLoaded = function() { - document.removeEventListener( "DOMContentLoaded", DOMContentLoaded, false ); - jQuery.ready(); - }; - -} else if ( document.attachEvent ) { - DOMContentLoaded = function() { - // Make sure body exists, at least, in case IE gets a little overzealous (ticket #5443). - if ( document.readyState === "complete" ) { - document.detachEvent( "onreadystatechange", DOMContentLoaded ); - jQuery.ready(); - } - }; -} - -// The DOM ready check for Internet Explorer -function doScrollCheck() { - if ( jQuery.isReady ) { - return; - } - - try { - // If IE is used, use the trick by Diego Perini - // http://javascript.nwbox.com/IEContentLoaded/ - document.documentElement.doScroll("left"); - } catch(e) { - setTimeout( doScrollCheck, 1 ); - return; - } - - // and execute any waiting functions - jQuery.ready(); -} - -return jQuery; - -})(); - - -var // Promise methods - promiseMethods = "done fail isResolved isRejected promise then always pipe".split( " " ), - // Static reference to slice - sliceDeferred = [].slice; - -jQuery.extend({ - // Create a simple deferred (one callbacks list) - _Deferred: function() { - var // callbacks list - callbacks = [], - // stored [ context , args ] - fired, - // to avoid firing when already doing so - firing, - // flag to know if the deferred has been cancelled - cancelled, - // the deferred itself - deferred = { - - // done( f1, f2, ...) - done: function() { - if ( !cancelled ) { - var args = arguments, - i, - length, - elem, - type, - _fired; - if ( fired ) { - _fired = fired; - fired = 0; - } - for ( i = 0, length = args.length; i < length; i++ ) { - elem = args[ i ]; - type = jQuery.type( elem ); - if ( type === "array" ) { - deferred.done.apply( deferred, elem ); - } else if ( type === "function" ) { - callbacks.push( elem ); - } - } - if ( _fired ) { - deferred.resolveWith( _fired[ 0 ], _fired[ 1 ] ); - } - } - return this; - }, - - // resolve with given context and args - resolveWith: function( context, args ) { - if ( !cancelled && !fired && !firing ) { - // make sure args are available (#8421) - args = args || []; - firing = 1; - try { - while( callbacks[ 0 ] ) { - callbacks.shift().apply( context, args ); - } - } - finally { - fired = [ context, args ]; - firing = 0; - } - } - return this; - }, - - // resolve with this as context and given arguments - resolve: function() { - deferred.resolveWith( this, arguments ); - return this; - }, - - // Has this deferred been resolved? - isResolved: function() { - return !!( firing || fired ); - }, - - // Cancel - cancel: function() { - cancelled = 1; - callbacks = []; - return this; - } - }; - - return deferred; - }, - - // Full fledged deferred (two callbacks list) - Deferred: function( func ) { - var deferred = jQuery._Deferred(), - failDeferred = jQuery._Deferred(), - promise; - // Add errorDeferred methods, then and promise - jQuery.extend( deferred, { - then: function( doneCallbacks, failCallbacks ) { - deferred.done( doneCallbacks ).fail( failCallbacks ); - return this; - }, - always: function() { - return deferred.done.apply( deferred, arguments ).fail.apply( this, arguments ); - }, - fail: failDeferred.done, - rejectWith: failDeferred.resolveWith, - reject: failDeferred.resolve, - isRejected: failDeferred.isResolved, - pipe: function( fnDone, fnFail ) { - return jQuery.Deferred(function( newDefer ) { - jQuery.each( { - done: [ fnDone, "resolve" ], - fail: [ fnFail, "reject" ] - }, function( handler, data ) { - var fn = data[ 0 ], - action = data[ 1 ], - returned; - if ( jQuery.isFunction( fn ) ) { - deferred[ handler ](function() { - returned = fn.apply( this, arguments ); - if ( returned && jQuery.isFunction( returned.promise ) ) { - returned.promise().then( newDefer.resolve, newDefer.reject ); - } else { - newDefer[ action ]( returned ); - } - }); - } else { - deferred[ handler ]( newDefer[ action ] ); - } - }); - }).promise(); - }, - // Get a promise for this deferred - // If obj is provided, the promise aspect is added to the object - promise: function( obj ) { - if ( obj == null ) { - if ( promise ) { - return promise; - } - promise = obj = {}; - } - var i = promiseMethods.length; - while( i-- ) { - obj[ promiseMethods[i] ] = deferred[ promiseMethods[i] ]; - } - return obj; - } - }); - // Make sure only one callback list will be used - deferred.done( failDeferred.cancel ).fail( deferred.cancel ); - // Unexpose cancel - delete deferred.cancel; - // Call given func if any - if ( func ) { - func.call( deferred, deferred ); - } - return deferred; - }, - - // Deferred helper - when: function( firstParam ) { - var args = arguments, - i = 0, - length = args.length, - count = length, - deferred = length <= 1 && firstParam && jQuery.isFunction( firstParam.promise ) ? - firstParam : - jQuery.Deferred(); - function resolveFunc( i ) { - return function( value ) { - args[ i ] = arguments.length > 1 ? sliceDeferred.call( arguments, 0 ) : value; - if ( !( --count ) ) { - // Strange bug in FF4: - // Values changed onto the arguments object sometimes end up as undefined values - // outside the $.when method. Cloning the object into a fresh array solves the issue - deferred.resolveWith( deferred, sliceDeferred.call( args, 0 ) ); - } - }; - } - if ( length > 1 ) { - for( ; i < length; i++ ) { - if ( args[ i ] && jQuery.isFunction( args[ i ].promise ) ) { - args[ i ].promise().then( resolveFunc(i), deferred.reject ); - } else { - --count; - } - } - if ( !count ) { - deferred.resolveWith( deferred, args ); - } - } else if ( deferred !== firstParam ) { - deferred.resolveWith( deferred, length ? [ firstParam ] : [] ); - } - return deferred.promise(); - } -}); - - - -jQuery.support = (function() { - - var div = document.createElement( "div" ), - documentElement = document.documentElement, - all, - a, - select, - opt, - input, - marginDiv, - support, - fragment, - body, - testElementParent, - testElement, - testElementStyle, - tds, - events, - eventName, - i, - isSupported; - - // Preliminary tests - div.setAttribute("className", "t"); - div.innerHTML = "
    a"; - - all = div.getElementsByTagName( "*" ); - a = div.getElementsByTagName( "a" )[ 0 ]; - - // Can't get basic test support - if ( !all || !all.length || !a ) { - return {}; - } - - // First batch of supports tests - select = document.createElement( "select" ); - opt = select.appendChild( document.createElement("option") ); - input = div.getElementsByTagName( "input" )[ 0 ]; - - support = { - // IE strips leading whitespace when .innerHTML is used - leadingWhitespace: ( div.firstChild.nodeType === 3 ), - - // Make sure that tbody elements aren't automatically inserted - // IE will insert them into empty tables - tbody: !div.getElementsByTagName( "tbody" ).length, - - // Make sure that link elements get serialized correctly by innerHTML - // This requires a wrapper element in IE - htmlSerialize: !!div.getElementsByTagName( "link" ).length, - - // Get the style information from getAttribute - // (IE uses .cssText instead) - style: /top/.test( a.getAttribute("style") ), - - // Make sure that URLs aren't manipulated - // (IE normalizes it by default) - hrefNormalized: ( a.getAttribute( "href" ) === "/a" ), - - // Make sure that element opacity exists - // (IE uses filter instead) - // Use a regex to work around a WebKit issue. See #5145 - opacity: /^0.55$/.test( a.style.opacity ), - - // Verify style float existence - // (IE uses styleFloat instead of cssFloat) - cssFloat: !!a.style.cssFloat, - - // Make sure that if no value is specified for a checkbox - // that it defaults to "on". - // (WebKit defaults to "" instead) - checkOn: ( input.value === "on" ), - - // Make sure that a selected-by-default option has a working selected property. - // (WebKit defaults to false instead of true, IE too, if it's in an optgroup) - optSelected: opt.selected, - - // Test setAttribute on camelCase class. If it works, we need attrFixes when doing get/setAttribute (ie6/7) - getSetAttribute: div.className !== "t", - - // Will be defined later - submitBubbles: true, - changeBubbles: true, - focusinBubbles: false, - deleteExpando: true, - noCloneEvent: true, - inlineBlockNeedsLayout: false, - shrinkWrapBlocks: false, - reliableMarginRight: true - }; - - // Make sure checked status is properly cloned - input.checked = true; - support.noCloneChecked = input.cloneNode( true ).checked; - - // Make sure that the options inside disabled selects aren't marked as disabled - // (WebKit marks them as disabled) - select.disabled = true; - support.optDisabled = !opt.disabled; - - // Test to see if it's possible to delete an expando from an element - // Fails in Internet Explorer - try { - delete div.test; - } catch( e ) { - support.deleteExpando = false; - } - - if ( !div.addEventListener && div.attachEvent && div.fireEvent ) { - div.attachEvent( "onclick", function() { - // Cloning a node shouldn't copy over any - // bound event handlers (IE does this) - support.noCloneEvent = false; - }); - div.cloneNode( true ).fireEvent( "onclick" ); - } - - // Check if a radio maintains it's value - // after being appended to the DOM - input = document.createElement("input"); - input.value = "t"; - input.setAttribute("type", "radio"); - support.radioValue = input.value === "t"; - - input.setAttribute("checked", "checked"); - div.appendChild( input ); - fragment = document.createDocumentFragment(); - fragment.appendChild( div.firstChild ); - - // WebKit doesn't clone checked state correctly in fragments - support.checkClone = fragment.cloneNode( true ).cloneNode( true ).lastChild.checked; - - div.innerHTML = ""; - - // Figure out if the W3C box model works as expected - div.style.width = div.style.paddingLeft = "1px"; - - body = document.getElementsByTagName( "body" )[ 0 ]; - // We use our own, invisible, body unless the body is already present - // in which case we use a div (#9239) - testElement = document.createElement( body ? "div" : "body" ); - testElementStyle = { - visibility: "hidden", - width: 0, - height: 0, - border: 0, - margin: 0 - }; - if ( body ) { - jQuery.extend( testElementStyle, { - position: "absolute", - left: -1000, - top: -1000 - }); - } - for ( i in testElementStyle ) { - testElement.style[ i ] = testElementStyle[ i ]; - } - testElement.appendChild( div ); - testElementParent = body || documentElement; - testElementParent.insertBefore( testElement, testElementParent.firstChild ); - - // Check if a disconnected checkbox will retain its checked - // value of true after appended to the DOM (IE6/7) - support.appendChecked = input.checked; - - support.boxModel = div.offsetWidth === 2; - - if ( "zoom" in div.style ) { - // Check if natively block-level elements act like inline-block - // elements when setting their display to 'inline' and giving - // them layout - // (IE < 8 does this) - div.style.display = "inline"; - div.style.zoom = 1; - support.inlineBlockNeedsLayout = ( div.offsetWidth === 2 ); - - // Check if elements with layout shrink-wrap their children - // (IE 6 does this) - div.style.display = ""; - div.innerHTML = "
    "; - support.shrinkWrapBlocks = ( div.offsetWidth !== 2 ); - } - - div.innerHTML = "
    t
    "; - tds = div.getElementsByTagName( "td" ); - - // Check if table cells still have offsetWidth/Height when they are set - // to display:none and there are still other visible table cells in a - // table row; if so, offsetWidth/Height are not reliable for use when - // determining if an element has been hidden directly using - // display:none (it is still safe to use offsets if a parent element is - // hidden; don safety goggles and see bug #4512 for more information). - // (only IE 8 fails this test) - isSupported = ( tds[ 0 ].offsetHeight === 0 ); - - tds[ 0 ].style.display = ""; - tds[ 1 ].style.display = "none"; - - // Check if empty table cells still have offsetWidth/Height - // (IE < 8 fail this test) - support.reliableHiddenOffsets = isSupported && ( tds[ 0 ].offsetHeight === 0 ); - div.innerHTML = ""; - - // Check if div with explicit width and no margin-right incorrectly - // gets computed margin-right based on width of container. For more - // info see bug #3333 - // Fails in WebKit before Feb 2011 nightlies - // WebKit Bug 13343 - getComputedStyle returns wrong value for margin-right - if ( document.defaultView && document.defaultView.getComputedStyle ) { - marginDiv = document.createElement( "div" ); - marginDiv.style.width = "0"; - marginDiv.style.marginRight = "0"; - div.appendChild( marginDiv ); - support.reliableMarginRight = - ( parseInt( ( document.defaultView.getComputedStyle( marginDiv, null ) || { marginRight: 0 } ).marginRight, 10 ) || 0 ) === 0; - } - - // Remove the body element we added - testElement.innerHTML = ""; - testElementParent.removeChild( testElement ); - - // Technique from Juriy Zaytsev - // http://thinkweb2.com/projects/prototype/detecting-event-support-without-browser-sniffing/ - // We only care about the case where non-standard event systems - // are used, namely in IE. Short-circuiting here helps us to - // avoid an eval call (in setAttribute) which can cause CSP - // to go haywire. See: https://developer.mozilla.org/en/Security/CSP - if ( div.attachEvent ) { - for( i in { - submit: 1, - change: 1, - focusin: 1 - } ) { - eventName = "on" + i; - isSupported = ( eventName in div ); - if ( !isSupported ) { - div.setAttribute( eventName, "return;" ); - isSupported = ( typeof div[ eventName ] === "function" ); - } - support[ i + "Bubbles" ] = isSupported; - } - } - - // Null connected elements to avoid leaks in IE - testElement = fragment = select = opt = body = marginDiv = div = input = null; - - return support; -})(); - -// Keep track of boxModel -jQuery.boxModel = jQuery.support.boxModel; - - - - -var rbrace = /^(?:\{.*\}|\[.*\])$/, - rmultiDash = /([a-z])([A-Z])/g; - -jQuery.extend({ - cache: {}, - - // Please use with caution - uuid: 0, - - // Unique for each copy of jQuery on the page - // Non-digits removed to match rinlinejQuery - expando: "jQuery" + ( jQuery.fn.jquery + Math.random() ).replace( /\D/g, "" ), - - // The following elements throw uncatchable exceptions if you - // attempt to add expando properties to them. - noData: { - "embed": true, - // Ban all objects except for Flash (which handle expandos) - "object": "clsid:D27CDB6E-AE6D-11cf-96B8-444553540000", - "applet": true - }, - - hasData: function( elem ) { - elem = elem.nodeType ? jQuery.cache[ elem[jQuery.expando] ] : elem[ jQuery.expando ]; - - return !!elem && !isEmptyDataObject( elem ); - }, - - data: function( elem, name, data, pvt /* Internal Use Only */ ) { - if ( !jQuery.acceptData( elem ) ) { - return; - } - - var internalKey = jQuery.expando, getByName = typeof name === "string", thisCache, - - // We have to handle DOM nodes and JS objects differently because IE6-7 - // can't GC object references properly across the DOM-JS boundary - isNode = elem.nodeType, - - // Only DOM nodes need the global jQuery cache; JS object data is - // attached directly to the object so GC can occur automatically - cache = isNode ? jQuery.cache : elem, - - // Only defining an ID for JS objects if its cache already exists allows - // the code to shortcut on the same path as a DOM node with no cache - id = isNode ? elem[ jQuery.expando ] : elem[ jQuery.expando ] && jQuery.expando; - - // Avoid doing any more work than we need to when trying to get data on an - // object that has no data at all - if ( (!id || (pvt && id && !cache[ id ][ internalKey ])) && getByName && data === undefined ) { - return; - } - - if ( !id ) { - // Only DOM nodes need a new unique ID for each element since their data - // ends up in the global cache - if ( isNode ) { - elem[ jQuery.expando ] = id = ++jQuery.uuid; - } else { - id = jQuery.expando; - } - } - - if ( !cache[ id ] ) { - cache[ id ] = {}; - - // TODO: This is a hack for 1.5 ONLY. Avoids exposing jQuery - // metadata on plain JS objects when the object is serialized using - // JSON.stringify - if ( !isNode ) { - cache[ id ].toJSON = jQuery.noop; - } - } - - // An object can be passed to jQuery.data instead of a key/value pair; this gets - // shallow copied over onto the existing cache - if ( typeof name === "object" || typeof name === "function" ) { - if ( pvt ) { - cache[ id ][ internalKey ] = jQuery.extend(cache[ id ][ internalKey ], name); - } else { - cache[ id ] = jQuery.extend(cache[ id ], name); - } - } - - thisCache = cache[ id ]; - - // Internal jQuery data is stored in a separate object inside the object's data - // cache in order to avoid key collisions between internal data and user-defined - // data - if ( pvt ) { - if ( !thisCache[ internalKey ] ) { - thisCache[ internalKey ] = {}; - } - - thisCache = thisCache[ internalKey ]; - } - - if ( data !== undefined ) { - thisCache[ jQuery.camelCase( name ) ] = data; - } - - // TODO: This is a hack for 1.5 ONLY. It will be removed in 1.6. Users should - // not attempt to inspect the internal events object using jQuery.data, as this - // internal data object is undocumented and subject to change. - if ( name === "events" && !thisCache[name] ) { - return thisCache[ internalKey ] && thisCache[ internalKey ].events; - } - - return getByName ? - // Check for both converted-to-camel and non-converted data property names - thisCache[ jQuery.camelCase( name ) ] || thisCache[ name ] : - thisCache; - }, - - removeData: function( elem, name, pvt /* Internal Use Only */ ) { - if ( !jQuery.acceptData( elem ) ) { - return; - } - - var internalKey = jQuery.expando, isNode = elem.nodeType, - - // See jQuery.data for more information - cache = isNode ? jQuery.cache : elem, - - // See jQuery.data for more information - id = isNode ? elem[ jQuery.expando ] : jQuery.expando; - - // If there is already no cache entry for this object, there is no - // purpose in continuing - if ( !cache[ id ] ) { - return; - } - - if ( name ) { - var thisCache = pvt ? cache[ id ][ internalKey ] : cache[ id ]; - - if ( thisCache ) { - delete thisCache[ name ]; - - // If there is no data left in the cache, we want to continue - // and let the cache object itself get destroyed - if ( !isEmptyDataObject(thisCache) ) { - return; - } - } - } - - // See jQuery.data for more information - if ( pvt ) { - delete cache[ id ][ internalKey ]; - - // Don't destroy the parent cache unless the internal data object - // had been the only thing left in it - if ( !isEmptyDataObject(cache[ id ]) ) { - return; - } - } - - var internalCache = cache[ id ][ internalKey ]; - - // Browsers that fail expando deletion also refuse to delete expandos on - // the window, but it will allow it on all other JS objects; other browsers - // don't care - if ( jQuery.support.deleteExpando || cache != window ) { - delete cache[ id ]; - } else { - cache[ id ] = null; - } - - // We destroyed the entire user cache at once because it's faster than - // iterating through each key, but we need to continue to persist internal - // data if it existed - if ( internalCache ) { - cache[ id ] = {}; - // TODO: This is a hack for 1.5 ONLY. Avoids exposing jQuery - // metadata on plain JS objects when the object is serialized using - // JSON.stringify - if ( !isNode ) { - cache[ id ].toJSON = jQuery.noop; - } - - cache[ id ][ internalKey ] = internalCache; - - // Otherwise, we need to eliminate the expando on the node to avoid - // false lookups in the cache for entries that no longer exist - } else if ( isNode ) { - // IE does not allow us to delete expando properties from nodes, - // nor does it have a removeAttribute function on Document nodes; - // we must handle all of these cases - if ( jQuery.support.deleteExpando ) { - delete elem[ jQuery.expando ]; - } else if ( elem.removeAttribute ) { - elem.removeAttribute( jQuery.expando ); - } else { - elem[ jQuery.expando ] = null; - } - } - }, - - // For internal use only. - _data: function( elem, name, data ) { - return jQuery.data( elem, name, data, true ); - }, - - // A method for determining if a DOM node can handle the data expando - acceptData: function( elem ) { - if ( elem.nodeName ) { - var match = jQuery.noData[ elem.nodeName.toLowerCase() ]; - - if ( match ) { - return !(match === true || elem.getAttribute("classid") !== match); - } - } - - return true; - } -}); - -jQuery.fn.extend({ - data: function( key, value ) { - var data = null; - - if ( typeof key === "undefined" ) { - if ( this.length ) { - data = jQuery.data( this[0] ); - - if ( this[0].nodeType === 1 ) { - var attr = this[0].attributes, name; - for ( var i = 0, l = attr.length; i < l; i++ ) { - name = attr[i].name; - - if ( name.indexOf( "data-" ) === 0 ) { - name = jQuery.camelCase( name.substring(5) ); - - dataAttr( this[0], name, data[ name ] ); - } - } - } - } - - return data; - - } else if ( typeof key === "object" ) { - return this.each(function() { - jQuery.data( this, key ); - }); - } - - var parts = key.split("."); - parts[1] = parts[1] ? "." + parts[1] : ""; - - if ( value === undefined ) { - data = this.triggerHandler("getData" + parts[1] + "!", [parts[0]]); - - // Try to fetch any internally stored data first - if ( data === undefined && this.length ) { - data = jQuery.data( this[0], key ); - data = dataAttr( this[0], key, data ); - } - - return data === undefined && parts[1] ? - this.data( parts[0] ) : - data; - - } else { - return this.each(function() { - var $this = jQuery( this ), - args = [ parts[0], value ]; - - $this.triggerHandler( "setData" + parts[1] + "!", args ); - jQuery.data( this, key, value ); - $this.triggerHandler( "changeData" + parts[1] + "!", args ); - }); - } - }, - - removeData: function( key ) { - return this.each(function() { - jQuery.removeData( this, key ); - }); - } -}); - -function dataAttr( elem, key, data ) { - // If nothing was found internally, try to fetch any - // data from the HTML5 data-* attribute - if ( data === undefined && elem.nodeType === 1 ) { - var name = "data-" + key.replace( rmultiDash, "$1-$2" ).toLowerCase(); - - data = elem.getAttribute( name ); - - if ( typeof data === "string" ) { - try { - data = data === "true" ? true : - data === "false" ? false : - data === "null" ? null : - !jQuery.isNaN( data ) ? parseFloat( data ) : - rbrace.test( data ) ? jQuery.parseJSON( data ) : - data; - } catch( e ) {} - - // Make sure we set the data so it isn't changed later - jQuery.data( elem, key, data ); - - } else { - data = undefined; - } - } - - return data; -} - -// TODO: This is a hack for 1.5 ONLY to allow objects with a single toJSON -// property to be considered empty objects; this property always exists in -// order to make sure JSON.stringify does not expose internal metadata -function isEmptyDataObject( obj ) { - for ( var name in obj ) { - if ( name !== "toJSON" ) { - return false; - } - } - - return true; -} - - - - -function handleQueueMarkDefer( elem, type, src ) { - var deferDataKey = type + "defer", - queueDataKey = type + "queue", - markDataKey = type + "mark", - defer = jQuery.data( elem, deferDataKey, undefined, true ); - if ( defer && - ( src === "queue" || !jQuery.data( elem, queueDataKey, undefined, true ) ) && - ( src === "mark" || !jQuery.data( elem, markDataKey, undefined, true ) ) ) { - // Give room for hard-coded callbacks to fire first - // and eventually mark/queue something else on the element - setTimeout( function() { - if ( !jQuery.data( elem, queueDataKey, undefined, true ) && - !jQuery.data( elem, markDataKey, undefined, true ) ) { - jQuery.removeData( elem, deferDataKey, true ); - defer.resolve(); - } - }, 0 ); - } -} - -jQuery.extend({ - - _mark: function( elem, type ) { - if ( elem ) { - type = (type || "fx") + "mark"; - jQuery.data( elem, type, (jQuery.data(elem,type,undefined,true) || 0) + 1, true ); - } - }, - - _unmark: function( force, elem, type ) { - if ( force !== true ) { - type = elem; - elem = force; - force = false; - } - if ( elem ) { - type = type || "fx"; - var key = type + "mark", - count = force ? 0 : ( (jQuery.data( elem, key, undefined, true) || 1 ) - 1 ); - if ( count ) { - jQuery.data( elem, key, count, true ); - } else { - jQuery.removeData( elem, key, true ); - handleQueueMarkDefer( elem, type, "mark" ); - } - } - }, - - queue: function( elem, type, data ) { - if ( elem ) { - type = (type || "fx") + "queue"; - var q = jQuery.data( elem, type, undefined, true ); - // Speed up dequeue by getting out quickly if this is just a lookup - if ( data ) { - if ( !q || jQuery.isArray(data) ) { - q = jQuery.data( elem, type, jQuery.makeArray(data), true ); - } else { - q.push( data ); - } - } - return q || []; - } - }, - - dequeue: function( elem, type ) { - type = type || "fx"; - - var queue = jQuery.queue( elem, type ), - fn = queue.shift(), - defer; - - // If the fx queue is dequeued, always remove the progress sentinel - if ( fn === "inprogress" ) { - fn = queue.shift(); - } - - if ( fn ) { - // Add a progress sentinel to prevent the fx queue from being - // automatically dequeued - if ( type === "fx" ) { - queue.unshift("inprogress"); - } - - fn.call(elem, function() { - jQuery.dequeue(elem, type); - }); - } - - if ( !queue.length ) { - jQuery.removeData( elem, type + "queue", true ); - handleQueueMarkDefer( elem, type, "queue" ); - } - } -}); - -jQuery.fn.extend({ - queue: function( type, data ) { - if ( typeof type !== "string" ) { - data = type; - type = "fx"; - } - - if ( data === undefined ) { - return jQuery.queue( this[0], type ); - } - return this.each(function() { - var queue = jQuery.queue( this, type, data ); - - if ( type === "fx" && queue[0] !== "inprogress" ) { - jQuery.dequeue( this, type ); - } - }); - }, - dequeue: function( type ) { - return this.each(function() { - jQuery.dequeue( this, type ); - }); - }, - // Based off of the plugin by Clint Helfers, with permission. - // http://blindsignals.com/index.php/2009/07/jquery-delay/ - delay: function( time, type ) { - time = jQuery.fx ? jQuery.fx.speeds[time] || time : time; - type = type || "fx"; - - return this.queue( type, function() { - var elem = this; - setTimeout(function() { - jQuery.dequeue( elem, type ); - }, time ); - }); - }, - clearQueue: function( type ) { - return this.queue( type || "fx", [] ); - }, - // Get a promise resolved when queues of a certain type - // are emptied (fx is the type by default) - promise: function( type, object ) { - if ( typeof type !== "string" ) { - object = type; - type = undefined; - } - type = type || "fx"; - var defer = jQuery.Deferred(), - elements = this, - i = elements.length, - count = 1, - deferDataKey = type + "defer", - queueDataKey = type + "queue", - markDataKey = type + "mark", - tmp; - function resolve() { - if ( !( --count ) ) { - defer.resolveWith( elements, [ elements ] ); - } - } - while( i-- ) { - if (( tmp = jQuery.data( elements[ i ], deferDataKey, undefined, true ) || - ( jQuery.data( elements[ i ], queueDataKey, undefined, true ) || - jQuery.data( elements[ i ], markDataKey, undefined, true ) ) && - jQuery.data( elements[ i ], deferDataKey, jQuery._Deferred(), true ) )) { - count++; - tmp.done( resolve ); - } - } - resolve(); - return defer.promise(); - } -}); - - - - -var rclass = /[\n\t\r]/g, - rspace = /\s+/, - rreturn = /\r/g, - rtype = /^(?:button|input)$/i, - rfocusable = /^(?:button|input|object|select|textarea)$/i, - rclickable = /^a(?:rea)?$/i, - rboolean = /^(?:autofocus|autoplay|async|checked|controls|defer|disabled|hidden|loop|multiple|open|readonly|required|scoped|selected)$/i, - rinvalidChar = /\:|^on/, - formHook, boolHook; - -jQuery.fn.extend({ - attr: function( name, value ) { - return jQuery.access( this, name, value, true, jQuery.attr ); - }, - - removeAttr: function( name ) { - return this.each(function() { - jQuery.removeAttr( this, name ); - }); - }, - - prop: function( name, value ) { - return jQuery.access( this, name, value, true, jQuery.prop ); - }, - - removeProp: function( name ) { - name = jQuery.propFix[ name ] || name; - return this.each(function() { - // try/catch handles cases where IE balks (such as removing a property on window) - try { - this[ name ] = undefined; - delete this[ name ]; - } catch( e ) {} - }); - }, - - addClass: function( value ) { - var classNames, i, l, elem, - setClass, c, cl; - - if ( jQuery.isFunction( value ) ) { - return this.each(function( j ) { - jQuery( this ).addClass( value.call(this, j, this.className) ); - }); - } - - if ( value && typeof value === "string" ) { - classNames = value.split( rspace ); - - for ( i = 0, l = this.length; i < l; i++ ) { - elem = this[ i ]; - - if ( elem.nodeType === 1 ) { - if ( !elem.className && classNames.length === 1 ) { - elem.className = value; - - } else { - setClass = " " + elem.className + " "; - - for ( c = 0, cl = classNames.length; c < cl; c++ ) { - if ( !~setClass.indexOf( " " + classNames[ c ] + " " ) ) { - setClass += classNames[ c ] + " "; - } - } - elem.className = jQuery.trim( setClass ); - } - } - } - } - - return this; - }, - - removeClass: function( value ) { - var classNames, i, l, elem, className, c, cl; - - if ( jQuery.isFunction( value ) ) { - return this.each(function( j ) { - jQuery( this ).removeClass( value.call(this, j, this.className) ); - }); - } - - if ( (value && typeof value === "string") || value === undefined ) { - classNames = (value || "").split( rspace ); - - for ( i = 0, l = this.length; i < l; i++ ) { - elem = this[ i ]; - - if ( elem.nodeType === 1 && elem.className ) { - if ( value ) { - className = (" " + elem.className + " ").replace( rclass, " " ); - for ( c = 0, cl = classNames.length; c < cl; c++ ) { - className = className.replace(" " + classNames[ c ] + " ", " "); - } - elem.className = jQuery.trim( className ); - - } else { - elem.className = ""; - } - } - } - } - - return this; - }, - - toggleClass: function( value, stateVal ) { - var type = typeof value, - isBool = typeof stateVal === "boolean"; - - if ( jQuery.isFunction( value ) ) { - return this.each(function( i ) { - jQuery( this ).toggleClass( value.call(this, i, this.className, stateVal), stateVal ); - }); - } - - return this.each(function() { - if ( type === "string" ) { - // toggle individual class names - var className, - i = 0, - self = jQuery( this ), - state = stateVal, - classNames = value.split( rspace ); - - while ( (className = classNames[ i++ ]) ) { - // check each className given, space seperated list - state = isBool ? state : !self.hasClass( className ); - self[ state ? "addClass" : "removeClass" ]( className ); - } - - } else if ( type === "undefined" || type === "boolean" ) { - if ( this.className ) { - // store className if set - jQuery._data( this, "__className__", this.className ); - } - - // toggle whole className - this.className = this.className || value === false ? "" : jQuery._data( this, "__className__" ) || ""; - } - }); - }, - - hasClass: function( selector ) { - var className = " " + selector + " "; - for ( var i = 0, l = this.length; i < l; i++ ) { - if ( (" " + this[i].className + " ").replace(rclass, " ").indexOf( className ) > -1 ) { - return true; - } - } - - return false; - }, - - val: function( value ) { - var hooks, ret, - elem = this[0]; - - if ( !arguments.length ) { - if ( elem ) { - hooks = jQuery.valHooks[ elem.nodeName.toLowerCase() ] || jQuery.valHooks[ elem.type ]; - - if ( hooks && "get" in hooks && (ret = hooks.get( elem, "value" )) !== undefined ) { - return ret; - } - - ret = elem.value; - - return typeof ret === "string" ? - // handle most common string cases - ret.replace(rreturn, "") : - // handle cases where value is null/undef or number - ret == null ? "" : ret; - } - - return undefined; - } - - var isFunction = jQuery.isFunction( value ); - - return this.each(function( i ) { - var self = jQuery(this), val; - - if ( this.nodeType !== 1 ) { - return; - } - - if ( isFunction ) { - val = value.call( this, i, self.val() ); - } else { - val = value; - } - - // Treat null/undefined as ""; convert numbers to string - if ( val == null ) { - val = ""; - } else if ( typeof val === "number" ) { - val += ""; - } else if ( jQuery.isArray( val ) ) { - val = jQuery.map(val, function ( value ) { - return value == null ? "" : value + ""; - }); - } - - hooks = jQuery.valHooks[ this.nodeName.toLowerCase() ] || jQuery.valHooks[ this.type ]; - - // If set returns undefined, fall back to normal setting - if ( !hooks || !("set" in hooks) || hooks.set( this, val, "value" ) === undefined ) { - this.value = val; - } - }); - } -}); - -jQuery.extend({ - valHooks: { - option: { - get: function( elem ) { - // attributes.value is undefined in Blackberry 4.7 but - // uses .value. See #6932 - var val = elem.attributes.value; - return !val || val.specified ? elem.value : elem.text; - } - }, - select: { - get: function( elem ) { - var value, - index = elem.selectedIndex, - values = [], - options = elem.options, - one = elem.type === "select-one"; - - // Nothing was selected - if ( index < 0 ) { - return null; - } - - // Loop through all the selected options - for ( var i = one ? index : 0, max = one ? index + 1 : options.length; i < max; i++ ) { - var option = options[ i ]; - - // Don't return options that are disabled or in a disabled optgroup - if ( option.selected && (jQuery.support.optDisabled ? !option.disabled : option.getAttribute("disabled") === null) && - (!option.parentNode.disabled || !jQuery.nodeName( option.parentNode, "optgroup" )) ) { - - // Get the specific value for the option - value = jQuery( option ).val(); - - // We don't need an array for one selects - if ( one ) { - return value; - } - - // Multi-Selects return an array - values.push( value ); - } - } - - // Fixes Bug #2551 -- select.val() broken in IE after form.reset() - if ( one && !values.length && options.length ) { - return jQuery( options[ index ] ).val(); - } - - return values; - }, - - set: function( elem, value ) { - var values = jQuery.makeArray( value ); - - jQuery(elem).find("option").each(function() { - this.selected = jQuery.inArray( jQuery(this).val(), values ) >= 0; - }); - - if ( !values.length ) { - elem.selectedIndex = -1; - } - return values; - } - } - }, - - attrFn: { - val: true, - css: true, - html: true, - text: true, - data: true, - width: true, - height: true, - offset: true - }, - - attrFix: { - // Always normalize to ensure hook usage - tabindex: "tabIndex" - }, - - attr: function( elem, name, value, pass ) { - var nType = elem.nodeType; - - // don't get/set attributes on text, comment and attribute nodes - if ( !elem || nType === 3 || nType === 8 || nType === 2 ) { - return undefined; - } - - if ( pass && name in jQuery.attrFn ) { - return jQuery( elem )[ name ]( value ); - } - - // Fallback to prop when attributes are not supported - if ( !("getAttribute" in elem) ) { - return jQuery.prop( elem, name, value ); - } - - var ret, hooks, - notxml = nType !== 1 || !jQuery.isXMLDoc( elem ); - - // Normalize the name if needed - if ( notxml ) { - name = jQuery.attrFix[ name ] || name; - - hooks = jQuery.attrHooks[ name ]; - - if ( !hooks ) { - // Use boolHook for boolean attributes - if ( rboolean.test( name ) ) { - - hooks = boolHook; - - // Use formHook for forms and if the name contains certain characters - } else if ( formHook && name !== "className" && - (jQuery.nodeName( elem, "form" ) || rinvalidChar.test( name )) ) { - - hooks = formHook; - } - } - } - - if ( value !== undefined ) { - - if ( value === null ) { - jQuery.removeAttr( elem, name ); - return undefined; - - } else if ( hooks && "set" in hooks && notxml && (ret = hooks.set( elem, value, name )) !== undefined ) { - return ret; - - } else { - elem.setAttribute( name, "" + value ); - return value; - } - - } else if ( hooks && "get" in hooks && notxml && (ret = hooks.get( elem, name )) !== null ) { - return ret; - - } else { - - ret = elem.getAttribute( name ); - - // Non-existent attributes return null, we normalize to undefined - return ret === null ? - undefined : - ret; - } - }, - - removeAttr: function( elem, name ) { - var propName; - if ( elem.nodeType === 1 ) { - name = jQuery.attrFix[ name ] || name; - - if ( jQuery.support.getSetAttribute ) { - // Use removeAttribute in browsers that support it - elem.removeAttribute( name ); - } else { - jQuery.attr( elem, name, "" ); - elem.removeAttributeNode( elem.getAttributeNode( name ) ); - } - - // Set corresponding property to false for boolean attributes - if ( rboolean.test( name ) && (propName = jQuery.propFix[ name ] || name) in elem ) { - elem[ propName ] = false; - } - } - }, - - attrHooks: { - type: { - set: function( elem, value ) { - // We can't allow the type property to be changed (since it causes problems in IE) - if ( rtype.test( elem.nodeName ) && elem.parentNode ) { - jQuery.error( "type property can't be changed" ); - } else if ( !jQuery.support.radioValue && value === "radio" && jQuery.nodeName(elem, "input") ) { - // Setting the type on a radio button after the value resets the value in IE6-9 - // Reset value to it's default in case type is set after value - // This is for element creation - var val = elem.value; - elem.setAttribute( "type", value ); - if ( val ) { - elem.value = val; - } - return value; - } - } - }, - tabIndex: { - get: function( elem ) { - // elem.tabIndex doesn't always return the correct value when it hasn't been explicitly set - // http://fluidproject.org/blog/2008/01/09/getting-setting-and-removing-tabindex-values-with-javascript/ - var attributeNode = elem.getAttributeNode("tabIndex"); - - return attributeNode && attributeNode.specified ? - parseInt( attributeNode.value, 10 ) : - rfocusable.test( elem.nodeName ) || rclickable.test( elem.nodeName ) && elem.href ? - 0 : - undefined; - } - }, - // Use the value property for back compat - // Use the formHook for button elements in IE6/7 (#1954) - value: { - get: function( elem, name ) { - if ( formHook && jQuery.nodeName( elem, "button" ) ) { - return formHook.get( elem, name ); - } - return name in elem ? - elem.value : - null; - }, - set: function( elem, value, name ) { - if ( formHook && jQuery.nodeName( elem, "button" ) ) { - return formHook.set( elem, value, name ); - } - // Does not return so that setAttribute is also used - elem.value = value; - } - } - }, - - propFix: { - tabindex: "tabIndex", - readonly: "readOnly", - "for": "htmlFor", - "class": "className", - maxlength: "maxLength", - cellspacing: "cellSpacing", - cellpadding: "cellPadding", - rowspan: "rowSpan", - colspan: "colSpan", - usemap: "useMap", - frameborder: "frameBorder", - contenteditable: "contentEditable" - }, - - prop: function( elem, name, value ) { - var nType = elem.nodeType; - - // don't get/set properties on text, comment and attribute nodes - if ( !elem || nType === 3 || nType === 8 || nType === 2 ) { - return undefined; - } - - var ret, hooks, - notxml = nType !== 1 || !jQuery.isXMLDoc( elem ); - - if ( notxml ) { - // Fix name and attach hooks - name = jQuery.propFix[ name ] || name; - hooks = jQuery.propHooks[ name ]; - } - - if ( value !== undefined ) { - if ( hooks && "set" in hooks && (ret = hooks.set( elem, value, name )) !== undefined ) { - return ret; - - } else { - return (elem[ name ] = value); - } - - } else { - if ( hooks && "get" in hooks && (ret = hooks.get( elem, name )) !== undefined ) { - return ret; - - } else { - return elem[ name ]; - } - } - }, - - propHooks: {} -}); - -// Hook for boolean attributes -boolHook = { - get: function( elem, name ) { - // Align boolean attributes with corresponding properties - return jQuery.prop( elem, name ) ? - name.toLowerCase() : - undefined; - }, - set: function( elem, value, name ) { - var propName; - if ( value === false ) { - // Remove boolean attributes when set to false - jQuery.removeAttr( elem, name ); - } else { - // value is true since we know at this point it's type boolean and not false - // Set boolean attributes to the same name and set the DOM property - propName = jQuery.propFix[ name ] || name; - if ( propName in elem ) { - // Only set the IDL specifically if it already exists on the element - elem[ propName ] = true; - } - - elem.setAttribute( name, name.toLowerCase() ); - } - return name; - } -}; - -// IE6/7 do not support getting/setting some attributes with get/setAttribute -if ( !jQuery.support.getSetAttribute ) { - - // propFix is more comprehensive and contains all fixes - jQuery.attrFix = jQuery.propFix; - - // Use this for any attribute on a form in IE6/7 - formHook = jQuery.attrHooks.name = jQuery.attrHooks.title = jQuery.valHooks.button = { - get: function( elem, name ) { - var ret; - ret = elem.getAttributeNode( name ); - // Return undefined if nodeValue is empty string - return ret && ret.nodeValue !== "" ? - ret.nodeValue : - undefined; - }, - set: function( elem, value, name ) { - // Check form objects in IE (multiple bugs related) - // Only use nodeValue if the attribute node exists on the form - var ret = elem.getAttributeNode( name ); - if ( ret ) { - ret.nodeValue = value; - return value; - } - } - }; - - // Set width and height to auto instead of 0 on empty string( Bug #8150 ) - // This is for removals - jQuery.each([ "width", "height" ], function( i, name ) { - jQuery.attrHooks[ name ] = jQuery.extend( jQuery.attrHooks[ name ], { - set: function( elem, value ) { - if ( value === "" ) { - elem.setAttribute( name, "auto" ); - return value; - } - } - }); - }); -} - - -// Some attributes require a special call on IE -if ( !jQuery.support.hrefNormalized ) { - jQuery.each([ "href", "src", "width", "height" ], function( i, name ) { - jQuery.attrHooks[ name ] = jQuery.extend( jQuery.attrHooks[ name ], { - get: function( elem ) { - var ret = elem.getAttribute( name, 2 ); - return ret === null ? undefined : ret; - } - }); - }); -} - -if ( !jQuery.support.style ) { - jQuery.attrHooks.style = { - get: function( elem ) { - // Return undefined in the case of empty string - // Normalize to lowercase since IE uppercases css property names - return elem.style.cssText.toLowerCase() || undefined; - }, - set: function( elem, value ) { - return (elem.style.cssText = "" + value); - } - }; -} - -// Safari mis-reports the default selected property of an option -// Accessing the parent's selectedIndex property fixes it -if ( !jQuery.support.optSelected ) { - jQuery.propHooks.selected = jQuery.extend( jQuery.propHooks.selected, { - get: function( elem ) { - var parent = elem.parentNode; - - if ( parent ) { - parent.selectedIndex; - - // Make sure that it also works with optgroups, see #5701 - if ( parent.parentNode ) { - parent.parentNode.selectedIndex; - } - } - } - }); -} - -// Radios and checkboxes getter/setter -if ( !jQuery.support.checkOn ) { - jQuery.each([ "radio", "checkbox" ], function() { - jQuery.valHooks[ this ] = { - get: function( elem ) { - // Handle the case where in Webkit "" is returned instead of "on" if a value isn't specified - return elem.getAttribute("value") === null ? "on" : elem.value; - } - }; - }); -} -jQuery.each([ "radio", "checkbox" ], function() { - jQuery.valHooks[ this ] = jQuery.extend( jQuery.valHooks[ this ], { - set: function( elem, value ) { - if ( jQuery.isArray( value ) ) { - return (elem.checked = jQuery.inArray( jQuery(elem).val(), value ) >= 0); - } - } - }); -}); - - - - -var rnamespaces = /\.(.*)$/, - rformElems = /^(?:textarea|input|select)$/i, - rperiod = /\./g, - rspaces = / /g, - rescape = /[^\w\s.|`]/g, - fcleanup = function( nm ) { - return nm.replace(rescape, "\\$&"); - }; - -/* - * A number of helper functions used for managing events. - * Many of the ideas behind this code originated from - * Dean Edwards' addEvent library. - */ -jQuery.event = { - - // Bind an event to an element - // Original by Dean Edwards - add: function( elem, types, handler, data ) { - if ( elem.nodeType === 3 || elem.nodeType === 8 ) { - return; - } - - if ( handler === false ) { - handler = returnFalse; - } else if ( !handler ) { - // Fixes bug #7229. Fix recommended by jdalton - return; - } - - var handleObjIn, handleObj; - - if ( handler.handler ) { - handleObjIn = handler; - handler = handleObjIn.handler; - } - - // Make sure that the function being executed has a unique ID - if ( !handler.guid ) { - handler.guid = jQuery.guid++; - } - - // Init the element's event structure - var elemData = jQuery._data( elem ); - - // If no elemData is found then we must be trying to bind to one of the - // banned noData elements - if ( !elemData ) { - return; - } - - var events = elemData.events, - eventHandle = elemData.handle; - - if ( !events ) { - elemData.events = events = {}; - } - - if ( !eventHandle ) { - elemData.handle = eventHandle = function( e ) { - // Discard the second event of a jQuery.event.trigger() and - // when an event is called after a page has unloaded - return typeof jQuery !== "undefined" && (!e || jQuery.event.triggered !== e.type) ? - jQuery.event.handle.apply( eventHandle.elem, arguments ) : - undefined; - }; - } - - // Add elem as a property of the handle function - // This is to prevent a memory leak with non-native events in IE. - eventHandle.elem = elem; - - // Handle multiple events separated by a space - // jQuery(...).bind("mouseover mouseout", fn); - types = types.split(" "); - - var type, i = 0, namespaces; - - while ( (type = types[ i++ ]) ) { - handleObj = handleObjIn ? - jQuery.extend({}, handleObjIn) : - { handler: handler, data: data }; - - // Namespaced event handlers - if ( type.indexOf(".") > -1 ) { - namespaces = type.split("."); - type = namespaces.shift(); - handleObj.namespace = namespaces.slice(0).sort().join("."); - - } else { - namespaces = []; - handleObj.namespace = ""; - } - - handleObj.type = type; - if ( !handleObj.guid ) { - handleObj.guid = handler.guid; - } - - // Get the current list of functions bound to this event - var handlers = events[ type ], - special = jQuery.event.special[ type ] || {}; - - // Init the event handler queue - if ( !handlers ) { - handlers = events[ type ] = []; - - // Check for a special event handler - // Only use addEventListener/attachEvent if the special - // events handler returns false - if ( !special.setup || special.setup.call( elem, data, namespaces, eventHandle ) === false ) { - // Bind the global event handler to the element - if ( elem.addEventListener ) { - elem.addEventListener( type, eventHandle, false ); - - } else if ( elem.attachEvent ) { - elem.attachEvent( "on" + type, eventHandle ); - } - } - } - - if ( special.add ) { - special.add.call( elem, handleObj ); - - if ( !handleObj.handler.guid ) { - handleObj.handler.guid = handler.guid; - } - } - - // Add the function to the element's handler list - handlers.push( handleObj ); - - // Keep track of which events have been used, for event optimization - jQuery.event.global[ type ] = true; - } - - // Nullify elem to prevent memory leaks in IE - elem = null; - }, - - global: {}, - - // Detach an event or set of events from an element - remove: function( elem, types, handler, pos ) { - // don't do events on text and comment nodes - if ( elem.nodeType === 3 || elem.nodeType === 8 ) { - return; - } - - if ( handler === false ) { - handler = returnFalse; - } - - var ret, type, fn, j, i = 0, all, namespaces, namespace, special, eventType, handleObj, origType, - elemData = jQuery.hasData( elem ) && jQuery._data( elem ), - events = elemData && elemData.events; - - if ( !elemData || !events ) { - return; - } - - // types is actually an event object here - if ( types && types.type ) { - handler = types.handler; - types = types.type; - } - - // Unbind all events for the element - if ( !types || typeof types === "string" && types.charAt(0) === "." ) { - types = types || ""; - - for ( type in events ) { - jQuery.event.remove( elem, type + types ); - } - - return; - } - - // Handle multiple events separated by a space - // jQuery(...).unbind("mouseover mouseout", fn); - types = types.split(" "); - - while ( (type = types[ i++ ]) ) { - origType = type; - handleObj = null; - all = type.indexOf(".") < 0; - namespaces = []; - - if ( !all ) { - // Namespaced event handlers - namespaces = type.split("."); - type = namespaces.shift(); - - namespace = new RegExp("(^|\\.)" + - jQuery.map( namespaces.slice(0).sort(), fcleanup ).join("\\.(?:.*\\.)?") + "(\\.|$)"); - } - - eventType = events[ type ]; - - if ( !eventType ) { - continue; - } - - if ( !handler ) { - for ( j = 0; j < eventType.length; j++ ) { - handleObj = eventType[ j ]; - - if ( all || namespace.test( handleObj.namespace ) ) { - jQuery.event.remove( elem, origType, handleObj.handler, j ); - eventType.splice( j--, 1 ); - } - } - - continue; - } - - special = jQuery.event.special[ type ] || {}; - - for ( j = pos || 0; j < eventType.length; j++ ) { - handleObj = eventType[ j ]; - - if ( handler.guid === handleObj.guid ) { - // remove the given handler for the given type - if ( all || namespace.test( handleObj.namespace ) ) { - if ( pos == null ) { - eventType.splice( j--, 1 ); - } - - if ( special.remove ) { - special.remove.call( elem, handleObj ); - } - } - - if ( pos != null ) { - break; - } - } - } - - // remove generic event handler if no more handlers exist - if ( eventType.length === 0 || pos != null && eventType.length === 1 ) { - if ( !special.teardown || special.teardown.call( elem, namespaces ) === false ) { - jQuery.removeEvent( elem, type, elemData.handle ); - } - - ret = null; - delete events[ type ]; - } - } - - // Remove the expando if it's no longer used - if ( jQuery.isEmptyObject( events ) ) { - var handle = elemData.handle; - if ( handle ) { - handle.elem = null; - } - - delete elemData.events; - delete elemData.handle; - - if ( jQuery.isEmptyObject( elemData ) ) { - jQuery.removeData( elem, undefined, true ); - } - } - }, - - // Events that are safe to short-circuit if no handlers are attached. - // Native DOM events should not be added, they may have inline handlers. - customEvent: { - "getData": true, - "setData": true, - "changeData": true - }, - - trigger: function( event, data, elem, onlyHandlers ) { - // Event object or event type - var type = event.type || event, - namespaces = [], - exclusive; - - if ( type.indexOf("!") >= 0 ) { - // Exclusive events trigger only for the exact event (no namespaces) - type = type.slice(0, -1); - exclusive = true; - } - - if ( type.indexOf(".") >= 0 ) { - // Namespaced trigger; create a regexp to match event type in handle() - namespaces = type.split("."); - type = namespaces.shift(); - namespaces.sort(); - } - - if ( (!elem || jQuery.event.customEvent[ type ]) && !jQuery.event.global[ type ] ) { - // No jQuery handlers for this event type, and it can't have inline handlers - return; - } - - // Caller can pass in an Event, Object, or just an event type string - event = typeof event === "object" ? - // jQuery.Event object - event[ jQuery.expando ] ? event : - // Object literal - new jQuery.Event( type, event ) : - // Just the event type (string) - new jQuery.Event( type ); - - event.type = type; - event.exclusive = exclusive; - event.namespace = namespaces.join("."); - event.namespace_re = new RegExp("(^|\\.)" + namespaces.join("\\.(?:.*\\.)?") + "(\\.|$)"); - - // triggerHandler() and global events don't bubble or run the default action - if ( onlyHandlers || !elem ) { - event.preventDefault(); - event.stopPropagation(); - } - - // Handle a global trigger - if ( !elem ) { - // TODO: Stop taunting the data cache; remove global events and always attach to document - jQuery.each( jQuery.cache, function() { - // internalKey variable is just used to make it easier to find - // and potentially change this stuff later; currently it just - // points to jQuery.expando - var internalKey = jQuery.expando, - internalCache = this[ internalKey ]; - if ( internalCache && internalCache.events && internalCache.events[ type ] ) { - jQuery.event.trigger( event, data, internalCache.handle.elem ); - } - }); - return; - } - - // Don't do events on text and comment nodes - if ( elem.nodeType === 3 || elem.nodeType === 8 ) { - return; - } - - // Clean up the event in case it is being reused - event.result = undefined; - event.target = elem; - - // Clone any incoming data and prepend the event, creating the handler arg list - data = data != null ? jQuery.makeArray( data ) : []; - data.unshift( event ); - - var cur = elem, - // IE doesn't like method names with a colon (#3533, #8272) - ontype = type.indexOf(":") < 0 ? "on" + type : ""; - - // Fire event on the current element, then bubble up the DOM tree - do { - var handle = jQuery._data( cur, "handle" ); - - event.currentTarget = cur; - if ( handle ) { - handle.apply( cur, data ); - } - - // Trigger an inline bound script - if ( ontype && jQuery.acceptData( cur ) && cur[ ontype ] && cur[ ontype ].apply( cur, data ) === false ) { - event.result = false; - event.preventDefault(); - } - - // Bubble up to document, then to window - cur = cur.parentNode || cur.ownerDocument || cur === event.target.ownerDocument && window; - } while ( cur && !event.isPropagationStopped() ); - - // If nobody prevented the default action, do it now - if ( !event.isDefaultPrevented() ) { - var old, - special = jQuery.event.special[ type ] || {}; - - if ( (!special._default || special._default.call( elem.ownerDocument, event ) === false) && - !(type === "click" && jQuery.nodeName( elem, "a" )) && jQuery.acceptData( elem ) ) { - - // Call a native DOM method on the target with the same name name as the event. - // Can't use an .isFunction)() check here because IE6/7 fails that test. - // IE<9 dies on focus to hidden element (#1486), may want to revisit a try/catch. - try { - if ( ontype && elem[ type ] ) { - // Don't re-trigger an onFOO event when we call its FOO() method - old = elem[ ontype ]; - - if ( old ) { - elem[ ontype ] = null; - } - - jQuery.event.triggered = type; - elem[ type ](); - } - } catch ( ieError ) {} - - if ( old ) { - elem[ ontype ] = old; - } - - jQuery.event.triggered = undefined; - } - } - - return event.result; - }, - - handle: function( event ) { - event = jQuery.event.fix( event || window.event ); - // Snapshot the handlers list since a called handler may add/remove events. - var handlers = ((jQuery._data( this, "events" ) || {})[ event.type ] || []).slice(0), - run_all = !event.exclusive && !event.namespace, - args = Array.prototype.slice.call( arguments, 0 ); - - // Use the fix-ed Event rather than the (read-only) native event - args[0] = event; - event.currentTarget = this; - - for ( var j = 0, l = handlers.length; j < l; j++ ) { - var handleObj = handlers[ j ]; - - // Triggered event must 1) be non-exclusive and have no namespace, or - // 2) have namespace(s) a subset or equal to those in the bound event. - if ( run_all || event.namespace_re.test( handleObj.namespace ) ) { - // Pass in a reference to the handler function itself - // So that we can later remove it - event.handler = handleObj.handler; - event.data = handleObj.data; - event.handleObj = handleObj; - - var ret = handleObj.handler.apply( this, args ); - - if ( ret !== undefined ) { - event.result = ret; - if ( ret === false ) { - event.preventDefault(); - event.stopPropagation(); - } - } - - if ( event.isImmediatePropagationStopped() ) { - break; - } - } - } - return event.result; - }, - - props: "altKey attrChange attrName bubbles button cancelable charCode clientX clientY ctrlKey currentTarget data detail eventPhase fromElement handler keyCode layerX layerY metaKey newValue offsetX offsetY pageX pageY prevValue relatedNode relatedTarget screenX screenY shiftKey srcElement target toElement view wheelDelta which".split(" "), - - fix: function( event ) { - if ( event[ jQuery.expando ] ) { - return event; - } - - // store a copy of the original event object - // and "clone" to set read-only properties - var originalEvent = event; - event = jQuery.Event( originalEvent ); - - for ( var i = this.props.length, prop; i; ) { - prop = this.props[ --i ]; - event[ prop ] = originalEvent[ prop ]; - } - - // Fix target property, if necessary - if ( !event.target ) { - // Fixes #1925 where srcElement might not be defined either - event.target = event.srcElement || document; - } - - // check if target is a textnode (safari) - if ( event.target.nodeType === 3 ) { - event.target = event.target.parentNode; - } - - // Add relatedTarget, if necessary - if ( !event.relatedTarget && event.fromElement ) { - event.relatedTarget = event.fromElement === event.target ? event.toElement : event.fromElement; - } - - // Calculate pageX/Y if missing and clientX/Y available - if ( event.pageX == null && event.clientX != null ) { - var eventDocument = event.target.ownerDocument || document, - doc = eventDocument.documentElement, - body = eventDocument.body; - - event.pageX = event.clientX + (doc && doc.scrollLeft || body && body.scrollLeft || 0) - (doc && doc.clientLeft || body && body.clientLeft || 0); - event.pageY = event.clientY + (doc && doc.scrollTop || body && body.scrollTop || 0) - (doc && doc.clientTop || body && body.clientTop || 0); - } - - // Add which for key events - if ( event.which == null && (event.charCode != null || event.keyCode != null) ) { - event.which = event.charCode != null ? event.charCode : event.keyCode; - } - - // Add metaKey to non-Mac browsers (use ctrl for PC's and Meta for Macs) - if ( !event.metaKey && event.ctrlKey ) { - event.metaKey = event.ctrlKey; - } - - // Add which for click: 1 === left; 2 === middle; 3 === right - // Note: button is not normalized, so don't use it - if ( !event.which && event.button !== undefined ) { - event.which = (event.button & 1 ? 1 : ( event.button & 2 ? 3 : ( event.button & 4 ? 2 : 0 ) )); - } - - return event; - }, - - // Deprecated, use jQuery.guid instead - guid: 1E8, - - // Deprecated, use jQuery.proxy instead - proxy: jQuery.proxy, - - special: { - ready: { - // Make sure the ready event is setup - setup: jQuery.bindReady, - teardown: jQuery.noop - }, - - live: { - add: function( handleObj ) { - jQuery.event.add( this, - liveConvert( handleObj.origType, handleObj.selector ), - jQuery.extend({}, handleObj, {handler: liveHandler, guid: handleObj.handler.guid}) ); - }, - - remove: function( handleObj ) { - jQuery.event.remove( this, liveConvert( handleObj.origType, handleObj.selector ), handleObj ); - } - }, - - beforeunload: { - setup: function( data, namespaces, eventHandle ) { - // We only want to do this special case on windows - if ( jQuery.isWindow( this ) ) { - this.onbeforeunload = eventHandle; - } - }, - - teardown: function( namespaces, eventHandle ) { - if ( this.onbeforeunload === eventHandle ) { - this.onbeforeunload = null; - } - } - } - } -}; - -jQuery.removeEvent = document.removeEventListener ? - function( elem, type, handle ) { - if ( elem.removeEventListener ) { - elem.removeEventListener( type, handle, false ); - } - } : - function( elem, type, handle ) { - if ( elem.detachEvent ) { - elem.detachEvent( "on" + type, handle ); - } - }; - -jQuery.Event = function( src, props ) { - // Allow instantiation without the 'new' keyword - if ( !this.preventDefault ) { - return new jQuery.Event( src, props ); - } - - // Event object - if ( src && src.type ) { - this.originalEvent = src; - this.type = src.type; - - // Events bubbling up the document may have been marked as prevented - // by a handler lower down the tree; reflect the correct value. - this.isDefaultPrevented = (src.defaultPrevented || src.returnValue === false || - src.getPreventDefault && src.getPreventDefault()) ? returnTrue : returnFalse; - - // Event type - } else { - this.type = src; - } - - // Put explicitly provided properties onto the event object - if ( props ) { - jQuery.extend( this, props ); - } - - // timeStamp is buggy for some events on Firefox(#3843) - // So we won't rely on the native value - this.timeStamp = jQuery.now(); - - // Mark it as fixed - this[ jQuery.expando ] = true; -}; - -function returnFalse() { - return false; -} -function returnTrue() { - return true; -} - -// jQuery.Event is based on DOM3 Events as specified by the ECMAScript Language Binding -// http://www.w3.org/TR/2003/WD-DOM-Level-3-Events-20030331/ecma-script-binding.html -jQuery.Event.prototype = { - preventDefault: function() { - this.isDefaultPrevented = returnTrue; - - var e = this.originalEvent; - if ( !e ) { - return; - } - - // if preventDefault exists run it on the original event - if ( e.preventDefault ) { - e.preventDefault(); - - // otherwise set the returnValue property of the original event to false (IE) - } else { - e.returnValue = false; - } - }, - stopPropagation: function() { - this.isPropagationStopped = returnTrue; - - var e = this.originalEvent; - if ( !e ) { - return; - } - // if stopPropagation exists run it on the original event - if ( e.stopPropagation ) { - e.stopPropagation(); - } - // otherwise set the cancelBubble property of the original event to true (IE) - e.cancelBubble = true; - }, - stopImmediatePropagation: function() { - this.isImmediatePropagationStopped = returnTrue; - this.stopPropagation(); - }, - isDefaultPrevented: returnFalse, - isPropagationStopped: returnFalse, - isImmediatePropagationStopped: returnFalse -}; - -// Checks if an event happened on an element within another element -// Used in jQuery.event.special.mouseenter and mouseleave handlers -var withinElement = function( event ) { - - // Check if mouse(over|out) are still within the same parent element - var related = event.relatedTarget, - inside = false, - eventType = event.type; - - event.type = event.data; - - if ( related !== this ) { - - if ( related ) { - inside = jQuery.contains( this, related ); - } - - if ( !inside ) { - - jQuery.event.handle.apply( this, arguments ); - - event.type = eventType; - } - } -}, - -// In case of event delegation, we only need to rename the event.type, -// liveHandler will take care of the rest. -delegate = function( event ) { - event.type = event.data; - jQuery.event.handle.apply( this, arguments ); -}; - -// Create mouseenter and mouseleave events -jQuery.each({ - mouseenter: "mouseover", - mouseleave: "mouseout" -}, function( orig, fix ) { - jQuery.event.special[ orig ] = { - setup: function( data ) { - jQuery.event.add( this, fix, data && data.selector ? delegate : withinElement, orig ); - }, - teardown: function( data ) { - jQuery.event.remove( this, fix, data && data.selector ? delegate : withinElement ); - } - }; -}); - -// submit delegation -if ( !jQuery.support.submitBubbles ) { - - jQuery.event.special.submit = { - setup: function( data, namespaces ) { - if ( !jQuery.nodeName( this, "form" ) ) { - jQuery.event.add(this, "click.specialSubmit", function( e ) { - var elem = e.target, - type = elem.type; - - if ( (type === "submit" || type === "image") && jQuery( elem ).closest("form").length ) { - trigger( "submit", this, arguments ); - } - }); - - jQuery.event.add(this, "keypress.specialSubmit", function( e ) { - var elem = e.target, - type = elem.type; - - if ( (type === "text" || type === "password") && jQuery( elem ).closest("form").length && e.keyCode === 13 ) { - trigger( "submit", this, arguments ); - } - }); - - } else { - return false; - } - }, - - teardown: function( namespaces ) { - jQuery.event.remove( this, ".specialSubmit" ); - } - }; - -} - -// change delegation, happens here so we have bind. -if ( !jQuery.support.changeBubbles ) { - - var changeFilters, - - getVal = function( elem ) { - var type = elem.type, val = elem.value; - - if ( type === "radio" || type === "checkbox" ) { - val = elem.checked; - - } else if ( type === "select-multiple" ) { - val = elem.selectedIndex > -1 ? - jQuery.map( elem.options, function( elem ) { - return elem.selected; - }).join("-") : - ""; - - } else if ( jQuery.nodeName( elem, "select" ) ) { - val = elem.selectedIndex; - } - - return val; - }, - - testChange = function testChange( e ) { - var elem = e.target, data, val; - - if ( !rformElems.test( elem.nodeName ) || elem.readOnly ) { - return; - } - - data = jQuery._data( elem, "_change_data" ); - val = getVal(elem); - - // the current data will be also retrieved by beforeactivate - if ( e.type !== "focusout" || elem.type !== "radio" ) { - jQuery._data( elem, "_change_data", val ); - } - - if ( data === undefined || val === data ) { - return; - } - - if ( data != null || val ) { - e.type = "change"; - e.liveFired = undefined; - jQuery.event.trigger( e, arguments[1], elem ); - } - }; - - jQuery.event.special.change = { - filters: { - focusout: testChange, - - beforedeactivate: testChange, - - click: function( e ) { - var elem = e.target, type = jQuery.nodeName( elem, "input" ) ? elem.type : ""; - - if ( type === "radio" || type === "checkbox" || jQuery.nodeName( elem, "select" ) ) { - testChange.call( this, e ); - } - }, - - // Change has to be called before submit - // Keydown will be called before keypress, which is used in submit-event delegation - keydown: function( e ) { - var elem = e.target, type = jQuery.nodeName( elem, "input" ) ? elem.type : ""; - - if ( (e.keyCode === 13 && !jQuery.nodeName( elem, "textarea" ) ) || - (e.keyCode === 32 && (type === "checkbox" || type === "radio")) || - type === "select-multiple" ) { - testChange.call( this, e ); - } - }, - - // Beforeactivate happens also before the previous element is blurred - // with this event you can't trigger a change event, but you can store - // information - beforeactivate: function( e ) { - var elem = e.target; - jQuery._data( elem, "_change_data", getVal(elem) ); - } - }, - - setup: function( data, namespaces ) { - if ( this.type === "file" ) { - return false; - } - - for ( var type in changeFilters ) { - jQuery.event.add( this, type + ".specialChange", changeFilters[type] ); - } - - return rformElems.test( this.nodeName ); - }, - - teardown: function( namespaces ) { - jQuery.event.remove( this, ".specialChange" ); - - return rformElems.test( this.nodeName ); - } - }; - - changeFilters = jQuery.event.special.change.filters; - - // Handle when the input is .focus()'d - changeFilters.focus = changeFilters.beforeactivate; -} - -function trigger( type, elem, args ) { - // Piggyback on a donor event to simulate a different one. - // Fake originalEvent to avoid donor's stopPropagation, but if the - // simulated event prevents default then we do the same on the donor. - // Don't pass args or remember liveFired; they apply to the donor event. - var event = jQuery.extend( {}, args[ 0 ] ); - event.type = type; - event.originalEvent = {}; - event.liveFired = undefined; - jQuery.event.handle.call( elem, event ); - if ( event.isDefaultPrevented() ) { - args[ 0 ].preventDefault(); - } -} - -// Create "bubbling" focus and blur events -if ( !jQuery.support.focusinBubbles ) { - jQuery.each({ focus: "focusin", blur: "focusout" }, function( orig, fix ) { - - // Attach a single capturing handler while someone wants focusin/focusout - var attaches = 0; - - jQuery.event.special[ fix ] = { - setup: function() { - if ( attaches++ === 0 ) { - document.addEventListener( orig, handler, true ); - } - }, - teardown: function() { - if ( --attaches === 0 ) { - document.removeEventListener( orig, handler, true ); - } - } - }; - - function handler( donor ) { - // Donor event is always a native one; fix it and switch its type. - // Let focusin/out handler cancel the donor focus/blur event. - var e = jQuery.event.fix( donor ); - e.type = fix; - e.originalEvent = {}; - jQuery.event.trigger( e, null, e.target ); - if ( e.isDefaultPrevented() ) { - donor.preventDefault(); - } - } - }); -} - -jQuery.each(["bind", "one"], function( i, name ) { - jQuery.fn[ name ] = function( type, data, fn ) { - var handler; - - // Handle object literals - if ( typeof type === "object" ) { - for ( var key in type ) { - this[ name ](key, data, type[key], fn); - } - return this; - } - - if ( arguments.length === 2 || data === false ) { - fn = data; - data = undefined; - } - - if ( name === "one" ) { - handler = function( event ) { - jQuery( this ).unbind( event, handler ); - return fn.apply( this, arguments ); - }; - handler.guid = fn.guid || jQuery.guid++; - } else { - handler = fn; - } - - if ( type === "unload" && name !== "one" ) { - this.one( type, data, fn ); - - } else { - for ( var i = 0, l = this.length; i < l; i++ ) { - jQuery.event.add( this[i], type, handler, data ); - } - } - - return this; - }; -}); - -jQuery.fn.extend({ - unbind: function( type, fn ) { - // Handle object literals - if ( typeof type === "object" && !type.preventDefault ) { - for ( var key in type ) { - this.unbind(key, type[key]); - } - - } else { - for ( var i = 0, l = this.length; i < l; i++ ) { - jQuery.event.remove( this[i], type, fn ); - } - } - - return this; - }, - - delegate: function( selector, types, data, fn ) { - return this.live( types, data, fn, selector ); - }, - - undelegate: function( selector, types, fn ) { - if ( arguments.length === 0 ) { - return this.unbind( "live" ); - - } else { - return this.die( types, null, fn, selector ); - } - }, - - trigger: function( type, data ) { - return this.each(function() { - jQuery.event.trigger( type, data, this ); - }); - }, - - triggerHandler: function( type, data ) { - if ( this[0] ) { - return jQuery.event.trigger( type, data, this[0], true ); - } - }, - - toggle: function( fn ) { - // Save reference to arguments for access in closure - var args = arguments, - guid = fn.guid || jQuery.guid++, - i = 0, - toggler = function( event ) { - // Figure out which function to execute - var lastToggle = ( jQuery.data( this, "lastToggle" + fn.guid ) || 0 ) % i; - jQuery.data( this, "lastToggle" + fn.guid, lastToggle + 1 ); - - // Make sure that clicks stop - event.preventDefault(); - - // and execute the function - return args[ lastToggle ].apply( this, arguments ) || false; - }; - - // link all the functions, so any of them can unbind this click handler - toggler.guid = guid; - while ( i < args.length ) { - args[ i++ ].guid = guid; - } - - return this.click( toggler ); - }, - - hover: function( fnOver, fnOut ) { - return this.mouseenter( fnOver ).mouseleave( fnOut || fnOver ); - } -}); - -var liveMap = { - focus: "focusin", - blur: "focusout", - mouseenter: "mouseover", - mouseleave: "mouseout" -}; - -jQuery.each(["live", "die"], function( i, name ) { - jQuery.fn[ name ] = function( types, data, fn, origSelector /* Internal Use Only */ ) { - var type, i = 0, match, namespaces, preType, - selector = origSelector || this.selector, - context = origSelector ? this : jQuery( this.context ); - - if ( typeof types === "object" && !types.preventDefault ) { - for ( var key in types ) { - context[ name ]( key, data, types[key], selector ); - } - - return this; - } - - if ( name === "die" && !types && - origSelector && origSelector.charAt(0) === "." ) { - - context.unbind( origSelector ); - - return this; - } - - if ( data === false || jQuery.isFunction( data ) ) { - fn = data || returnFalse; - data = undefined; - } - - types = (types || "").split(" "); - - while ( (type = types[ i++ ]) != null ) { - match = rnamespaces.exec( type ); - namespaces = ""; - - if ( match ) { - namespaces = match[0]; - type = type.replace( rnamespaces, "" ); - } - - if ( type === "hover" ) { - types.push( "mouseenter" + namespaces, "mouseleave" + namespaces ); - continue; - } - - preType = type; - - if ( liveMap[ type ] ) { - types.push( liveMap[ type ] + namespaces ); - type = type + namespaces; - - } else { - type = (liveMap[ type ] || type) + namespaces; - } - - if ( name === "live" ) { - // bind live handler - for ( var j = 0, l = context.length; j < l; j++ ) { - jQuery.event.add( context[j], "live." + liveConvert( type, selector ), - { data: data, selector: selector, handler: fn, origType: type, origHandler: fn, preType: preType } ); - } - - } else { - // unbind live handler - context.unbind( "live." + liveConvert( type, selector ), fn ); - } - } - - return this; - }; -}); - -function liveHandler( event ) { - var stop, maxLevel, related, match, handleObj, elem, j, i, l, data, close, namespace, ret, - elems = [], - selectors = [], - events = jQuery._data( this, "events" ); - - // Make sure we avoid non-left-click bubbling in Firefox (#3861) and disabled elements in IE (#6911) - if ( event.liveFired === this || !events || !events.live || event.target.disabled || event.button && event.type === "click" ) { - return; - } - - if ( event.namespace ) { - namespace = new RegExp("(^|\\.)" + event.namespace.split(".").join("\\.(?:.*\\.)?") + "(\\.|$)"); - } - - event.liveFired = this; - - var live = events.live.slice(0); - - for ( j = 0; j < live.length; j++ ) { - handleObj = live[j]; - - if ( handleObj.origType.replace( rnamespaces, "" ) === event.type ) { - selectors.push( handleObj.selector ); - - } else { - live.splice( j--, 1 ); - } - } - - match = jQuery( event.target ).closest( selectors, event.currentTarget ); - - for ( i = 0, l = match.length; i < l; i++ ) { - close = match[i]; - - for ( j = 0; j < live.length; j++ ) { - handleObj = live[j]; - - if ( close.selector === handleObj.selector && (!namespace || namespace.test( handleObj.namespace )) && !close.elem.disabled ) { - elem = close.elem; - related = null; - - // Those two events require additional checking - if ( handleObj.preType === "mouseenter" || handleObj.preType === "mouseleave" ) { - event.type = handleObj.preType; - related = jQuery( event.relatedTarget ).closest( handleObj.selector )[0]; - - // Make sure not to accidentally match a child element with the same selector - if ( related && jQuery.contains( elem, related ) ) { - related = elem; - } - } - - if ( !related || related !== elem ) { - elems.push({ elem: elem, handleObj: handleObj, level: close.level }); - } - } - } - } - - for ( i = 0, l = elems.length; i < l; i++ ) { - match = elems[i]; - - if ( maxLevel && match.level > maxLevel ) { - break; - } - - event.currentTarget = match.elem; - event.data = match.handleObj.data; - event.handleObj = match.handleObj; - - ret = match.handleObj.origHandler.apply( match.elem, arguments ); - - if ( ret === false || event.isPropagationStopped() ) { - maxLevel = match.level; - - if ( ret === false ) { - stop = false; - } - if ( event.isImmediatePropagationStopped() ) { - break; - } - } - } - - return stop; -} - -function liveConvert( type, selector ) { - return (type && type !== "*" ? type + "." : "") + selector.replace(rperiod, "`").replace(rspaces, "&"); -} - -jQuery.each( ("blur focus focusin focusout load resize scroll unload click dblclick " + - "mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave " + - "change select submit keydown keypress keyup error").split(" "), function( i, name ) { - - // Handle event binding - jQuery.fn[ name ] = function( data, fn ) { - if ( fn == null ) { - fn = data; - data = null; - } - - return arguments.length > 0 ? - this.bind( name, data, fn ) : - this.trigger( name ); - }; - - if ( jQuery.attrFn ) { - jQuery.attrFn[ name ] = true; - } -}); - - - -/*! - * Sizzle CSS Selector Engine - * Copyright 2011, The Dojo Foundation - * Released under the MIT, BSD, and GPL Licenses. - * More information: http://sizzlejs.com/ - */ -(function(){ - -var chunker = /((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^\[\]]*\]|['"][^'"]*['"]|[^\[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g, - done = 0, - toString = Object.prototype.toString, - hasDuplicate = false, - baseHasDuplicate = true, - rBackslash = /\\/g, - rNonWord = /\W/; - -// Here we check if the JavaScript engine is using some sort of -// optimization where it does not always call our comparision -// function. If that is the case, discard the hasDuplicate value. -// Thus far that includes Google Chrome. -[0, 0].sort(function() { - baseHasDuplicate = false; - return 0; -}); - -var Sizzle = function( selector, context, results, seed ) { - results = results || []; - context = context || document; - - var origContext = context; - - if ( context.nodeType !== 1 && context.nodeType !== 9 ) { - return []; - } - - if ( !selector || typeof selector !== "string" ) { - return results; - } - - var m, set, checkSet, extra, ret, cur, pop, i, - prune = true, - contextXML = Sizzle.isXML( context ), - parts = [], - soFar = selector; - - // Reset the position of the chunker regexp (start from head) - do { - chunker.exec( "" ); - m = chunker.exec( soFar ); - - if ( m ) { - soFar = m[3]; - - parts.push( m[1] ); - - if ( m[2] ) { - extra = m[3]; - break; - } - } - } while ( m ); - - if ( parts.length > 1 && origPOS.exec( selector ) ) { - - if ( parts.length === 2 && Expr.relative[ parts[0] ] ) { - set = posProcess( parts[0] + parts[1], context ); - - } else { - set = Expr.relative[ parts[0] ] ? - [ context ] : - Sizzle( parts.shift(), context ); - - while ( parts.length ) { - selector = parts.shift(); - - if ( Expr.relative[ selector ] ) { - selector += parts.shift(); - } - - set = posProcess( selector, set ); - } - } - - } else { - // Take a shortcut and set the context if the root selector is an ID - // (but not if it'll be faster if the inner selector is an ID) - if ( !seed && parts.length > 1 && context.nodeType === 9 && !contextXML && - Expr.match.ID.test(parts[0]) && !Expr.match.ID.test(parts[parts.length - 1]) ) { - - ret = Sizzle.find( parts.shift(), context, contextXML ); - context = ret.expr ? - Sizzle.filter( ret.expr, ret.set )[0] : - ret.set[0]; - } - - if ( context ) { - ret = seed ? - { expr: parts.pop(), set: makeArray(seed) } : - Sizzle.find( parts.pop(), parts.length === 1 && (parts[0] === "~" || parts[0] === "+") && context.parentNode ? context.parentNode : context, contextXML ); - - set = ret.expr ? - Sizzle.filter( ret.expr, ret.set ) : - ret.set; - - if ( parts.length > 0 ) { - checkSet = makeArray( set ); - - } else { - prune = false; - } - - while ( parts.length ) { - cur = parts.pop(); - pop = cur; - - if ( !Expr.relative[ cur ] ) { - cur = ""; - } else { - pop = parts.pop(); - } - - if ( pop == null ) { - pop = context; - } - - Expr.relative[ cur ]( checkSet, pop, contextXML ); - } - - } else { - checkSet = parts = []; - } - } - - if ( !checkSet ) { - checkSet = set; - } - - if ( !checkSet ) { - Sizzle.error( cur || selector ); - } - - if ( toString.call(checkSet) === "[object Array]" ) { - if ( !prune ) { - results.push.apply( results, checkSet ); - - } else if ( context && context.nodeType === 1 ) { - for ( i = 0; checkSet[i] != null; i++ ) { - if ( checkSet[i] && (checkSet[i] === true || checkSet[i].nodeType === 1 && Sizzle.contains(context, checkSet[i])) ) { - results.push( set[i] ); - } - } - - } else { - for ( i = 0; checkSet[i] != null; i++ ) { - if ( checkSet[i] && checkSet[i].nodeType === 1 ) { - results.push( set[i] ); - } - } - } - - } else { - makeArray( checkSet, results ); - } - - if ( extra ) { - Sizzle( extra, origContext, results, seed ); - Sizzle.uniqueSort( results ); - } - - return results; -}; - -Sizzle.uniqueSort = function( results ) { - if ( sortOrder ) { - hasDuplicate = baseHasDuplicate; - results.sort( sortOrder ); - - if ( hasDuplicate ) { - for ( var i = 1; i < results.length; i++ ) { - if ( results[i] === results[ i - 1 ] ) { - results.splice( i--, 1 ); - } - } - } - } - - return results; -}; - -Sizzle.matches = function( expr, set ) { - return Sizzle( expr, null, null, set ); -}; - -Sizzle.matchesSelector = function( node, expr ) { - return Sizzle( expr, null, null, [node] ).length > 0; -}; - -Sizzle.find = function( expr, context, isXML ) { - var set; - - if ( !expr ) { - return []; - } - - for ( var i = 0, l = Expr.order.length; i < l; i++ ) { - var match, - type = Expr.order[i]; - - if ( (match = Expr.leftMatch[ type ].exec( expr )) ) { - var left = match[1]; - match.splice( 1, 1 ); - - if ( left.substr( left.length - 1 ) !== "\\" ) { - match[1] = (match[1] || "").replace( rBackslash, "" ); - set = Expr.find[ type ]( match, context, isXML ); - - if ( set != null ) { - expr = expr.replace( Expr.match[ type ], "" ); - break; - } - } - } - } - - if ( !set ) { - set = typeof context.getElementsByTagName !== "undefined" ? - context.getElementsByTagName( "*" ) : - []; - } - - return { set: set, expr: expr }; -}; - -Sizzle.filter = function( expr, set, inplace, not ) { - var match, anyFound, - old = expr, - result = [], - curLoop = set, - isXMLFilter = set && set[0] && Sizzle.isXML( set[0] ); - - while ( expr && set.length ) { - for ( var type in Expr.filter ) { - if ( (match = Expr.leftMatch[ type ].exec( expr )) != null && match[2] ) { - var found, item, - filter = Expr.filter[ type ], - left = match[1]; - - anyFound = false; - - match.splice(1,1); - - if ( left.substr( left.length - 1 ) === "\\" ) { - continue; - } - - if ( curLoop === result ) { - result = []; - } - - if ( Expr.preFilter[ type ] ) { - match = Expr.preFilter[ type ]( match, curLoop, inplace, result, not, isXMLFilter ); - - if ( !match ) { - anyFound = found = true; - - } else if ( match === true ) { - continue; - } - } - - if ( match ) { - for ( var i = 0; (item = curLoop[i]) != null; i++ ) { - if ( item ) { - found = filter( item, match, i, curLoop ); - var pass = not ^ !!found; - - if ( inplace && found != null ) { - if ( pass ) { - anyFound = true; - - } else { - curLoop[i] = false; - } - - } else if ( pass ) { - result.push( item ); - anyFound = true; - } - } - } - } - - if ( found !== undefined ) { - if ( !inplace ) { - curLoop = result; - } - - expr = expr.replace( Expr.match[ type ], "" ); - - if ( !anyFound ) { - return []; - } - - break; - } - } - } - - // Improper expression - if ( expr === old ) { - if ( anyFound == null ) { - Sizzle.error( expr ); - - } else { - break; - } - } - - old = expr; - } - - return curLoop; -}; - -Sizzle.error = function( msg ) { - throw "Syntax error, unrecognized expression: " + msg; -}; - -var Expr = Sizzle.selectors = { - order: [ "ID", "NAME", "TAG" ], - - match: { - ID: /#((?:[\w\u00c0-\uFFFF\-]|\\.)+)/, - CLASS: /\.((?:[\w\u00c0-\uFFFF\-]|\\.)+)/, - NAME: /\[name=['"]*((?:[\w\u00c0-\uFFFF\-]|\\.)+)['"]*\]/, - ATTR: /\[\s*((?:[\w\u00c0-\uFFFF\-]|\\.)+)\s*(?:(\S?=)\s*(?:(['"])(.*?)\3|(#?(?:[\w\u00c0-\uFFFF\-]|\\.)*)|)|)\s*\]/, - TAG: /^((?:[\w\u00c0-\uFFFF\*\-]|\\.)+)/, - CHILD: /:(only|nth|last|first)-child(?:\(\s*(even|odd|(?:[+\-]?\d+|(?:[+\-]?\d*)?n\s*(?:[+\-]\s*\d+)?))\s*\))?/, - POS: /:(nth|eq|gt|lt|first|last|even|odd)(?:\((\d*)\))?(?=[^\-]|$)/, - PSEUDO: /:((?:[\w\u00c0-\uFFFF\-]|\\.)+)(?:\((['"]?)((?:\([^\)]+\)|[^\(\)]*)+)\2\))?/ - }, - - leftMatch: {}, - - attrMap: { - "class": "className", - "for": "htmlFor" - }, - - attrHandle: { - href: function( elem ) { - return elem.getAttribute( "href" ); - }, - type: function( elem ) { - return elem.getAttribute( "type" ); - } - }, - - relative: { - "+": function(checkSet, part){ - var isPartStr = typeof part === "string", - isTag = isPartStr && !rNonWord.test( part ), - isPartStrNotTag = isPartStr && !isTag; - - if ( isTag ) { - part = part.toLowerCase(); - } - - for ( var i = 0, l = checkSet.length, elem; i < l; i++ ) { - if ( (elem = checkSet[i]) ) { - while ( (elem = elem.previousSibling) && elem.nodeType !== 1 ) {} - - checkSet[i] = isPartStrNotTag || elem && elem.nodeName.toLowerCase() === part ? - elem || false : - elem === part; - } - } - - if ( isPartStrNotTag ) { - Sizzle.filter( part, checkSet, true ); - } - }, - - ">": function( checkSet, part ) { - var elem, - isPartStr = typeof part === "string", - i = 0, - l = checkSet.length; - - if ( isPartStr && !rNonWord.test( part ) ) { - part = part.toLowerCase(); - - for ( ; i < l; i++ ) { - elem = checkSet[i]; - - if ( elem ) { - var parent = elem.parentNode; - checkSet[i] = parent.nodeName.toLowerCase() === part ? parent : false; - } - } - - } else { - for ( ; i < l; i++ ) { - elem = checkSet[i]; - - if ( elem ) { - checkSet[i] = isPartStr ? - elem.parentNode : - elem.parentNode === part; - } - } - - if ( isPartStr ) { - Sizzle.filter( part, checkSet, true ); - } - } - }, - - "": function(checkSet, part, isXML){ - var nodeCheck, - doneName = done++, - checkFn = dirCheck; - - if ( typeof part === "string" && !rNonWord.test( part ) ) { - part = part.toLowerCase(); - nodeCheck = part; - checkFn = dirNodeCheck; - } - - checkFn( "parentNode", part, doneName, checkSet, nodeCheck, isXML ); - }, - - "~": function( checkSet, part, isXML ) { - var nodeCheck, - doneName = done++, - checkFn = dirCheck; - - if ( typeof part === "string" && !rNonWord.test( part ) ) { - part = part.toLowerCase(); - nodeCheck = part; - checkFn = dirNodeCheck; - } - - checkFn( "previousSibling", part, doneName, checkSet, nodeCheck, isXML ); - } - }, - - find: { - ID: function( match, context, isXML ) { - if ( typeof context.getElementById !== "undefined" && !isXML ) { - var m = context.getElementById(match[1]); - // Check parentNode to catch when Blackberry 4.6 returns - // nodes that are no longer in the document #6963 - return m && m.parentNode ? [m] : []; - } - }, - - NAME: function( match, context ) { - if ( typeof context.getElementsByName !== "undefined" ) { - var ret = [], - results = context.getElementsByName( match[1] ); - - for ( var i = 0, l = results.length; i < l; i++ ) { - if ( results[i].getAttribute("name") === match[1] ) { - ret.push( results[i] ); - } - } - - return ret.length === 0 ? null : ret; - } - }, - - TAG: function( match, context ) { - if ( typeof context.getElementsByTagName !== "undefined" ) { - return context.getElementsByTagName( match[1] ); - } - } - }, - preFilter: { - CLASS: function( match, curLoop, inplace, result, not, isXML ) { - match = " " + match[1].replace( rBackslash, "" ) + " "; - - if ( isXML ) { - return match; - } - - for ( var i = 0, elem; (elem = curLoop[i]) != null; i++ ) { - if ( elem ) { - if ( not ^ (elem.className && (" " + elem.className + " ").replace(/[\t\n\r]/g, " ").indexOf(match) >= 0) ) { - if ( !inplace ) { - result.push( elem ); - } - - } else if ( inplace ) { - curLoop[i] = false; - } - } - } - - return false; - }, - - ID: function( match ) { - return match[1].replace( rBackslash, "" ); - }, - - TAG: function( match, curLoop ) { - return match[1].replace( rBackslash, "" ).toLowerCase(); - }, - - CHILD: function( match ) { - if ( match[1] === "nth" ) { - if ( !match[2] ) { - Sizzle.error( match[0] ); - } - - match[2] = match[2].replace(/^\+|\s*/g, ''); - - // parse equations like 'even', 'odd', '5', '2n', '3n+2', '4n-1', '-n+6' - var test = /(-?)(\d*)(?:n([+\-]?\d*))?/.exec( - match[2] === "even" && "2n" || match[2] === "odd" && "2n+1" || - !/\D/.test( match[2] ) && "0n+" + match[2] || match[2]); - - // calculate the numbers (first)n+(last) including if they are negative - match[2] = (test[1] + (test[2] || 1)) - 0; - match[3] = test[3] - 0; - } - else if ( match[2] ) { - Sizzle.error( match[0] ); - } - - // TODO: Move to normal caching system - match[0] = done++; - - return match; - }, - - ATTR: function( match, curLoop, inplace, result, not, isXML ) { - var name = match[1] = match[1].replace( rBackslash, "" ); - - if ( !isXML && Expr.attrMap[name] ) { - match[1] = Expr.attrMap[name]; - } - - // Handle if an un-quoted value was used - match[4] = ( match[4] || match[5] || "" ).replace( rBackslash, "" ); - - if ( match[2] === "~=" ) { - match[4] = " " + match[4] + " "; - } - - return match; - }, - - PSEUDO: function( match, curLoop, inplace, result, not ) { - if ( match[1] === "not" ) { - // If we're dealing with a complex expression, or a simple one - if ( ( chunker.exec(match[3]) || "" ).length > 1 || /^\w/.test(match[3]) ) { - match[3] = Sizzle(match[3], null, null, curLoop); - - } else { - var ret = Sizzle.filter(match[3], curLoop, inplace, true ^ not); - - if ( !inplace ) { - result.push.apply( result, ret ); - } - - return false; - } - - } else if ( Expr.match.POS.test( match[0] ) || Expr.match.CHILD.test( match[0] ) ) { - return true; - } - - return match; - }, - - POS: function( match ) { - match.unshift( true ); - - return match; - } - }, - - filters: { - enabled: function( elem ) { - return elem.disabled === false && elem.type !== "hidden"; - }, - - disabled: function( elem ) { - return elem.disabled === true; - }, - - checked: function( elem ) { - return elem.checked === true; - }, - - selected: function( elem ) { - // Accessing this property makes selected-by-default - // options in Safari work properly - if ( elem.parentNode ) { - elem.parentNode.selectedIndex; - } - - return elem.selected === true; - }, - - parent: function( elem ) { - return !!elem.firstChild; - }, - - empty: function( elem ) { - return !elem.firstChild; - }, - - has: function( elem, i, match ) { - return !!Sizzle( match[3], elem ).length; - }, - - header: function( elem ) { - return (/h\d/i).test( elem.nodeName ); - }, - - text: function( elem ) { - var attr = elem.getAttribute( "type" ), type = elem.type; - // IE6 and 7 will map elem.type to 'text' for new HTML5 types (search, etc) - // use getAttribute instead to test this case - return elem.nodeName.toLowerCase() === "input" && "text" === type && ( attr === type || attr === null ); - }, - - radio: function( elem ) { - return elem.nodeName.toLowerCase() === "input" && "radio" === elem.type; - }, - - checkbox: function( elem ) { - return elem.nodeName.toLowerCase() === "input" && "checkbox" === elem.type; - }, - - file: function( elem ) { - return elem.nodeName.toLowerCase() === "input" && "file" === elem.type; - }, - - password: function( elem ) { - return elem.nodeName.toLowerCase() === "input" && "password" === elem.type; - }, - - submit: function( elem ) { - var name = elem.nodeName.toLowerCase(); - return (name === "input" || name === "button") && "submit" === elem.type; - }, - - image: function( elem ) { - return elem.nodeName.toLowerCase() === "input" && "image" === elem.type; - }, - - reset: function( elem ) { - var name = elem.nodeName.toLowerCase(); - return (name === "input" || name === "button") && "reset" === elem.type; - }, - - button: function( elem ) { - var name = elem.nodeName.toLowerCase(); - return name === "input" && "button" === elem.type || name === "button"; - }, - - input: function( elem ) { - return (/input|select|textarea|button/i).test( elem.nodeName ); - }, - - focus: function( elem ) { - return elem === elem.ownerDocument.activeElement; - } - }, - setFilters: { - first: function( elem, i ) { - return i === 0; - }, - - last: function( elem, i, match, array ) { - return i === array.length - 1; - }, - - even: function( elem, i ) { - return i % 2 === 0; - }, - - odd: function( elem, i ) { - return i % 2 === 1; - }, - - lt: function( elem, i, match ) { - return i < match[3] - 0; - }, - - gt: function( elem, i, match ) { - return i > match[3] - 0; - }, - - nth: function( elem, i, match ) { - return match[3] - 0 === i; - }, - - eq: function( elem, i, match ) { - return match[3] - 0 === i; - } - }, - filter: { - PSEUDO: function( elem, match, i, array ) { - var name = match[1], - filter = Expr.filters[ name ]; - - if ( filter ) { - return filter( elem, i, match, array ); - - } else if ( name === "contains" ) { - return (elem.textContent || elem.innerText || Sizzle.getText([ elem ]) || "").indexOf(match[3]) >= 0; - - } else if ( name === "not" ) { - var not = match[3]; - - for ( var j = 0, l = not.length; j < l; j++ ) { - if ( not[j] === elem ) { - return false; - } - } - - return true; - - } else { - Sizzle.error( name ); - } - }, - - CHILD: function( elem, match ) { - var type = match[1], - node = elem; - - switch ( type ) { - case "only": - case "first": - while ( (node = node.previousSibling) ) { - if ( node.nodeType === 1 ) { - return false; - } - } - - if ( type === "first" ) { - return true; - } - - node = elem; - - case "last": - while ( (node = node.nextSibling) ) { - if ( node.nodeType === 1 ) { - return false; - } - } - - return true; - - case "nth": - var first = match[2], - last = match[3]; - - if ( first === 1 && last === 0 ) { - return true; - } - - var doneName = match[0], - parent = elem.parentNode; - - if ( parent && (parent.sizcache !== doneName || !elem.nodeIndex) ) { - var count = 0; - - for ( node = parent.firstChild; node; node = node.nextSibling ) { - if ( node.nodeType === 1 ) { - node.nodeIndex = ++count; - } - } - - parent.sizcache = doneName; - } - - var diff = elem.nodeIndex - last; - - if ( first === 0 ) { - return diff === 0; - - } else { - return ( diff % first === 0 && diff / first >= 0 ); - } - } - }, - - ID: function( elem, match ) { - return elem.nodeType === 1 && elem.getAttribute("id") === match; - }, - - TAG: function( elem, match ) { - return (match === "*" && elem.nodeType === 1) || elem.nodeName.toLowerCase() === match; - }, - - CLASS: function( elem, match ) { - return (" " + (elem.className || elem.getAttribute("class")) + " ") - .indexOf( match ) > -1; - }, - - ATTR: function( elem, match ) { - var name = match[1], - result = Expr.attrHandle[ name ] ? - Expr.attrHandle[ name ]( elem ) : - elem[ name ] != null ? - elem[ name ] : - elem.getAttribute( name ), - value = result + "", - type = match[2], - check = match[4]; - - return result == null ? - type === "!=" : - type === "=" ? - value === check : - type === "*=" ? - value.indexOf(check) >= 0 : - type === "~=" ? - (" " + value + " ").indexOf(check) >= 0 : - !check ? - value && result !== false : - type === "!=" ? - value !== check : - type === "^=" ? - value.indexOf(check) === 0 : - type === "$=" ? - value.substr(value.length - check.length) === check : - type === "|=" ? - value === check || value.substr(0, check.length + 1) === check + "-" : - false; - }, - - POS: function( elem, match, i, array ) { - var name = match[2], - filter = Expr.setFilters[ name ]; - - if ( filter ) { - return filter( elem, i, match, array ); - } - } - } -}; - -var origPOS = Expr.match.POS, - fescape = function(all, num){ - return "\\" + (num - 0 + 1); - }; - -for ( var type in Expr.match ) { - Expr.match[ type ] = new RegExp( Expr.match[ type ].source + (/(?![^\[]*\])(?![^\(]*\))/.source) ); - Expr.leftMatch[ type ] = new RegExp( /(^(?:.|\r|\n)*?)/.source + Expr.match[ type ].source.replace(/\\(\d+)/g, fescape) ); -} - -var makeArray = function( array, results ) { - array = Array.prototype.slice.call( array, 0 ); - - if ( results ) { - results.push.apply( results, array ); - return results; - } - - return array; -}; - -// Perform a simple check to determine if the browser is capable of -// converting a NodeList to an array using builtin methods. -// Also verifies that the returned array holds DOM nodes -// (which is not the case in the Blackberry browser) -try { - Array.prototype.slice.call( document.documentElement.childNodes, 0 )[0].nodeType; - -// Provide a fallback method if it does not work -} catch( e ) { - makeArray = function( array, results ) { - var i = 0, - ret = results || []; - - if ( toString.call(array) === "[object Array]" ) { - Array.prototype.push.apply( ret, array ); - - } else { - if ( typeof array.length === "number" ) { - for ( var l = array.length; i < l; i++ ) { - ret.push( array[i] ); - } - - } else { - for ( ; array[i]; i++ ) { - ret.push( array[i] ); - } - } - } - - return ret; - }; -} - -var sortOrder, siblingCheck; - -if ( document.documentElement.compareDocumentPosition ) { - sortOrder = function( a, b ) { - if ( a === b ) { - hasDuplicate = true; - return 0; - } - - if ( !a.compareDocumentPosition || !b.compareDocumentPosition ) { - return a.compareDocumentPosition ? -1 : 1; - } - - return a.compareDocumentPosition(b) & 4 ? -1 : 1; - }; - -} else { - sortOrder = function( a, b ) { - // The nodes are identical, we can exit early - if ( a === b ) { - hasDuplicate = true; - return 0; - - // Fallback to using sourceIndex (in IE) if it's available on both nodes - } else if ( a.sourceIndex && b.sourceIndex ) { - return a.sourceIndex - b.sourceIndex; - } - - var al, bl, - ap = [], - bp = [], - aup = a.parentNode, - bup = b.parentNode, - cur = aup; - - // If the nodes are siblings (or identical) we can do a quick check - if ( aup === bup ) { - return siblingCheck( a, b ); - - // If no parents were found then the nodes are disconnected - } else if ( !aup ) { - return -1; - - } else if ( !bup ) { - return 1; - } - - // Otherwise they're somewhere else in the tree so we need - // to build up a full list of the parentNodes for comparison - while ( cur ) { - ap.unshift( cur ); - cur = cur.parentNode; - } - - cur = bup; - - while ( cur ) { - bp.unshift( cur ); - cur = cur.parentNode; - } - - al = ap.length; - bl = bp.length; - - // Start walking down the tree looking for a discrepancy - for ( var i = 0; i < al && i < bl; i++ ) { - if ( ap[i] !== bp[i] ) { - return siblingCheck( ap[i], bp[i] ); - } - } - - // We ended someplace up the tree so do a sibling check - return i === al ? - siblingCheck( a, bp[i], -1 ) : - siblingCheck( ap[i], b, 1 ); - }; - - siblingCheck = function( a, b, ret ) { - if ( a === b ) { - return ret; - } - - var cur = a.nextSibling; - - while ( cur ) { - if ( cur === b ) { - return -1; - } - - cur = cur.nextSibling; - } - - return 1; - }; -} - -// Utility function for retreiving the text value of an array of DOM nodes -Sizzle.getText = function( elems ) { - var ret = "", elem; - - for ( var i = 0; elems[i]; i++ ) { - elem = elems[i]; - - // Get the text from text nodes and CDATA nodes - if ( elem.nodeType === 3 || elem.nodeType === 4 ) { - ret += elem.nodeValue; - - // Traverse everything else, except comment nodes - } else if ( elem.nodeType !== 8 ) { - ret += Sizzle.getText( elem.childNodes ); - } - } - - return ret; -}; - -// Check to see if the browser returns elements by name when -// querying by getElementById (and provide a workaround) -(function(){ - // We're going to inject a fake input element with a specified name - var form = document.createElement("div"), - id = "script" + (new Date()).getTime(), - root = document.documentElement; - - form.innerHTML = ""; - - // Inject it into the root element, check its status, and remove it quickly - root.insertBefore( form, root.firstChild ); - - // The workaround has to do additional checks after a getElementById - // Which slows things down for other browsers (hence the branching) - if ( document.getElementById( id ) ) { - Expr.find.ID = function( match, context, isXML ) { - if ( typeof context.getElementById !== "undefined" && !isXML ) { - var m = context.getElementById(match[1]); - - return m ? - m.id === match[1] || typeof m.getAttributeNode !== "undefined" && m.getAttributeNode("id").nodeValue === match[1] ? - [m] : - undefined : - []; - } - }; - - Expr.filter.ID = function( elem, match ) { - var node = typeof elem.getAttributeNode !== "undefined" && elem.getAttributeNode("id"); - - return elem.nodeType === 1 && node && node.nodeValue === match; - }; - } - - root.removeChild( form ); - - // release memory in IE - root = form = null; -})(); - -(function(){ - // Check to see if the browser returns only elements - // when doing getElementsByTagName("*") - - // Create a fake element - var div = document.createElement("div"); - div.appendChild( document.createComment("") ); - - // Make sure no comments are found - if ( div.getElementsByTagName("*").length > 0 ) { - Expr.find.TAG = function( match, context ) { - var results = context.getElementsByTagName( match[1] ); - - // Filter out possible comments - if ( match[1] === "*" ) { - var tmp = []; - - for ( var i = 0; results[i]; i++ ) { - if ( results[i].nodeType === 1 ) { - tmp.push( results[i] ); - } - } - - results = tmp; - } - - return results; - }; - } - - // Check to see if an attribute returns normalized href attributes - div.innerHTML = ""; - - if ( div.firstChild && typeof div.firstChild.getAttribute !== "undefined" && - div.firstChild.getAttribute("href") !== "#" ) { - - Expr.attrHandle.href = function( elem ) { - return elem.getAttribute( "href", 2 ); - }; - } - - // release memory in IE - div = null; -})(); - -if ( document.querySelectorAll ) { - (function(){ - var oldSizzle = Sizzle, - div = document.createElement("div"), - id = "__sizzle__"; - - div.innerHTML = "

    "; - - // Safari can't handle uppercase or unicode characters when - // in quirks mode. - if ( div.querySelectorAll && div.querySelectorAll(".TEST").length === 0 ) { - return; - } - - Sizzle = function( query, context, extra, seed ) { - context = context || document; - - // Only use querySelectorAll on non-XML documents - // (ID selectors don't work in non-HTML documents) - if ( !seed && !Sizzle.isXML(context) ) { - // See if we find a selector to speed up - var match = /^(\w+$)|^\.([\w\-]+$)|^#([\w\-]+$)/.exec( query ); - - if ( match && (context.nodeType === 1 || context.nodeType === 9) ) { - // Speed-up: Sizzle("TAG") - if ( match[1] ) { - return makeArray( context.getElementsByTagName( query ), extra ); - - // Speed-up: Sizzle(".CLASS") - } else if ( match[2] && Expr.find.CLASS && context.getElementsByClassName ) { - return makeArray( context.getElementsByClassName( match[2] ), extra ); - } - } - - if ( context.nodeType === 9 ) { - // Speed-up: Sizzle("body") - // The body element only exists once, optimize finding it - if ( query === "body" && context.body ) { - return makeArray( [ context.body ], extra ); - - // Speed-up: Sizzle("#ID") - } else if ( match && match[3] ) { - var elem = context.getElementById( match[3] ); - - // Check parentNode to catch when Blackberry 4.6 returns - // nodes that are no longer in the document #6963 - if ( elem && elem.parentNode ) { - // Handle the case where IE and Opera return items - // by name instead of ID - if ( elem.id === match[3] ) { - return makeArray( [ elem ], extra ); - } - - } else { - return makeArray( [], extra ); - } - } - - try { - return makeArray( context.querySelectorAll(query), extra ); - } catch(qsaError) {} - - // qSA works strangely on Element-rooted queries - // We can work around this by specifying an extra ID on the root - // and working up from there (Thanks to Andrew Dupont for the technique) - // IE 8 doesn't work on object elements - } else if ( context.nodeType === 1 && context.nodeName.toLowerCase() !== "object" ) { - var oldContext = context, - old = context.getAttribute( "id" ), - nid = old || id, - hasParent = context.parentNode, - relativeHierarchySelector = /^\s*[+~]/.test( query ); - - if ( !old ) { - context.setAttribute( "id", nid ); - } else { - nid = nid.replace( /'/g, "\\$&" ); - } - if ( relativeHierarchySelector && hasParent ) { - context = context.parentNode; - } - - try { - if ( !relativeHierarchySelector || hasParent ) { - return makeArray( context.querySelectorAll( "[id='" + nid + "'] " + query ), extra ); - } - - } catch(pseudoError) { - } finally { - if ( !old ) { - oldContext.removeAttribute( "id" ); - } - } - } - } - - return oldSizzle(query, context, extra, seed); - }; - - for ( var prop in oldSizzle ) { - Sizzle[ prop ] = oldSizzle[ prop ]; - } - - // release memory in IE - div = null; - })(); -} - -(function(){ - var html = document.documentElement, - matches = html.matchesSelector || html.mozMatchesSelector || html.webkitMatchesSelector || html.msMatchesSelector; - - if ( matches ) { - // Check to see if it's possible to do matchesSelector - // on a disconnected node (IE 9 fails this) - var disconnectedMatch = !matches.call( document.createElement( "div" ), "div" ), - pseudoWorks = false; - - try { - // This should fail with an exception - // Gecko does not error, returns false instead - matches.call( document.documentElement, "[test!='']:sizzle" ); - - } catch( pseudoError ) { - pseudoWorks = true; - } - - Sizzle.matchesSelector = function( node, expr ) { - // Make sure that attribute selectors are quoted - expr = expr.replace(/\=\s*([^'"\]]*)\s*\]/g, "='$1']"); - - if ( !Sizzle.isXML( node ) ) { - try { - if ( pseudoWorks || !Expr.match.PSEUDO.test( expr ) && !/!=/.test( expr ) ) { - var ret = matches.call( node, expr ); - - // IE 9's matchesSelector returns false on disconnected nodes - if ( ret || !disconnectedMatch || - // As well, disconnected nodes are said to be in a document - // fragment in IE 9, so check for that - node.document && node.document.nodeType !== 11 ) { - return ret; - } - } - } catch(e) {} - } - - return Sizzle(expr, null, null, [node]).length > 0; - }; - } -})(); - -(function(){ - var div = document.createElement("div"); - - div.innerHTML = "
    "; - - // Opera can't find a second classname (in 9.6) - // Also, make sure that getElementsByClassName actually exists - if ( !div.getElementsByClassName || div.getElementsByClassName("e").length === 0 ) { - return; - } - - // Safari caches class attributes, doesn't catch changes (in 3.2) - div.lastChild.className = "e"; - - if ( div.getElementsByClassName("e").length === 1 ) { - return; - } - - Expr.order.splice(1, 0, "CLASS"); - Expr.find.CLASS = function( match, context, isXML ) { - if ( typeof context.getElementsByClassName !== "undefined" && !isXML ) { - return context.getElementsByClassName(match[1]); - } - }; - - // release memory in IE - div = null; -})(); - -function dirNodeCheck( dir, cur, doneName, checkSet, nodeCheck, isXML ) { - for ( var i = 0, l = checkSet.length; i < l; i++ ) { - var elem = checkSet[i]; - - if ( elem ) { - var match = false; - - elem = elem[dir]; - - while ( elem ) { - if ( elem.sizcache === doneName ) { - match = checkSet[elem.sizset]; - break; - } - - if ( elem.nodeType === 1 && !isXML ){ - elem.sizcache = doneName; - elem.sizset = i; - } - - if ( elem.nodeName.toLowerCase() === cur ) { - match = elem; - break; - } - - elem = elem[dir]; - } - - checkSet[i] = match; - } - } -} - -function dirCheck( dir, cur, doneName, checkSet, nodeCheck, isXML ) { - for ( var i = 0, l = checkSet.length; i < l; i++ ) { - var elem = checkSet[i]; - - if ( elem ) { - var match = false; - - elem = elem[dir]; - - while ( elem ) { - if ( elem.sizcache === doneName ) { - match = checkSet[elem.sizset]; - break; - } - - if ( elem.nodeType === 1 ) { - if ( !isXML ) { - elem.sizcache = doneName; - elem.sizset = i; - } - - if ( typeof cur !== "string" ) { - if ( elem === cur ) { - match = true; - break; - } - - } else if ( Sizzle.filter( cur, [elem] ).length > 0 ) { - match = elem; - break; - } - } - - elem = elem[dir]; - } - - checkSet[i] = match; - } - } -} - -if ( document.documentElement.contains ) { - Sizzle.contains = function( a, b ) { - return a !== b && (a.contains ? a.contains(b) : true); - }; - -} else if ( document.documentElement.compareDocumentPosition ) { - Sizzle.contains = function( a, b ) { - return !!(a.compareDocumentPosition(b) & 16); - }; - -} else { - Sizzle.contains = function() { - return false; - }; -} - -Sizzle.isXML = function( elem ) { - // documentElement is verified for cases where it doesn't yet exist - // (such as loading iframes in IE - #4833) - var documentElement = (elem ? elem.ownerDocument || elem : 0).documentElement; - - return documentElement ? documentElement.nodeName !== "HTML" : false; -}; - -var posProcess = function( selector, context ) { - var match, - tmpSet = [], - later = "", - root = context.nodeType ? [context] : context; - - // Position selectors must be done after the filter - // And so must :not(positional) so we move all PSEUDOs to the end - while ( (match = Expr.match.PSEUDO.exec( selector )) ) { - later += match[0]; - selector = selector.replace( Expr.match.PSEUDO, "" ); - } - - selector = Expr.relative[selector] ? selector + "*" : selector; - - for ( var i = 0, l = root.length; i < l; i++ ) { - Sizzle( selector, root[i], tmpSet ); - } - - return Sizzle.filter( later, tmpSet ); -}; - -// EXPOSE -jQuery.find = Sizzle; -jQuery.expr = Sizzle.selectors; -jQuery.expr[":"] = jQuery.expr.filters; -jQuery.unique = Sizzle.uniqueSort; -jQuery.text = Sizzle.getText; -jQuery.isXMLDoc = Sizzle.isXML; -jQuery.contains = Sizzle.contains; - - -})(); - - -var runtil = /Until$/, - rparentsprev = /^(?:parents|prevUntil|prevAll)/, - // Note: This RegExp should be improved, or likely pulled from Sizzle - rmultiselector = /,/, - isSimple = /^.[^:#\[\.,]*$/, - slice = Array.prototype.slice, - POS = jQuery.expr.match.POS, - // methods guaranteed to produce a unique set when starting from a unique set - guaranteedUnique = { - children: true, - contents: true, - next: true, - prev: true - }; - -jQuery.fn.extend({ - find: function( selector ) { - var self = this, - i, l; - - if ( typeof selector !== "string" ) { - return jQuery( selector ).filter(function() { - for ( i = 0, l = self.length; i < l; i++ ) { - if ( jQuery.contains( self[ i ], this ) ) { - return true; - } - } - }); - } - - var ret = this.pushStack( "", "find", selector ), - length, n, r; - - for ( i = 0, l = this.length; i < l; i++ ) { - length = ret.length; - jQuery.find( selector, this[i], ret ); - - if ( i > 0 ) { - // Make sure that the results are unique - for ( n = length; n < ret.length; n++ ) { - for ( r = 0; r < length; r++ ) { - if ( ret[r] === ret[n] ) { - ret.splice(n--, 1); - break; - } - } - } - } - } - - return ret; - }, - - has: function( target ) { - var targets = jQuery( target ); - return this.filter(function() { - for ( var i = 0, l = targets.length; i < l; i++ ) { - if ( jQuery.contains( this, targets[i] ) ) { - return true; - } - } - }); - }, - - not: function( selector ) { - return this.pushStack( winnow(this, selector, false), "not", selector); - }, - - filter: function( selector ) { - return this.pushStack( winnow(this, selector, true), "filter", selector ); - }, - - is: function( selector ) { - return !!selector && ( typeof selector === "string" ? - jQuery.filter( selector, this ).length > 0 : - this.filter( selector ).length > 0 ); - }, - - closest: function( selectors, context ) { - var ret = [], i, l, cur = this[0]; - - // Array - if ( jQuery.isArray( selectors ) ) { - var match, selector, - matches = {}, - level = 1; - - if ( cur && selectors.length ) { - for ( i = 0, l = selectors.length; i < l; i++ ) { - selector = selectors[i]; - - if ( !matches[ selector ] ) { - matches[ selector ] = POS.test( selector ) ? - jQuery( selector, context || this.context ) : - selector; - } - } - - while ( cur && cur.ownerDocument && cur !== context ) { - for ( selector in matches ) { - match = matches[ selector ]; - - if ( match.jquery ? match.index( cur ) > -1 : jQuery( cur ).is( match ) ) { - ret.push({ selector: selector, elem: cur, level: level }); - } - } - - cur = cur.parentNode; - level++; - } - } - - return ret; - } - - // String - var pos = POS.test( selectors ) || typeof selectors !== "string" ? - jQuery( selectors, context || this.context ) : - 0; - - for ( i = 0, l = this.length; i < l; i++ ) { - cur = this[i]; - - while ( cur ) { - if ( pos ? pos.index(cur) > -1 : jQuery.find.matchesSelector(cur, selectors) ) { - ret.push( cur ); - break; - - } else { - cur = cur.parentNode; - if ( !cur || !cur.ownerDocument || cur === context || cur.nodeType === 11 ) { - break; - } - } - } - } - - ret = ret.length > 1 ? jQuery.unique( ret ) : ret; - - return this.pushStack( ret, "closest", selectors ); - }, - - // Determine the position of an element within - // the matched set of elements - index: function( elem ) { - if ( !elem || typeof elem === "string" ) { - return jQuery.inArray( this[0], - // If it receives a string, the selector is used - // If it receives nothing, the siblings are used - elem ? jQuery( elem ) : this.parent().children() ); - } - // Locate the position of the desired element - return jQuery.inArray( - // If it receives a jQuery object, the first element is used - elem.jquery ? elem[0] : elem, this ); - }, - - add: function( selector, context ) { - var set = typeof selector === "string" ? - jQuery( selector, context ) : - jQuery.makeArray( selector && selector.nodeType ? [ selector ] : selector ), - all = jQuery.merge( this.get(), set ); - - return this.pushStack( isDisconnected( set[0] ) || isDisconnected( all[0] ) ? - all : - jQuery.unique( all ) ); - }, - - andSelf: function() { - return this.add( this.prevObject ); - } -}); - -// A painfully simple check to see if an element is disconnected -// from a document (should be improved, where feasible). -function isDisconnected( node ) { - return !node || !node.parentNode || node.parentNode.nodeType === 11; -} - -jQuery.each({ - parent: function( elem ) { - var parent = elem.parentNode; - return parent && parent.nodeType !== 11 ? parent : null; - }, - parents: function( elem ) { - return jQuery.dir( elem, "parentNode" ); - }, - parentsUntil: function( elem, i, until ) { - return jQuery.dir( elem, "parentNode", until ); - }, - next: function( elem ) { - return jQuery.nth( elem, 2, "nextSibling" ); - }, - prev: function( elem ) { - return jQuery.nth( elem, 2, "previousSibling" ); - }, - nextAll: function( elem ) { - return jQuery.dir( elem, "nextSibling" ); - }, - prevAll: function( elem ) { - return jQuery.dir( elem, "previousSibling" ); - }, - nextUntil: function( elem, i, until ) { - return jQuery.dir( elem, "nextSibling", until ); - }, - prevUntil: function( elem, i, until ) { - return jQuery.dir( elem, "previousSibling", until ); - }, - siblings: function( elem ) { - return jQuery.sibling( elem.parentNode.firstChild, elem ); - }, - children: function( elem ) { - return jQuery.sibling( elem.firstChild ); - }, - contents: function( elem ) { - return jQuery.nodeName( elem, "iframe" ) ? - elem.contentDocument || elem.contentWindow.document : - jQuery.makeArray( elem.childNodes ); - } -}, function( name, fn ) { - jQuery.fn[ name ] = function( until, selector ) { - var ret = jQuery.map( this, fn, until ), - // The variable 'args' was introduced in - // https://github.com/jquery/jquery/commit/52a0238 - // to work around a bug in Chrome 10 (Dev) and should be removed when the bug is fixed. - // http://code.google.com/p/v8/issues/detail?id=1050 - args = slice.call(arguments); - - if ( !runtil.test( name ) ) { - selector = until; - } - - if ( selector && typeof selector === "string" ) { - ret = jQuery.filter( selector, ret ); - } - - ret = this.length > 1 && !guaranteedUnique[ name ] ? jQuery.unique( ret ) : ret; - - if ( (this.length > 1 || rmultiselector.test( selector )) && rparentsprev.test( name ) ) { - ret = ret.reverse(); - } - - return this.pushStack( ret, name, args.join(",") ); - }; -}); - -jQuery.extend({ - filter: function( expr, elems, not ) { - if ( not ) { - expr = ":not(" + expr + ")"; - } - - return elems.length === 1 ? - jQuery.find.matchesSelector(elems[0], expr) ? [ elems[0] ] : [] : - jQuery.find.matches(expr, elems); - }, - - dir: function( elem, dir, until ) { - var matched = [], - cur = elem[ dir ]; - - while ( cur && cur.nodeType !== 9 && (until === undefined || cur.nodeType !== 1 || !jQuery( cur ).is( until )) ) { - if ( cur.nodeType === 1 ) { - matched.push( cur ); - } - cur = cur[dir]; - } - return matched; - }, - - nth: function( cur, result, dir, elem ) { - result = result || 1; - var num = 0; - - for ( ; cur; cur = cur[dir] ) { - if ( cur.nodeType === 1 && ++num === result ) { - break; - } - } - - return cur; - }, - - sibling: function( n, elem ) { - var r = []; - - for ( ; n; n = n.nextSibling ) { - if ( n.nodeType === 1 && n !== elem ) { - r.push( n ); - } - } - - return r; - } -}); - -// Implement the identical functionality for filter and not -function winnow( elements, qualifier, keep ) { - - // Can't pass null or undefined to indexOf in Firefox 4 - // Set to 0 to skip string check - qualifier = qualifier || 0; - - if ( jQuery.isFunction( qualifier ) ) { - return jQuery.grep(elements, function( elem, i ) { - var retVal = !!qualifier.call( elem, i, elem ); - return retVal === keep; - }); - - } else if ( qualifier.nodeType ) { - return jQuery.grep(elements, function( elem, i ) { - return (elem === qualifier) === keep; - }); - - } else if ( typeof qualifier === "string" ) { - var filtered = jQuery.grep(elements, function( elem ) { - return elem.nodeType === 1; - }); - - if ( isSimple.test( qualifier ) ) { - return jQuery.filter(qualifier, filtered, !keep); - } else { - qualifier = jQuery.filter( qualifier, filtered ); - } - } - - return jQuery.grep(elements, function( elem, i ) { - return (jQuery.inArray( elem, qualifier ) >= 0) === keep; - }); -} - - - - -var rinlinejQuery = / jQuery\d+="(?:\d+|null)"/g, - rleadingWhitespace = /^\s+/, - rxhtmlTag = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig, - rtagName = /<([\w:]+)/, - rtbody = /", "" ], - legend: [ 1, "
    ", "
    " ], - thead: [ 1, "", "
    " ], - tr: [ 2, "", "
    " ], - td: [ 3, "", "
    " ], - col: [ 2, "", "
    " ], - area: [ 1, "", "" ], - _default: [ 0, "", "" ] - }; - -wrapMap.optgroup = wrapMap.option; -wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead; -wrapMap.th = wrapMap.td; - -// IE can't serialize and - - - - - - diff --git a/node_modules/express/node_modules/debug/example/wildcards.js b/node_modules/express/node_modules/debug/example/wildcards.js deleted file mode 100644 index 1fdac20..0000000 --- a/node_modules/express/node_modules/debug/example/wildcards.js +++ /dev/null @@ -1,10 +0,0 @@ - -var debug = { - foo: require('../')('test:foo'), - bar: require('../')('test:bar'), - baz: require('../')('test:baz') -}; - -debug.foo('foo') -debug.bar('bar') -debug.baz('baz') \ No newline at end of file diff --git a/node_modules/express/node_modules/debug/example/worker.js b/node_modules/express/node_modules/debug/example/worker.js deleted file mode 100644 index 7f6d288..0000000 --- a/node_modules/express/node_modules/debug/example/worker.js +++ /dev/null @@ -1,22 +0,0 @@ - -// DEBUG=* node example/worker -// DEBUG=worker:* node example/worker -// DEBUG=worker:a node example/worker -// DEBUG=worker:b node example/worker - -var a = require('../')('worker:a') - , b = require('../')('worker:b'); - -function work() { - a('doing lots of uninteresting work'); - setTimeout(work, Math.random() * 1000); -} - -work(); - -function workb() { - b('doing some work'); - setTimeout(workb, Math.random() * 2000); -} - -workb(); \ No newline at end of file diff --git a/node_modules/express/node_modules/debug/index.js b/node_modules/express/node_modules/debug/index.js deleted file mode 100644 index e02c13b..0000000 --- a/node_modules/express/node_modules/debug/index.js +++ /dev/null @@ -1,5 +0,0 @@ -if ('undefined' == typeof window) { - module.exports = require('./lib/debug'); -} else { - module.exports = require('./debug'); -} diff --git a/node_modules/express/node_modules/debug/lib/debug.js b/node_modules/express/node_modules/debug/lib/debug.js deleted file mode 100644 index 0b07aa1..0000000 --- a/node_modules/express/node_modules/debug/lib/debug.js +++ /dev/null @@ -1,134 +0,0 @@ -/** - * Module dependencies. - */ - -var tty = require('tty'); - -/** - * Expose `debug()` as the module. - */ - -module.exports = debug; - -/** - * Enabled debuggers. - */ - -var names = [] - , skips = []; - -(process.env.DEBUG || '') - .split(/[\s,]+/) - .forEach(function(name){ - name = name.replace('*', '.*?'); - if (name[0] === '-') { - skips.push(new RegExp('^' + name.substr(1) + '$')); - } else { - names.push(new RegExp('^' + name + '$')); - } - }); - -/** - * Colors. - */ - -var colors = [6, 2, 3, 4, 5, 1]; - -/** - * Previous debug() call. - */ - -var prev = {}; - -/** - * Previously assigned color. - */ - -var prevColor = 0; - -/** - * Is stdout a TTY? Colored output is disabled when `true`. - */ - -var isatty = tty.isatty(2); - -/** - * Select a color. - * - * @return {Number} - * @api private - */ - -function color() { - return colors[prevColor++ % colors.length]; -} - -/** - * Humanize the given `ms`. - * - * @param {Number} m - * @return {String} - * @api private - */ - -function humanize(ms) { - var sec = 1000 - , min = 60 * 1000 - , hour = 60 * min; - - if (ms >= hour) return (ms / hour).toFixed(1) + 'h'; - if (ms >= min) return (ms / min).toFixed(1) + 'm'; - if (ms >= sec) return (ms / sec | 0) + 's'; - return ms + 'ms'; -} - -/** - * Create a debugger with the given `name`. - * - * @param {String} name - * @return {Type} - * @api public - */ - -function debug(name) { - function disabled(){} - disabled.enabled = false; - - var match = skips.some(function(re){ - return re.test(name); - }); - - if (match) return disabled; - - match = names.some(function(re){ - return re.test(name); - }); - - if (!match) return disabled; - var c = color(); - - function colored(fmt) { - var curr = new Date; - var ms = curr - (prev[name] || curr); - prev[name] = curr; - - fmt = ' \u001b[9' + c + 'm' + name + ' ' - + '\u001b[3' + c + 'm\u001b[90m' - + fmt + '\u001b[3' + c + 'm' - + ' +' + humanize(ms) + '\u001b[0m'; - - console.error.apply(this, arguments); - } - - function plain(fmt) { - fmt = new Date().toUTCString() - + ' ' + name + ' ' + fmt; - console.error.apply(this, arguments); - } - - colored.enabled = plain.enabled = true; - - return isatty || process.env.DEBUG_COLORS - ? colored - : plain; -} diff --git a/node_modules/express/node_modules/debug/package.json b/node_modules/express/node_modules/debug/package.json deleted file mode 100644 index 112d22b..0000000 --- a/node_modules/express/node_modules/debug/package.json +++ /dev/null @@ -1,40 +0,0 @@ -{ - "name": "debug", - "version": "0.7.2", - "repository": { - "type": "git", - "url": "git://github.com/visionmedia/debug.git" - }, - "description": "small debugging utility", - "keywords": [ - "debug", - "log", - "debugger" - ], - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca" - }, - "dependencies": {}, - "devDependencies": { - "mocha": "*" - }, - "main": "lib/debug.js", - "browserify": "debug.js", - "engines": { - "node": "*" - }, - "component": { - "scripts": { - "debug/index.js": "index.js", - "debug/debug.js": "debug.js" - } - }, - "readme": "\n# debug\n\n tiny node.js debugging utility modelled after node core's debugging technique.\n\n## Installation\n\n```\n$ npm install debug\n```\n\n## Usage\n\n With `debug` you simply invoke the exported function to generate your debug function, passing it a name which will determine if a noop function is returned, or a decorated `console.error`, so all of the `console` format string goodies you're used to work fine. A unique color is selected per-function for visibility.\n \nExample _app.js_:\n\n```js\nvar debug = require('debug')('http')\n , http = require('http')\n , name = 'My App';\n\n// fake app\n\ndebug('booting %s', name);\n\nhttp.createServer(function(req, res){\n debug(req.method + ' ' + req.url);\n res.end('hello\\n');\n}).listen(3000, function(){\n debug('listening');\n});\n\n// fake worker of some kind\n\nrequire('./worker');\n```\n\nExample _worker.js_:\n\n```js\nvar debug = require('debug')('worker');\n\nsetInterval(function(){\n debug('doing some work');\n}, 1000);\n```\n\n The __DEBUG__ environment variable is then used to enable these based on space or comma-delimited names. Here are some examples:\n\n ![debug http and worker](http://f.cl.ly/items/18471z1H402O24072r1J/Screenshot.png)\n\n ![debug worker](http://f.cl.ly/items/1X413v1a3M0d3C2c1E0i/Screenshot.png)\n\n## Millisecond diff\n\n When actively developing an application it can be useful to see when the time spent between one `debug()` call and the next. Suppose for example you invoke `debug()` before requesting a resource, and after as well, the \"+NNNms\" will show you how much time was spent between calls.\n\n ![](http://f.cl.ly/items/2i3h1d3t121M2Z1A3Q0N/Screenshot.png)\n\n When stdout is not a TTY, `Date#toUTCString()` is used, making it more useful for logging the debug information as shown below:\n \n ![](http://f.cl.ly/items/112H3i0e0o0P0a2Q2r11/Screenshot.png)\n\n## Conventions\n\n If you're using this in one or more of your libraries, you _should_ use the name of your library so that developers may toggle debugging as desired without guessing names. If you have more than one debuggers you _should_ prefix them with your library name and use \":\" to separate features. For example \"bodyParser\" from Connect would then be \"connect:bodyParser\". \n\n## Wildcards\n\n The \"*\" character may be used as a wildcard. Suppose for example your library has debuggers named \"connect:bodyParser\", \"connect:compress\", \"connect:session\", instead of listing all three with `DEBUG=connect:bodyParser,connect.compress,connect:session`, you may simply do `DEBUG=connect:*`, or to run everything using this module simply use `DEBUG=*`.\n\n You can also exclude specific debuggers by prefixing them with a \"-\" character. For example, `DEBUG=* -connect:*` would include all debuggers except those starting with \"connect:\".\n\n## Browser support\n\n Debug works in the browser as well, currently persisted by `localStorage`. For example if you have `worker:a` and `worker:b` as shown below, and wish to debug both type `debug.enable('worker:*')` in the console and refresh the page, this will remain until you disable with `debug.disable()`. \n\n```js\na = debug('worker:a');\nb = debug('worker:b');\n\nsetInterval(function(){\n a('doing some work');\n}, 1000);\n\nsetInterval(function(){\n a('doing some work');\n}, 1200);\n```\n\n## License \n\n(The MIT License)\n\nCopyright (c) 2011 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.", - "readmeFilename": "Readme.md", - "bugs": { - "url": "https://github.com/visionmedia/debug/issues" - }, - "_id": "debug@0.7.2", - "_from": "debug@*" -} diff --git a/node_modules/express/node_modules/fresh/.npmignore b/node_modules/express/node_modules/fresh/.npmignore deleted file mode 100644 index 9daeafb..0000000 --- a/node_modules/express/node_modules/fresh/.npmignore +++ /dev/null @@ -1 +0,0 @@ -test diff --git a/node_modules/express/node_modules/fresh/Makefile b/node_modules/express/node_modules/fresh/Makefile deleted file mode 100644 index 8e8640f..0000000 --- a/node_modules/express/node_modules/fresh/Makefile +++ /dev/null @@ -1,7 +0,0 @@ - -test: - @./node_modules/.bin/mocha \ - --reporter spec \ - --require should - -.PHONY: test \ No newline at end of file diff --git a/node_modules/express/node_modules/fresh/Readme.md b/node_modules/express/node_modules/fresh/Readme.md deleted file mode 100644 index 273130d..0000000 --- a/node_modules/express/node_modules/fresh/Readme.md +++ /dev/null @@ -1,32 +0,0 @@ - -# node-fresh - - HTTP response freshness testing - -## fresh(req, res) - - Check freshness of `req` and `res` headers. - - When the cache is "fresh" __true__ is returned, - otherwise __false__ is returned to indicate that - the cache is now stale. - -## Example: - -```js -var req = { 'if-none-match': 'tobi' }; -var res = { 'etag': 'luna' }; -fresh(req, res); -// => false - -var req = { 'if-none-match': 'tobi' }; -var res = { 'etag': 'tobi' }; -fresh(req, res); -// => true -``` - -## Installation - -``` -$ npm install fresh -``` \ No newline at end of file diff --git a/node_modules/express/node_modules/fresh/index.js b/node_modules/express/node_modules/fresh/index.js deleted file mode 100644 index b2f4d41..0000000 --- a/node_modules/express/node_modules/fresh/index.js +++ /dev/null @@ -1,49 +0,0 @@ - -/** - * Expose `fresh()`. - */ - -module.exports = fresh; - -/** - * Check freshness of `req` and `res` headers. - * - * When the cache is "fresh" __true__ is returned, - * otherwise __false__ is returned to indicate that - * the cache is now stale. - * - * @param {Object} req - * @param {Object} res - * @return {Boolean} - * @api public - */ - -function fresh(req, res) { - // defaults - var etagMatches = true; - var notModified = true; - - // fields - var modifiedSince = req['if-modified-since']; - var noneMatch = req['if-none-match']; - var lastModified = res['last-modified']; - var etag = res['etag']; - - // unconditional request - if (!modifiedSince && !noneMatch) return false; - - // parse if-none-match - if (noneMatch) noneMatch = noneMatch.split(/ *, */); - - // if-none-match - if (noneMatch) etagMatches = ~noneMatch.indexOf(etag) || '*' == noneMatch[0]; - - // if-modified-since - if (modifiedSince) { - modifiedSince = new Date(modifiedSince); - lastModified = new Date(lastModified); - notModified = lastModified <= modifiedSince; - } - - return !! (etagMatches && notModified); -} \ No newline at end of file diff --git a/node_modules/express/node_modules/fresh/package.json b/node_modules/express/node_modules/fresh/package.json deleted file mode 100644 index d9fddb2..0000000 --- a/node_modules/express/node_modules/fresh/package.json +++ /dev/null @@ -1,20 +0,0 @@ -{ - "name": "fresh", - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca", - "url": "http://tjholowaychuk.com" - }, - "description": "HTTP response freshness testing", - "version": "0.1.0", - "main": "index.js", - "dependencies": {}, - "devDependencies": { - "mocha": "*", - "should": "*" - }, - "readme": "\n# node-fresh\n\n HTTP response freshness testing\n\n## fresh(req, res)\n\n Check freshness of `req` and `res` headers.\n\n When the cache is \"fresh\" __true__ is returned,\n otherwise __false__ is returned to indicate that\n the cache is now stale.\n\n## Example:\n\n```js\nvar req = { 'if-none-match': 'tobi' };\nvar res = { 'etag': 'luna' };\nfresh(req, res);\n// => false\n\nvar req = { 'if-none-match': 'tobi' };\nvar res = { 'etag': 'tobi' };\nfresh(req, res);\n// => true\n```\n\n## Installation\n\n```\n$ npm install fresh\n```", - "readmeFilename": "Readme.md", - "_id": "fresh@0.1.0", - "_from": "fresh@0.1.0" -} diff --git a/node_modules/express/node_modules/methods/index.js b/node_modules/express/node_modules/methods/index.js deleted file mode 100644 index 297d022..0000000 --- a/node_modules/express/node_modules/methods/index.js +++ /dev/null @@ -1,26 +0,0 @@ - -module.exports = [ - 'get' - , 'post' - , 'put' - , 'head' - , 'delete' - , 'options' - , 'trace' - , 'copy' - , 'lock' - , 'mkcol' - , 'move' - , 'propfind' - , 'proppatch' - , 'unlock' - , 'report' - , 'mkactivity' - , 'checkout' - , 'merge' - , 'm-search' - , 'notify' - , 'subscribe' - , 'unsubscribe' - , 'patch' -]; \ No newline at end of file diff --git a/node_modules/express/node_modules/methods/package.json b/node_modules/express/node_modules/methods/package.json deleted file mode 100644 index f867a8b..0000000 --- a/node_modules/express/node_modules/methods/package.json +++ /dev/null @@ -1,20 +0,0 @@ -{ - "name": "methods", - "version": "0.0.1", - "description": "HTTP methods that node supports", - "main": "index.js", - "scripts": { - "test": "echo \"Error: no test specified\" && exit 1" - }, - "keywords": [ - "http", - "methods" - ], - "author": { - "name": "TJ Holowaychuk" - }, - "license": "MIT", - "readme": "ERROR: No README data found!", - "_id": "methods@0.0.1", - "_from": "methods@0.0.1" -} diff --git a/node_modules/express/node_modules/mkdirp/.npmignore b/node_modules/express/node_modules/mkdirp/.npmignore deleted file mode 100644 index 9303c34..0000000 --- a/node_modules/express/node_modules/mkdirp/.npmignore +++ /dev/null @@ -1,2 +0,0 @@ -node_modules/ -npm-debug.log \ No newline at end of file diff --git a/node_modules/express/node_modules/mkdirp/.travis.yml b/node_modules/express/node_modules/mkdirp/.travis.yml deleted file mode 100644 index 84fd7ca..0000000 --- a/node_modules/express/node_modules/mkdirp/.travis.yml +++ /dev/null @@ -1,5 +0,0 @@ -language: node_js -node_js: - - 0.6 - - 0.8 - - 0.9 diff --git a/node_modules/express/node_modules/mkdirp/LICENSE b/node_modules/express/node_modules/mkdirp/LICENSE deleted file mode 100644 index 432d1ae..0000000 --- a/node_modules/express/node_modules/mkdirp/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -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/node_modules/express/node_modules/mkdirp/examples/pow.js b/node_modules/express/node_modules/mkdirp/examples/pow.js deleted file mode 100644 index e692421..0000000 --- a/node_modules/express/node_modules/mkdirp/examples/pow.js +++ /dev/null @@ -1,6 +0,0 @@ -var mkdirp = require('mkdirp'); - -mkdirp('/tmp/foo/bar/baz', function (err) { - if (err) console.error(err) - else console.log('pow!') -}); diff --git a/node_modules/express/node_modules/mkdirp/index.js b/node_modules/express/node_modules/mkdirp/index.js deleted file mode 100644 index fda6de8..0000000 --- a/node_modules/express/node_modules/mkdirp/index.js +++ /dev/null @@ -1,82 +0,0 @@ -var path = require('path'); -var fs = require('fs'); - -module.exports = mkdirP.mkdirp = mkdirP.mkdirP = mkdirP; - -function mkdirP (p, mode, f, made) { - if (typeof mode === 'function' || mode === undefined) { - f = mode; - mode = 0777 & (~process.umask()); - } - if (!made) made = null; - - var cb = f || function () {}; - if (typeof mode === 'string') mode = parseInt(mode, 8); - p = path.resolve(p); - - fs.mkdir(p, mode, function (er) { - if (!er) { - made = made || p; - return cb(null, made); - } - switch (er.code) { - case 'ENOENT': - mkdirP(path.dirname(p), mode, function (er, made) { - if (er) cb(er, made); - else mkdirP(p, mode, cb, made); - }); - break; - - // In the case of any other error, just see if there's a dir - // there already. If so, then hooray! If not, then something - // is borked. - default: - fs.stat(p, function (er2, stat) { - // if the stat fails, then that's super weird. - // let the original error be the failure reason. - if (er2 || !stat.isDirectory()) cb(er, made) - else cb(null, made); - }); - break; - } - }); -} - -mkdirP.sync = function sync (p, mode, made) { - if (mode === undefined) { - mode = 0777 & (~process.umask()); - } - if (!made) made = null; - - if (typeof mode === 'string') mode = parseInt(mode, 8); - p = path.resolve(p); - - try { - fs.mkdirSync(p, mode); - made = made || p; - } - catch (err0) { - switch (err0.code) { - case 'ENOENT' : - made = sync(path.dirname(p), mode, made); - sync(p, mode, made); - break; - - // In the case of any other error, just see if there's a dir - // there already. If so, then hooray! If not, then something - // is borked. - default: - var stat; - try { - stat = fs.statSync(p); - } - catch (err1) { - throw err0; - } - if (!stat.isDirectory()) throw err0; - break; - } - } - - return made; -}; diff --git a/node_modules/express/node_modules/mkdirp/package.json b/node_modules/express/node_modules/mkdirp/package.json deleted file mode 100644 index b5200ad..0000000 --- a/node_modules/express/node_modules/mkdirp/package.json +++ /dev/null @@ -1,33 +0,0 @@ -{ - "name": "mkdirp", - "description": "Recursively mkdir, like `mkdir -p`", - "version": "0.3.5", - "author": { - "name": "James Halliday", - "email": "mail@substack.net", - "url": "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.4.0" - }, - "license": "MIT", - "readme": "# mkdirp\n\nLike `mkdir -p`, but in node.js!\n\n[![build status](https://secure.travis-ci.org/substack/node-mkdirp.png)](http://travis-ci.org/substack/node-mkdirp)\n\n# example\n\n## pow.js\n\n```js\nvar mkdirp = require('mkdirp');\n \nmkdirp('/tmp/foo/bar/baz', function (err) {\n if (err) console.error(err)\n else console.log('pow!')\n});\n```\n\nOutput\n\n```\npow!\n```\n\nAnd now /tmp/foo/bar/baz exists, huzzah!\n\n# methods\n\n```js\nvar mkdirp = require('mkdirp');\n```\n\n## mkdirp(dir, mode, cb)\n\nCreate a new directory and any necessary subdirectories at `dir` with octal\npermission string `mode`.\n\nIf `mode` isn't specified, it defaults to `0777 & (~process.umask())`.\n\n`cb(err, made)` fires with the error or the first directory `made`\nthat had to be created, if any.\n\n## mkdirp.sync(dir, mode)\n\nSynchronously create a new directory and any necessary subdirectories at `dir`\nwith octal permission string `mode`.\n\nIf `mode` isn't specified, it defaults to `0777 & (~process.umask())`.\n\nReturns the first directory that had to be created, if any.\n\n# install\n\nWith [npm](http://npmjs.org) do:\n\n```\nnpm install mkdirp\n```\n\n# license\n\nMIT\n", - "readmeFilename": "readme.markdown", - "bugs": { - "url": "https://github.com/substack/node-mkdirp/issues" - }, - "_id": "mkdirp@0.3.5", - "_from": "mkdirp@~0.3.4" -} diff --git a/node_modules/express/node_modules/mkdirp/readme.markdown b/node_modules/express/node_modules/mkdirp/readme.markdown deleted file mode 100644 index 83b0216..0000000 --- a/node_modules/express/node_modules/mkdirp/readme.markdown +++ /dev/null @@ -1,63 +0,0 @@ -# mkdirp - -Like `mkdir -p`, but in node.js! - -[![build status](https://secure.travis-ci.org/substack/node-mkdirp.png)](http://travis-ci.org/substack/node-mkdirp) - -# example - -## pow.js - -```js -var mkdirp = require('mkdirp'); - -mkdirp('/tmp/foo/bar/baz', function (err) { - if (err) console.error(err) - else console.log('pow!') -}); -``` - -Output - -``` -pow! -``` - -And now /tmp/foo/bar/baz exists, huzzah! - -# methods - -```js -var mkdirp = require('mkdirp'); -``` - -## mkdirp(dir, mode, cb) - -Create a new directory and any necessary subdirectories at `dir` with octal -permission string `mode`. - -If `mode` isn't specified, it defaults to `0777 & (~process.umask())`. - -`cb(err, made)` fires with the error or the first directory `made` -that had to be created, if any. - -## mkdirp.sync(dir, mode) - -Synchronously create a new directory and any necessary subdirectories at `dir` -with octal permission string `mode`. - -If `mode` isn't specified, it defaults to `0777 & (~process.umask())`. - -Returns the first directory that had to be created, if any. - -# install - -With [npm](http://npmjs.org) do: - -``` -npm install mkdirp -``` - -# license - -MIT diff --git a/node_modules/express/node_modules/mkdirp/test/chmod.js b/node_modules/express/node_modules/mkdirp/test/chmod.js deleted file mode 100644 index 520dcb8..0000000 --- a/node_modules/express/node_modules/mkdirp/test/chmod.js +++ /dev/null @@ -1,38 +0,0 @@ -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.end(); - }); - }); -}); diff --git a/node_modules/express/node_modules/mkdirp/test/clobber.js b/node_modules/express/node_modules/mkdirp/test/clobber.js deleted file mode 100644 index 0eb7099..0000000 --- a/node_modules/express/node_modules/mkdirp/test/clobber.js +++ /dev/null @@ -1,37 +0,0 @@ -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/node_modules/express/node_modules/mkdirp/test/mkdirp.js b/node_modules/express/node_modules/mkdirp/test/mkdirp.js deleted file mode 100644 index b07cd70..0000000 --- a/node_modules/express/node_modules/mkdirp/test/mkdirp.js +++ /dev/null @@ -1,28 +0,0 @@ -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/node_modules/express/node_modules/mkdirp/test/perm.js b/node_modules/express/node_modules/mkdirp/test/perm.js deleted file mode 100644 index 23a7abb..0000000 --- a/node_modules/express/node_modules/mkdirp/test/perm.js +++ /dev/null @@ -1,32 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('async perm', function (t) { - t.plan(2); - var file = '/tmp/' + (Math.random() * (1<<30)).toString(16); - - 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(); - } - }) - }) - }); -}); - -test('async root perm', function (t) { - mkdirp('/tmp', 0755, function (err) { - if (err) t.fail(err); - t.end(); - }); - t.end(); -}); diff --git a/node_modules/express/node_modules/mkdirp/test/perm_sync.js b/node_modules/express/node_modules/mkdirp/test/perm_sync.js deleted file mode 100644 index f685f60..0000000 --- a/node_modules/express/node_modules/mkdirp/test/perm_sync.js +++ /dev/null @@ -1,39 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('sync perm', function (t) { - t.plan(2); - var file = '/tmp/' + (Math.random() * (1<<30)).toString(16) + '.json'; - - mkdirp.sync(file, 0755); - 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(); - } - }) - }); -}); - -test('sync root perm', function (t) { - t.plan(1); - - var file = '/tmp'; - mkdirp.sync(file, 0755); - 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.ok(stat.isDirectory(), 'target not a directory'); - t.end(); - } - }) - }); -}); diff --git a/node_modules/express/node_modules/mkdirp/test/race.js b/node_modules/express/node_modules/mkdirp/test/race.js deleted file mode 100644 index 96a0447..0000000 --- a/node_modules/express/node_modules/mkdirp/test/race.js +++ /dev/null @@ -1,41 +0,0 @@ -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/node_modules/express/node_modules/mkdirp/test/rel.js b/node_modules/express/node_modules/mkdirp/test/rel.js deleted file mode 100644 index 7985824..0000000 --- a/node_modules/express/node_modules/mkdirp/test/rel.js +++ /dev/null @@ -1,32 +0,0 @@ -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/node_modules/express/node_modules/mkdirp/test/return.js b/node_modules/express/node_modules/mkdirp/test/return.js deleted file mode 100644 index bce68e5..0000000 --- a/node_modules/express/node_modules/mkdirp/test/return.js +++ /dev/null @@ -1,25 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('return value', function (t) { - t.plan(4); - 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('/'); - - // should return the first dir created. - // By this point, it would be profoundly surprising if /tmp didn't - // already exist, since every other test makes things in there. - mkdirp(file, function (err, made) { - t.ifError(err); - t.equal(made, '/tmp/' + x); - mkdirp(file, function (err, made) { - t.ifError(err); - t.equal(made, null); - }); - }); -}); diff --git a/node_modules/express/node_modules/mkdirp/test/return_sync.js b/node_modules/express/node_modules/mkdirp/test/return_sync.js deleted file mode 100644 index 7c222d3..0000000 --- a/node_modules/express/node_modules/mkdirp/test/return_sync.js +++ /dev/null @@ -1,24 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('return value', 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('/'); - - // should return the first dir created. - // By this point, it would be profoundly surprising if /tmp didn't - // already exist, since every other test makes things in there. - // Note that this will throw on failure, which will fail the test. - var made = mkdirp.sync(file); - t.equal(made, '/tmp/' + x); - - // making the same file again should have no effect. - made = mkdirp.sync(file); - t.equal(made, null); -}); diff --git a/node_modules/express/node_modules/mkdirp/test/root.js b/node_modules/express/node_modules/mkdirp/test/root.js deleted file mode 100644 index 97ad7a2..0000000 --- a/node_modules/express/node_modules/mkdirp/test/root.js +++ /dev/null @@ -1,18 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('root', function (t) { - // '/' on unix, 'c:/' on windows. - var file = path.resolve('/'); - - mkdirp(file, 0755, function (err) { - if (err) throw err - fs.stat(file, function (er, stat) { - if (er) throw er - t.ok(stat.isDirectory(), 'target is a directory'); - t.end(); - }) - }); -}); diff --git a/node_modules/express/node_modules/mkdirp/test/sync.js b/node_modules/express/node_modules/mkdirp/test/sync.js deleted file mode 100644 index 7530cad..0000000 --- a/node_modules/express/node_modules/mkdirp/test/sync.js +++ /dev/null @@ -1,32 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('sync', 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('/'); - - try { - mkdirp.sync(file, 0755); - } catch (err) { - t.fail(err); - return t.end(); - } - - 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/node_modules/express/node_modules/mkdirp/test/umask.js b/node_modules/express/node_modules/mkdirp/test/umask.js deleted file mode 100644 index 64ccafe..0000000 --- a/node_modules/express/node_modules/mkdirp/test/umask.js +++ /dev/null @@ -1,28 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('implicit mode from umask', 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, 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, 0777 & (~process.umask())); - t.ok(stat.isDirectory(), 'target not a directory'); - t.end(); - } - }) - }) - }); -}); diff --git a/node_modules/express/node_modules/mkdirp/test/umask_sync.js b/node_modules/express/node_modules/mkdirp/test/umask_sync.js deleted file mode 100644 index 35bd5cb..0000000 --- a/node_modules/express/node_modules/mkdirp/test/umask_sync.js +++ /dev/null @@ -1,32 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('umask sync modes', 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('/'); - - try { - mkdirp.sync(file); - } catch (err) { - t.fail(err); - return t.end(); - } - - 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, (0777 & (~process.umask()))); - t.ok(stat.isDirectory(), 'target not a directory'); - t.end(); - } - }); - }); -}); diff --git a/node_modules/express/node_modules/range-parser/.npmignore b/node_modules/express/node_modules/range-parser/.npmignore deleted file mode 100644 index 9daeafb..0000000 --- a/node_modules/express/node_modules/range-parser/.npmignore +++ /dev/null @@ -1 +0,0 @@ -test diff --git a/node_modules/express/node_modules/range-parser/History.md b/node_modules/express/node_modules/range-parser/History.md deleted file mode 100644 index 82df7b1..0000000 --- a/node_modules/express/node_modules/range-parser/History.md +++ /dev/null @@ -1,15 +0,0 @@ - -0.0.4 / 2012-06-17 -================== - - * changed: ret -1 for unsatisfiable and -2 when invalid - -0.0.3 / 2012-06-17 -================== - - * fix last-byte-pos default to len - 1 - -0.0.2 / 2012-06-14 -================== - - * add `.type` diff --git a/node_modules/express/node_modules/range-parser/Makefile b/node_modules/express/node_modules/range-parser/Makefile deleted file mode 100644 index 8e8640f..0000000 --- a/node_modules/express/node_modules/range-parser/Makefile +++ /dev/null @@ -1,7 +0,0 @@ - -test: - @./node_modules/.bin/mocha \ - --reporter spec \ - --require should - -.PHONY: test \ No newline at end of file diff --git a/node_modules/express/node_modules/range-parser/Readme.md b/node_modules/express/node_modules/range-parser/Readme.md deleted file mode 100644 index b2a67fe..0000000 --- a/node_modules/express/node_modules/range-parser/Readme.md +++ /dev/null @@ -1,28 +0,0 @@ - -# node-range-parser - - Range header field parser. - -## Example: - -```js -assert(-1 == parse(200, 'bytes=500-20')); -assert(-2 == parse(200, 'bytes=malformed')); -parse(200, 'bytes=0-499').should.eql(arr('bytes', [{ start: 0, end: 199 }])); -parse(1000, 'bytes=0-499').should.eql(arr('bytes', [{ start: 0, end: 499 }])); -parse(1000, 'bytes=40-80').should.eql(arr('bytes', [{ start: 40, end: 80 }])); -parse(1000, 'bytes=-500').should.eql(arr('bytes', [{ start: 500, end: 999 }])); -parse(1000, 'bytes=-400').should.eql(arr('bytes', [{ start: 600, end: 999 }])); -parse(1000, 'bytes=500-').should.eql(arr('bytes', [{ start: 500, end: 999 }])); -parse(1000, 'bytes=400-').should.eql(arr('bytes', [{ start: 400, end: 999 }])); -parse(1000, 'bytes=0-0').should.eql(arr('bytes', [{ start: 0, end: 0 }])); -parse(1000, 'bytes=-1').should.eql(arr('bytes', [{ start: 999, end: 999 }])); -parse(1000, 'items=0-5').should.eql(arr('items', [{ start: 0, end: 5 }])); -parse(1000, 'bytes=40-80,-1').should.eql(arr('bytes', [{ start: 40, end: 80 }, { start: 999, end: 999 }])); -``` - -## Installation - -``` -$ npm install range-parser -``` \ No newline at end of file diff --git a/node_modules/express/node_modules/range-parser/index.js b/node_modules/express/node_modules/range-parser/index.js deleted file mode 100644 index 9b0f7a8..0000000 --- a/node_modules/express/node_modules/range-parser/index.js +++ /dev/null @@ -1,49 +0,0 @@ - -/** - * Parse "Range" header `str` relative to the given file `size`. - * - * @param {Number} size - * @param {String} str - * @return {Array} - * @api public - */ - -module.exports = function(size, str){ - var valid = true; - var i = str.indexOf('='); - - if (-1 == i) return -2; - - var arr = str.slice(i + 1).split(',').map(function(range){ - var range = range.split('-') - , start = parseInt(range[0], 10) - , end = parseInt(range[1], 10); - - // -nnn - if (isNaN(start)) { - start = size - end; - end = size - 1; - // nnn- - } else if (isNaN(end)) { - end = size - 1; - } - - // limit last-byte-pos to current length - if (end > size - 1) end = size - 1; - - // invalid - if (isNaN(start) - || isNaN(end) - || start > end - || start < 0) valid = false; - - return { - start: start, - end: end - }; - }); - - arr.type = str.slice(0, i); - - return valid ? arr : -1; -}; \ No newline at end of file diff --git a/node_modules/express/node_modules/range-parser/package.json b/node_modules/express/node_modules/range-parser/package.json deleted file mode 100644 index efdf450..0000000 --- a/node_modules/express/node_modules/range-parser/package.json +++ /dev/null @@ -1,20 +0,0 @@ -{ - "name": "range-parser", - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca", - "url": "http://tjholowaychuk.com" - }, - "description": "Range header field string parser", - "version": "0.0.4", - "main": "index.js", - "dependencies": {}, - "devDependencies": { - "mocha": "*", - "should": "*" - }, - "readme": "\n# node-range-parser\n\n Range header field parser.\n\n## Example:\n\n```js\nassert(-1 == parse(200, 'bytes=500-20'));\nassert(-2 == parse(200, 'bytes=malformed'));\nparse(200, 'bytes=0-499').should.eql(arr('bytes', [{ start: 0, end: 199 }]));\nparse(1000, 'bytes=0-499').should.eql(arr('bytes', [{ start: 0, end: 499 }]));\nparse(1000, 'bytes=40-80').should.eql(arr('bytes', [{ start: 40, end: 80 }]));\nparse(1000, 'bytes=-500').should.eql(arr('bytes', [{ start: 500, end: 999 }]));\nparse(1000, 'bytes=-400').should.eql(arr('bytes', [{ start: 600, end: 999 }]));\nparse(1000, 'bytes=500-').should.eql(arr('bytes', [{ start: 500, end: 999 }]));\nparse(1000, 'bytes=400-').should.eql(arr('bytes', [{ start: 400, end: 999 }]));\nparse(1000, 'bytes=0-0').should.eql(arr('bytes', [{ start: 0, end: 0 }]));\nparse(1000, 'bytes=-1').should.eql(arr('bytes', [{ start: 999, end: 999 }]));\nparse(1000, 'items=0-5').should.eql(arr('items', [{ start: 0, end: 5 }]));\nparse(1000, 'bytes=40-80,-1').should.eql(arr('bytes', [{ start: 40, end: 80 }, { start: 999, end: 999 }]));\n```\n\n## Installation\n\n```\n$ npm install range-parser\n```", - "readmeFilename": "Readme.md", - "_id": "range-parser@0.0.4", - "_from": "range-parser@0.0.4" -} diff --git a/node_modules/express/node_modules/send/.npmignore b/node_modules/express/node_modules/send/.npmignore deleted file mode 100644 index f1250e5..0000000 --- a/node_modules/express/node_modules/send/.npmignore +++ /dev/null @@ -1,4 +0,0 @@ -support -test -examples -*.sock diff --git a/node_modules/express/node_modules/send/History.md b/node_modules/express/node_modules/send/History.md deleted file mode 100644 index 20c5319..0000000 --- a/node_modules/express/node_modules/send/History.md +++ /dev/null @@ -1,25 +0,0 @@ - -0.1.0 / 2012-08-25 -================== - - * add options parameter to send() that is passed to fs.createReadStream() [kanongil] - -0.0.4 / 2012-08-16 -================== - - * allow custom "Accept-Ranges" definition - -0.0.3 / 2012-07-16 -================== - - * fix normalization of the root directory. Closes #3 - -0.0.2 / 2012-07-09 -================== - - * add passing of req explicitly for now (YUCK) - -0.0.1 / 2010-01-03 -================== - - * Initial release diff --git a/node_modules/express/node_modules/send/Makefile b/node_modules/express/node_modules/send/Makefile deleted file mode 100644 index a9dcfd5..0000000 --- a/node_modules/express/node_modules/send/Makefile +++ /dev/null @@ -1,8 +0,0 @@ - -test: - @./node_modules/.bin/mocha \ - --require should \ - --reporter spec \ - --bail - -.PHONY: test \ No newline at end of file diff --git a/node_modules/express/node_modules/send/Readme.md b/node_modules/express/node_modules/send/Readme.md deleted file mode 100644 index 85171a9..0000000 --- a/node_modules/express/node_modules/send/Readme.md +++ /dev/null @@ -1,123 +0,0 @@ - -# send - - Send is Connect's `static()` extracted for generalized use, a streaming static file - server supporting partial responses (Ranges), conditional-GET negotiation, high test coverage, and granular events which may be leveraged to take appropriate actions in your application or framework. - -## Installation - - $ npm install send - -## Examples - - Small: - -```js -var http = require('http'); -var send = require('send'); - -var app = http.createServer(function(req, res){ - send(req, req.url).pipe(res); -}); -``` - - Serving from a root directory with custom error-handling: - -```js -var http = require('http'); -var send = require('send'); - -var app = http.createServer(function(req, res){ - // your custom error-handling logic: - function error(err) { - res.statusCode = err.status || 500; - res.end(err.message); - } - - // your custom directory handling logic: - function redirect() { - res.statusCode = 301; - res.setHeader('Location', req.url + '/'); - res.end('Redirecting to ' + req.url + '/'); - } - - // transfer arbitrary files from within - // /www/example.com/public/* - send(req, url.parse(req.url).pathname) - .root('/www/example.com/public') - .on('error', error) - .on('directory', redirect) - .pipe(res); -}); -``` - -## API - -### Events - - - `error` an error occurred `(err)` - - `directory` a directory was requested - - `stream` file streaming has started `(stream)` - - `end` streaming has completed - -### .root(dir) - - Serve files relative to `path`. Aliased as `.from(dir)`. - -### .index(path) - - By default send supports "index.html" files, to disable this - invoke `.index(false)` or to supply a new index pass a string. - -### .maxage(ms) - - Provide a max-age in milliseconds for http caching, defaults to 0. - -## Error-handling - - By default when no `error` listeners are present an automatic response will be made, otherwise you have full control over the response, aka you may show a 5xx page etc. - -## Caching - - It does _not_ perform internal caching, you should use a reverse proxy cache such - as Varnish for this, or those fancy things called CDNs. If your application is small enough that it would benefit from single-node memory caching, it's small enough that it does not need caching at all ;). - -## Debugging - - To enable `debug()` instrumentation output export __DEBUG__: - -``` -$ DEBUG=send node app -``` - -## Running tests - -``` -$ npm install -$ make test -``` - -## License - -(The MIT License) - -Copyright (c) 2012 TJ Holowaychuk <tj@vision-media.ca> - -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. \ No newline at end of file diff --git a/node_modules/express/node_modules/send/index.js b/node_modules/express/node_modules/send/index.js deleted file mode 100644 index f17158d..0000000 --- a/node_modules/express/node_modules/send/index.js +++ /dev/null @@ -1,2 +0,0 @@ - -module.exports = require('./lib/send'); \ No newline at end of file diff --git a/node_modules/express/node_modules/send/lib/send.js b/node_modules/express/node_modules/send/lib/send.js deleted file mode 100644 index de72146..0000000 --- a/node_modules/express/node_modules/send/lib/send.js +++ /dev/null @@ -1,473 +0,0 @@ - -/** - * Module dependencies. - */ - -var debug = require('debug')('send') - , parseRange = require('range-parser') - , Stream = require('stream') - , mime = require('mime') - , fresh = require('fresh') - , path = require('path') - , http = require('http') - , fs = require('fs') - , basename = path.basename - , normalize = path.normalize - , join = path.join - , utils = require('./utils'); - -/** - * Expose `send`. - */ - -exports = module.exports = send; - -/** - * Expose mime module. - */ - -exports.mime = mime; - -/** - * Return a `SendStream` for `req` and `path`. - * - * @param {Request} req - * @param {String} path - * @param {Object} options - * @return {SendStream} - * @api public - */ - -function send(req, path, options) { - return new SendStream(req, path, options); -} - -/** - * Initialize a `SendStream` with the given `path`. - * - * Events: - * - * - `error` an error occurred - * - `stream` file streaming has started - * - `end` streaming has completed - * - `directory` a directory was requested - * - * @param {Request} req - * @param {String} path - * @param {Object} options - * @api private - */ - -function SendStream(req, path, options) { - var self = this; - this.req = req; - this.path = path; - this.options = options || {}; - this.maxage(0); - this.hidden(false); - this.index('index.html'); -} - -/** - * Inherits from `Stream.prototype`. - */ - -SendStream.prototype.__proto__ = Stream.prototype; - -/** - * Enable or disable "hidden" (dot) files. - * - * @param {Boolean} path - * @return {SendStream} - * @api public - */ - -SendStream.prototype.hidden = function(val){ - debug('hidden %s', val); - this._hidden = val; - return this; -}; - -/** - * Set index `path`, set to a falsy - * value to disable index support. - * - * @param {String|Boolean} path - * @return {SendStream} - * @api public - */ - -SendStream.prototype.index = function(path){ - debug('index %s', path); - this._index = path; - return this; -}; - -/** - * Set root `path`. - * - * @param {String} path - * @return {SendStream} - * @api public - */ - -SendStream.prototype.root = -SendStream.prototype.from = function(path){ - this._root = normalize(path); - return this; -}; - -/** - * Set max-age to `ms`. - * - * @param {Number} ms - * @return {SendStream} - * @api public - */ - -SendStream.prototype.maxage = function(ms){ - if (Infinity == ms) ms = 60 * 60 * 24 * 365 * 1000; - debug('max-age %d', ms); - this._maxage = ms; - return this; -}; - -/** - * Emit error with `status`. - * - * @param {Number} status - * @api private - */ - -SendStream.prototype.error = function(status, err){ - var res = this.res; - var msg = http.STATUS_CODES[status]; - err = err || new Error(msg); - err.status = status; - if (this.listeners('error').length) return this.emit('error', err); - res.statusCode = err.status; - res.end(msg); -}; - -/** - * Check if the pathname is potentially malicious. - * - * @return {Boolean} - * @api private - */ - -SendStream.prototype.isMalicious = function(){ - return !this._root && ~this.path.indexOf('..'); -}; - -/** - * Check if the pathname ends with "/". - * - * @return {Boolean} - * @api private - */ - -SendStream.prototype.hasTrailingSlash = function(){ - return '/' == this.path[this.path.length - 1]; -}; - -/** - * Check if the basename leads with ".". - * - * @return {Boolean} - * @api private - */ - -SendStream.prototype.hasLeadingDot = function(){ - return '.' == basename(this.path)[0]; -}; - -/** - * Check if this is a conditional GET request. - * - * @return {Boolean} - * @api private - */ - -SendStream.prototype.isConditionalGET = function(){ - return this.req.headers['if-none-match'] - || this.req.headers['if-modified-since']; -}; - -/** - * Strip content-* header fields. - * - * @api private - */ - -SendStream.prototype.removeContentHeaderFields = function(){ - var res = this.res; - Object.keys(res._headers).forEach(function(field){ - if (0 == field.indexOf('content')) { - res.removeHeader(field); - } - }); -}; - -/** - * Respond with 304 not modified. - * - * @api private - */ - -SendStream.prototype.notModified = function(){ - var res = this.res; - debug('not modified'); - this.removeContentHeaderFields(); - res.statusCode = 304; - res.end(); -}; - -/** - * Check if the request is cacheable, aka - * responded with 2xx or 304 (see RFC 2616 section 14.2{5,6}). - * - * @return {Boolean} - * @api private - */ - -SendStream.prototype.isCachable = function(){ - var res = this.res; - return (res.statusCode >= 200 && res.statusCode < 300) || 304 == res.statusCode; -}; - -/** - * Handle stat() error. - * - * @param {Error} err - * @api private - */ - -SendStream.prototype.onStatError = function(err){ - var notfound = ['ENOENT', 'ENAMETOOLONG', 'ENOTDIR']; - if (~notfound.indexOf(err.code)) return this.error(404, err); - this.error(500, err); -}; - -/** - * Check if the cache is fresh. - * - * @return {Boolean} - * @api private - */ - -SendStream.prototype.isFresh = function(){ - return fresh(this.req.headers, this.res._headers); -}; - -/** - * Redirect to `path`. - * - * @param {String} path - * @api private - */ - -SendStream.prototype.redirect = function(path){ - if (this.listeners('directory').length) return this.emit('directory'); - var res = this.res; - path += '/'; - res.statusCode = 301; - res.setHeader('Location', path); - res.end('Redirecting to ' + utils.escape(path)); -}; - -/** - * Pipe to `res. - * - * @param {Stream} res - * @return {Stream} res - * @api public - */ - -SendStream.prototype.pipe = function(res){ - var self = this - , args = arguments - , path = this.path - , root = this._root; - - // references - this.res = res; - - // invalid request uri - path = utils.decode(path); - if (-1 == path) return this.error(400); - - // null byte(s) - if (~path.indexOf('\0')) return this.error(400); - - // join / normalize from optional root dir - if (root) path = normalize(join(this._root, path)); - - // ".." is malicious without "root" - if (this.isMalicious()) return this.error(403); - - // malicious path - if (root && 0 != path.indexOf(root)) return this.error(403); - - // hidden file support - if (!this._hidden && this.hasLeadingDot()) return this.error(404); - - // index file support - if (this._index && this.hasTrailingSlash()) path += this._index; - - debug('stat "%s"', path); - fs.stat(path, function(err, stat){ - if (err) return self.onStatError(err); - if (stat.isDirectory()) return self.redirect(self.path); - self.send(path, stat); - }); - - return res; -}; - -/** - * Transfer `path`. - * - * @param {String} path - * @api public - */ - -SendStream.prototype.send = function(path, stat){ - var options = this.options; - var len = stat.size; - var res = this.res; - var req = this.req; - var ranges = req.headers.range; - var offset = options.start || 0; - - // set header fields - this.setHeader(stat); - - // set content-type - this.type(path); - - // conditional GET support - if (this.isConditionalGET() - && this.isCachable() - && this.isFresh()) { - return this.notModified(); - } - - // adjust len to start/end options - len = Math.max(0, len - offset); - if (options.end !== undefined) { - var bytes = options.end - offset + 1; - if (len > bytes) len = bytes; - } - - // Range support - if (ranges) { - ranges = parseRange(len, ranges); - - // unsatisfiable - if (-1 == ranges) { - res.setHeader('Content-Range', 'bytes */' + stat.size); - return this.error(416); - } - - // valid (syntactically invalid ranges are treated as a regular response) - if (-2 != ranges) { - options.start = offset + ranges[0].start; - options.end = offset + ranges[0].end; - - // Content-Range - res.statusCode = 206; - res.setHeader('Content-Range', 'bytes ' - + ranges[0].start - + '-' - + ranges[0].end - + '/' - + len); - len = options.end - options.start + 1; - } - } - - // content-length - res.setHeader('Content-Length', len); - - // HEAD support - if ('HEAD' == req.method) return res.end(); - - this.stream(path, options); -}; - -/** - * Stream `path` to the response. - * - * @param {String} path - * @param {Object} options - * @api private - */ - -SendStream.prototype.stream = function(path, options){ - // TODO: this is all lame, refactor meeee - var self = this; - var res = this.res; - var req = this.req; - - // pipe - var stream = fs.createReadStream(path, options); - this.emit('stream', stream); - stream.pipe(res); - - // socket closed, done with the fd - req.on('close', stream.destroy.bind(stream)); - - // error handling code-smell - stream.on('error', function(err){ - // no hope in responding - if (res._header) { - console.error(err.stack); - req.destroy(); - return; - } - - // 500 - err.status = 500; - self.emit('error', err); - }); - - // end - stream.on('end', function(){ - self.emit('end'); - }); -}; - -/** - * Set content-type based on `path` - * if it hasn't been explicitly set. - * - * @param {String} path - * @api private - */ - -SendStream.prototype.type = function(path){ - var res = this.res; - if (res.getHeader('Content-Type')) return; - var type = mime.lookup(path); - var charset = mime.charsets.lookup(type); - debug('content-type %s', type); - res.setHeader('Content-Type', type + (charset ? '; charset=' + charset : '')); -}; - -/** - * Set reaponse header fields, most - * fields may be pre-defined. - * - * @param {Object} stat - * @api private - */ - -SendStream.prototype.setHeader = function(stat){ - var res = this.res; - if (!res.getHeader('Accept-Ranges')) res.setHeader('Accept-Ranges', 'bytes'); - if (!res.getHeader('ETag')) res.setHeader('ETag', utils.etag(stat)); - if (!res.getHeader('Date')) res.setHeader('Date', new Date().toUTCString()); - if (!res.getHeader('Cache-Control')) res.setHeader('Cache-Control', 'public, max-age=' + (this._maxage / 1000)); - if (!res.getHeader('Last-Modified')) res.setHeader('Last-Modified', stat.mtime.toUTCString()); -}; diff --git a/node_modules/express/node_modules/send/lib/utils.js b/node_modules/express/node_modules/send/lib/utils.js deleted file mode 100644 index 950e5a2..0000000 --- a/node_modules/express/node_modules/send/lib/utils.js +++ /dev/null @@ -1,47 +0,0 @@ - -/** - * Return an ETag in the form of `"-"` - * from the given `stat`. - * - * @param {Object} stat - * @return {String} - * @api private - */ - -exports.etag = function(stat) { - return '"' + stat.size + '-' + Number(stat.mtime) + '"'; -}; - -/** - * decodeURIComponent. - * - * Allows V8 to only deoptimize this fn instead of all - * of send(). - * - * @param {String} path - * @api private - */ - -exports.decode = function(path){ - try { - return decodeURIComponent(path); - } catch (err) { - return -1; - } -}; - -/** - * Escape the given string of `html`. - * - * @param {String} html - * @return {String} - * @api private - */ - -exports.escape = function(html){ - return String(html) - .replace(/&(?!\w+;)/g, '&') - .replace(//g, '>') - .replace(/"/g, '"'); -}; \ No newline at end of file diff --git a/node_modules/express/node_modules/send/node_modules/mime/LICENSE b/node_modules/express/node_modules/send/node_modules/mime/LICENSE deleted file mode 100644 index 451fc45..0000000 --- a/node_modules/express/node_modules/send/node_modules/mime/LICENSE +++ /dev/null @@ -1,19 +0,0 @@ -Copyright (c) 2010 Benjamin Thomas, Robert Kieffer - -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/node_modules/express/node_modules/send/node_modules/mime/README.md b/node_modules/express/node_modules/send/node_modules/mime/README.md deleted file mode 100644 index d8b66a8..0000000 --- a/node_modules/express/node_modules/send/node_modules/mime/README.md +++ /dev/null @@ -1,63 +0,0 @@ -# mime - -Comprehensive MIME type mapping API. Includes all 600+ types and 800+ extensions defined by the Apache project, plus additional types submitted by the node.js community. - -## Install - -Install with [npm](http://github.com/isaacs/npm): - - npm install mime - -## API - Queries - -### mime.lookup(path) -Get the mime type associated with a file. Performs a case-insensitive lookup using the extension in `path` (the substring after the last '/' or '.'). E.g. - - var mime = require('mime'); - - mime.lookup('/path/to/file.txt'); // => 'text/plain' - mime.lookup('file.txt'); // => 'text/plain' - mime.lookup('.TXT'); // => 'text/plain' - mime.lookup('htm'); // => 'text/html' - -### mime.extension(type) -Get the default extension for `type` - - mime.extension('text/html'); // => 'html' - mime.extension('application/octet-stream'); // => 'bin' - -### mime.charsets.lookup() - -Map mime-type to charset - - mime.charsets.lookup('text/plain'); // => 'UTF-8' - -(The logic for charset lookups is pretty rudimentary. Feel free to suggest improvements.) - -## API - Defining Custom Types - -The following APIs allow you to add your own type mappings within your project. If you feel a type should be included as part of node-mime, see [requesting new types](https://github.com/bentomas/node-mime/wiki/Requesting-New-Types). - -### mime.define() - -Add custom mime/extension mappings - - mime.define({ - 'text/x-some-format': ['x-sf', 'x-sft', 'x-sfml'], - 'application/x-my-type': ['x-mt', 'x-mtt'], - // etc ... - }); - - mime.lookup('x-sft'); // => 'text/x-some-format' - -The first entry in the extensions array is returned by `mime.extension()`. E.g. - - mime.extension('text/x-some-format'); // => 'x-sf' - -### mime.load(filepath) - -Load mappings from an Apache ".types" format file - - mime.load('./my_project.types'); - -The .types file format is simple - See the `types` dir for examples. diff --git a/node_modules/express/node_modules/send/node_modules/mime/mime.js b/node_modules/express/node_modules/send/node_modules/mime/mime.js deleted file mode 100644 index 1e00585..0000000 --- a/node_modules/express/node_modules/send/node_modules/mime/mime.js +++ /dev/null @@ -1,104 +0,0 @@ -var path = require('path'); -var fs = require('fs'); - -function Mime() { - // Map of extension -> mime type - this.types = Object.create(null); - - // Map of mime type -> extension - this.extensions = Object.create(null); -} - -/** - * Define mimetype -> extension mappings. Each key is a mime-type that maps - * to an array of extensions associated with the type. The first extension is - * used as the default extension for the type. - * - * e.g. mime.define({'audio/ogg', ['oga', 'ogg', 'spx']}); - * - * @param map (Object) type definitions - */ -Mime.prototype.define = function (map) { - for (var type in map) { - var exts = map[type]; - - for (var i = 0; i < exts.length; i++) { - this.types[exts[i]] = type; - } - - // Default extension is the first one we encounter - if (!this.extensions[type]) { - this.extensions[type] = exts[0]; - } - } -}; - -/** - * Load an Apache2-style ".types" file - * - * This may be called multiple times (it's expected). Where files declare - * overlapping types/extensions, the last file wins. - * - * @param file (String) path of file to load. - */ -Mime.prototype.load = function(file) { - // Read file and split into lines - var map = {}, - content = fs.readFileSync(file, 'ascii'), - lines = content.split(/[\r\n]+/); - - lines.forEach(function(line) { - // Clean up whitespace/comments, and split into fields - var fields = line.replace(/\s*#.*|^\s*|\s*$/g, '').split(/\s+/); - map[fields.shift()] = fields; - }); - - this.define(map); -}; - -/** - * Lookup a mime type based on extension - */ -Mime.prototype.lookup = function(path, fallback) { - var ext = path.replace(/.*[\.\/]/, '').toLowerCase(); - - return this.types[ext] || fallback || this.default_type; -}; - -/** - * Return file extension associated with a mime type - */ -Mime.prototype.extension = function(mimeType) { - return this.extensions[mimeType]; -}; - -// Default instance -var mime = new Mime(); - -// Load local copy of -// http://svn.apache.org/repos/asf/httpd/httpd/trunk/docs/conf/mime.types -mime.load(path.join(__dirname, 'types/mime.types')); - -// Load additional types from node.js community -mime.load(path.join(__dirname, 'types/node.types')); - -// Default type -mime.default_type = mime.lookup('bin'); - -// -// Additional API specific to the default instance -// - -mime.Mime = Mime; - -/** - * Lookup a charset based on mime type. - */ -mime.charsets = { - lookup: function(mimeType, fallback) { - // Assume text types are utf8 - return (/^text\//).test(mimeType) ? 'UTF-8' : fallback; - } -} - -module.exports = mime; diff --git a/node_modules/express/node_modules/send/node_modules/mime/package.json b/node_modules/express/node_modules/send/node_modules/mime/package.json deleted file mode 100644 index 74a6b69..0000000 --- a/node_modules/express/node_modules/send/node_modules/mime/package.json +++ /dev/null @@ -1,35 +0,0 @@ -{ - "author": { - "name": "Robert Kieffer", - "email": "robert@broofa.com", - "url": "http://github.com/broofa" - }, - "contributors": [ - { - "name": "Benjamin Thomas", - "email": "benjamin@benjaminthomas.org", - "url": "http://github.com/bentomas" - } - ], - "dependencies": {}, - "description": "A comprehensive library for mime-type mapping", - "devDependencies": {}, - "keywords": [ - "util", - "mime" - ], - "main": "mime.js", - "name": "mime", - "repository": { - "url": "https://github.com/broofa/node-mime", - "type": "git" - }, - "version": "1.2.6", - "readme": "# mime\n\nComprehensive MIME type mapping API. Includes all 600+ types and 800+ extensions defined by the Apache project, plus additional types submitted by the node.js community.\n\n## Install\n\nInstall with [npm](http://github.com/isaacs/npm):\n\n npm install mime\n\n## API - Queries\n\n### mime.lookup(path)\nGet the mime type associated with a file. Performs a case-insensitive lookup using the extension in `path` (the substring after the last '/' or '.'). E.g.\n\n var mime = require('mime');\n\n mime.lookup('/path/to/file.txt'); // => 'text/plain'\n mime.lookup('file.txt'); // => 'text/plain'\n mime.lookup('.TXT'); // => 'text/plain'\n mime.lookup('htm'); // => 'text/html'\n\n### mime.extension(type)\nGet the default extension for `type`\n\n mime.extension('text/html'); // => 'html'\n mime.extension('application/octet-stream'); // => 'bin'\n\n### mime.charsets.lookup()\n\nMap mime-type to charset\n\n mime.charsets.lookup('text/plain'); // => 'UTF-8'\n\n(The logic for charset lookups is pretty rudimentary. Feel free to suggest improvements.)\n\n## API - Defining Custom Types\n\nThe following APIs allow you to add your own type mappings within your project. If you feel a type should be included as part of node-mime, see [requesting new types](https://github.com/bentomas/node-mime/wiki/Requesting-New-Types).\n\n### mime.define()\n\nAdd custom mime/extension mappings\n\n mime.define({\n 'text/x-some-format': ['x-sf', 'x-sft', 'x-sfml'],\n 'application/x-my-type': ['x-mt', 'x-mtt'],\n // etc ...\n });\n\n mime.lookup('x-sft'); // => 'text/x-some-format'\n\nThe first entry in the extensions array is returned by `mime.extension()`. E.g.\n\n mime.extension('text/x-some-format'); // => 'x-sf'\n\n### mime.load(filepath)\n\nLoad mappings from an Apache \".types\" format file\n\n mime.load('./my_project.types');\n\nThe .types file format is simple - See the `types` dir for examples.\n", - "readmeFilename": "README.md", - "bugs": { - "url": "https://github.com/broofa/node-mime/issues" - }, - "_id": "mime@1.2.6", - "_from": "mime@1.2.6" -} diff --git a/node_modules/express/node_modules/send/node_modules/mime/test.js b/node_modules/express/node_modules/send/node_modules/mime/test.js deleted file mode 100644 index cbad034..0000000 --- a/node_modules/express/node_modules/send/node_modules/mime/test.js +++ /dev/null @@ -1,55 +0,0 @@ -/** - * Usage: node test.js - */ - -var mime = require('./mime'); -var assert = require('assert'); - -function eq(a, b) { - console.log('Test: ' + a + ' === ' + b); - assert.strictEqual.apply(null, arguments); -} - -console.log(Object.keys(mime.extensions).length + ' types'); -console.log(Object.keys(mime.types).length + ' extensions\n'); - -// -// Test mime lookups -// - -eq('text/plain', mime.lookup('text.txt')); -eq('text/plain', mime.lookup('.text.txt')); -eq('text/plain', mime.lookup('.txt')); -eq('text/plain', mime.lookup('txt')); -eq('application/octet-stream', mime.lookup('text.nope')); -eq('fallback', mime.lookup('text.fallback', 'fallback')); -eq('application/octet-stream', mime.lookup('constructor')); -eq('text/plain', mime.lookup('TEXT.TXT')); -eq('text/event-stream', mime.lookup('text/event-stream')); -eq('application/x-web-app-manifest+json', mime.lookup('text.webapp')); - -// -// Test extensions -// - -eq('txt', mime.extension(mime.types.text)); -eq('html', mime.extension(mime.types.htm)); -eq('bin', mime.extension('application/octet-stream')); -eq(undefined, mime.extension('constructor')); - -// -// Test node types -// - -eq('application/octet-stream', mime.lookup('file.buffer')); -eq('audio/mp4', mime.lookup('file.m4a')); - -// -// Test charsets -// - -eq('UTF-8', mime.charsets.lookup('text/plain')); -eq(undefined, mime.charsets.lookup(mime.types.js)); -eq('fallback', mime.charsets.lookup('application/octet-stream', 'fallback')); - -console.log('\nOK'); diff --git a/node_modules/express/node_modules/send/node_modules/mime/types/mime.types b/node_modules/express/node_modules/send/node_modules/mime/types/mime.types deleted file mode 100644 index b3cae2e..0000000 --- a/node_modules/express/node_modules/send/node_modules/mime/types/mime.types +++ /dev/null @@ -1,1510 +0,0 @@ -# This file maps Internet media types to unique file extension(s). -# Although created for httpd, this file is used by many software systems -# and has been placed in the public domain for unlimited redisribution. -# -# The table below contains both registered and (common) unregistered types. -# A type that has no unique extension can be ignored -- they are listed -# here to guide configurations toward known types and to make it easier to -# identify "new" types. File extensions are also commonly used to indicate -# content languages and encodings, so choose them carefully. -# -# Internet media types should be registered as described in RFC 4288. -# The registry is at . -# -# MIME type (lowercased) Extensions -# ============================================ ========== -# application/1d-interleaved-parityfec -# application/3gpp-ims+xml -# application/activemessage -application/andrew-inset ez -# application/applefile -application/applixware aw -application/atom+xml atom -application/atomcat+xml atomcat -# application/atomicmail -application/atomsvc+xml atomsvc -# application/auth-policy+xml -# application/batch-smtp -# application/beep+xml -# application/calendar+xml -# application/cals-1840 -# application/ccmp+xml -application/ccxml+xml ccxml -application/cdmi-capability cdmia -application/cdmi-container cdmic -application/cdmi-domain cdmid -application/cdmi-object cdmio -application/cdmi-queue cdmiq -# application/cea-2018+xml -# application/cellml+xml -# application/cfw -# application/cnrp+xml -# application/commonground -# application/conference-info+xml -# application/cpl+xml -# application/csta+xml -# application/cstadata+xml -application/cu-seeme cu -# application/cybercash -application/davmount+xml davmount -# application/dca-rft -# application/dec-dx -# application/dialog-info+xml -# application/dicom -# application/dns -# application/dskpp+xml -application/dssc+der dssc -application/dssc+xml xdssc -# application/dvcs -application/ecmascript ecma -# application/edi-consent -# application/edi-x12 -# application/edifact -application/emma+xml emma -# application/epp+xml -application/epub+zip epub -# application/eshop -# application/example -application/exi exi -# application/fastinfoset -# application/fastsoap -# application/fits -application/font-tdpfr pfr -# application/framework-attributes+xml -# application/h224 -# application/held+xml -# application/http -application/hyperstudio stk -# application/ibe-key-request+xml -# application/ibe-pkg-reply+xml -# application/ibe-pp-data -# application/iges -# application/im-iscomposing+xml -# application/index -# application/index.cmd -# application/index.obj -# application/index.response -# application/index.vnd -application/inkml+xml ink inkml -# application/iotp -application/ipfix ipfix -# application/ipp -# application/isup -application/java-archive jar -application/java-serialized-object ser -application/java-vm class -application/javascript js -application/json json -# application/kpml-request+xml -# application/kpml-response+xml -application/lost+xml lostxml -application/mac-binhex40 hqx -application/mac-compactpro cpt -# application/macwriteii -application/mads+xml mads -application/marc mrc -application/marcxml+xml mrcx -application/mathematica ma nb mb -# application/mathml-content+xml -# application/mathml-presentation+xml -application/mathml+xml mathml -# application/mbms-associated-procedure-description+xml -# application/mbms-deregister+xml -# application/mbms-envelope+xml -# application/mbms-msk+xml -# application/mbms-msk-response+xml -# application/mbms-protection-description+xml -# application/mbms-reception-report+xml -# application/mbms-register+xml -# application/mbms-register-response+xml -# application/mbms-user-service-description+xml -application/mbox mbox -# application/media_control+xml -application/mediaservercontrol+xml mscml -application/metalink4+xml meta4 -application/mets+xml mets -# application/mikey -application/mods+xml mods -# application/moss-keys -# application/moss-signature -# application/mosskey-data -# application/mosskey-request -application/mp21 m21 mp21 -application/mp4 mp4s -# application/mpeg4-generic -# application/mpeg4-iod -# application/mpeg4-iod-xmt -# application/msc-ivr+xml -# application/msc-mixer+xml -application/msword doc dot -application/mxf mxf -# application/nasdata -# application/news-checkgroups -# application/news-groupinfo -# application/news-transmission -# application/nss -# application/ocsp-request -# application/ocsp-response -application/octet-stream bin dms lha lrf lzh so iso dmg dist distz pkg bpk dump elc deploy -application/oda oda -application/oebps-package+xml opf -application/ogg ogx -application/onenote onetoc onetoc2 onetmp onepkg -application/oxps oxps -# application/parityfec -application/patch-ops-error+xml xer -application/pdf pdf -application/pgp-encrypted pgp -# application/pgp-keys -application/pgp-signature asc sig -application/pics-rules prf -# application/pidf+xml -# application/pidf-diff+xml -application/pkcs10 p10 -application/pkcs7-mime p7m p7c -application/pkcs7-signature p7s -application/pkcs8 p8 -application/pkix-attr-cert ac -application/pkix-cert cer -application/pkix-crl crl -application/pkix-pkipath pkipath -application/pkixcmp pki -application/pls+xml pls -# application/poc-settings+xml -application/postscript ai eps ps -# application/prs.alvestrand.titrax-sheet -application/prs.cww cww -# application/prs.nprend -# application/prs.plucker -# application/prs.rdf-xml-crypt -# application/prs.xsf+xml -application/pskc+xml pskcxml -# application/qsig -application/rdf+xml rdf -application/reginfo+xml rif -application/relax-ng-compact-syntax rnc -# application/remote-printing -application/resource-lists+xml rl -application/resource-lists-diff+xml rld -# application/riscos -# application/rlmi+xml -application/rls-services+xml rs -application/rpki-ghostbusters gbr -application/rpki-manifest mft -application/rpki-roa roa -# application/rpki-updown -application/rsd+xml rsd -application/rss+xml rss -application/rtf rtf -# application/rtx -# application/samlassertion+xml -# application/samlmetadata+xml -application/sbml+xml sbml -application/scvp-cv-request scq -application/scvp-cv-response scs -application/scvp-vp-request spq -application/scvp-vp-response spp -application/sdp sdp -# application/set-payment -application/set-payment-initiation setpay -# application/set-registration -application/set-registration-initiation setreg -# application/sgml -# application/sgml-open-catalog -application/shf+xml shf -# application/sieve -# application/simple-filter+xml -# application/simple-message-summary -# application/simplesymbolcontainer -# application/slate -# application/smil -application/smil+xml smi smil -# application/soap+fastinfoset -# application/soap+xml -application/sparql-query rq -application/sparql-results+xml srx -# application/spirits-event+xml -application/srgs gram -application/srgs+xml grxml -application/sru+xml sru -application/ssml+xml ssml -# application/tamp-apex-update -# application/tamp-apex-update-confirm -# application/tamp-community-update -# application/tamp-community-update-confirm -# application/tamp-error -# application/tamp-sequence-adjust -# application/tamp-sequence-adjust-confirm -# application/tamp-status-query -# application/tamp-status-response -# application/tamp-update -# application/tamp-update-confirm -application/tei+xml tei teicorpus -application/thraud+xml tfi -# application/timestamp-query -# application/timestamp-reply -application/timestamped-data tsd -# application/tve-trigger -# application/ulpfec -# application/vcard+xml -# application/vemmi -# application/vividence.scriptfile -# application/vnd.3gpp.bsf+xml -application/vnd.3gpp.pic-bw-large plb -application/vnd.3gpp.pic-bw-small psb -application/vnd.3gpp.pic-bw-var pvb -# application/vnd.3gpp.sms -# application/vnd.3gpp2.bcmcsinfo+xml -# application/vnd.3gpp2.sms -application/vnd.3gpp2.tcap tcap -application/vnd.3m.post-it-notes pwn -application/vnd.accpac.simply.aso aso -application/vnd.accpac.simply.imp imp -application/vnd.acucobol acu -application/vnd.acucorp atc acutc -application/vnd.adobe.air-application-installer-package+zip air -application/vnd.adobe.fxp fxp fxpl -# application/vnd.adobe.partial-upload -application/vnd.adobe.xdp+xml xdp -application/vnd.adobe.xfdf xfdf -# application/vnd.aether.imp -# application/vnd.ah-barcode -application/vnd.ahead.space ahead -application/vnd.airzip.filesecure.azf azf -application/vnd.airzip.filesecure.azs azs -application/vnd.amazon.ebook azw -application/vnd.americandynamics.acc acc -application/vnd.amiga.ami ami -# application/vnd.amundsen.maze+xml -application/vnd.android.package-archive apk -application/vnd.anser-web-certificate-issue-initiation cii -application/vnd.anser-web-funds-transfer-initiation fti -application/vnd.antix.game-component atx -application/vnd.apple.installer+xml mpkg -application/vnd.apple.mpegurl m3u8 -# application/vnd.arastra.swi -application/vnd.aristanetworks.swi swi -application/vnd.astraea-software.iota iota -application/vnd.audiograph aep -# application/vnd.autopackage -# application/vnd.avistar+xml -application/vnd.blueice.multipass mpm -# application/vnd.bluetooth.ep.oob -application/vnd.bmi bmi -application/vnd.businessobjects rep -# application/vnd.cab-jscript -# application/vnd.canon-cpdl -# application/vnd.canon-lips -# application/vnd.cendio.thinlinc.clientconf -application/vnd.chemdraw+xml cdxml -application/vnd.chipnuts.karaoke-mmd mmd -application/vnd.cinderella cdy -# application/vnd.cirpack.isdn-ext -application/vnd.claymore cla -application/vnd.cloanto.rp9 rp9 -application/vnd.clonk.c4group c4g c4d c4f c4p c4u -application/vnd.cluetrust.cartomobile-config c11amc -application/vnd.cluetrust.cartomobile-config-pkg c11amz -# application/vnd.collection+json -# application/vnd.commerce-battelle -application/vnd.commonspace csp -application/vnd.contact.cmsg cdbcmsg -application/vnd.cosmocaller cmc -application/vnd.crick.clicker clkx -application/vnd.crick.clicker.keyboard clkk -application/vnd.crick.clicker.palette clkp -application/vnd.crick.clicker.template clkt -application/vnd.crick.clicker.wordbank clkw -application/vnd.criticaltools.wbs+xml wbs -application/vnd.ctc-posml pml -# application/vnd.ctct.ws+xml -# application/vnd.cups-pdf -# application/vnd.cups-postscript -application/vnd.cups-ppd ppd -# application/vnd.cups-raster -# application/vnd.cups-raw -# application/vnd.curl -application/vnd.curl.car car -application/vnd.curl.pcurl pcurl -# application/vnd.cybank -application/vnd.data-vision.rdz rdz -application/vnd.dece.data uvf uvvf uvd uvvd -application/vnd.dece.ttml+xml uvt uvvt -application/vnd.dece.unspecified uvx uvvx -application/vnd.dece.zip uvz uvvz -application/vnd.denovo.fcselayout-link fe_launch -# application/vnd.dir-bi.plate-dl-nosuffix -application/vnd.dna dna -application/vnd.dolby.mlp mlp -# application/vnd.dolby.mobile.1 -# application/vnd.dolby.mobile.2 -application/vnd.dpgraph dpg -application/vnd.dreamfactory dfac -application/vnd.dvb.ait ait -# application/vnd.dvb.dvbj -# application/vnd.dvb.esgcontainer -# application/vnd.dvb.ipdcdftnotifaccess -# application/vnd.dvb.ipdcesgaccess -# application/vnd.dvb.ipdcesgaccess2 -# application/vnd.dvb.ipdcesgpdd -# application/vnd.dvb.ipdcroaming -# application/vnd.dvb.iptv.alfec-base -# application/vnd.dvb.iptv.alfec-enhancement -# application/vnd.dvb.notif-aggregate-root+xml -# application/vnd.dvb.notif-container+xml -# application/vnd.dvb.notif-generic+xml -# application/vnd.dvb.notif-ia-msglist+xml -# application/vnd.dvb.notif-ia-registration-request+xml -# application/vnd.dvb.notif-ia-registration-response+xml -# application/vnd.dvb.notif-init+xml -# application/vnd.dvb.pfr -application/vnd.dvb.service svc -# application/vnd.dxr -application/vnd.dynageo geo -# application/vnd.easykaraoke.cdgdownload -# application/vnd.ecdis-update -application/vnd.ecowin.chart mag -# application/vnd.ecowin.filerequest -# application/vnd.ecowin.fileupdate -# application/vnd.ecowin.series -# application/vnd.ecowin.seriesrequest -# application/vnd.ecowin.seriesupdate -# application/vnd.emclient.accessrequest+xml -application/vnd.enliven nml -# application/vnd.eprints.data+xml -application/vnd.epson.esf esf -application/vnd.epson.msf msf -application/vnd.epson.quickanime qam -application/vnd.epson.salt slt -application/vnd.epson.ssf ssf -# application/vnd.ericsson.quickcall -application/vnd.eszigno3+xml es3 et3 -# application/vnd.etsi.aoc+xml -# application/vnd.etsi.cug+xml -# application/vnd.etsi.iptvcommand+xml -# application/vnd.etsi.iptvdiscovery+xml -# application/vnd.etsi.iptvprofile+xml -# application/vnd.etsi.iptvsad-bc+xml -# application/vnd.etsi.iptvsad-cod+xml -# application/vnd.etsi.iptvsad-npvr+xml -# application/vnd.etsi.iptvservice+xml -# application/vnd.etsi.iptvsync+xml -# application/vnd.etsi.iptvueprofile+xml -# application/vnd.etsi.mcid+xml -# application/vnd.etsi.overload-control-policy-dataset+xml -# application/vnd.etsi.sci+xml -# application/vnd.etsi.simservs+xml -# application/vnd.etsi.tsl+xml -# application/vnd.etsi.tsl.der -# application/vnd.eudora.data -application/vnd.ezpix-album ez2 -application/vnd.ezpix-package ez3 -# application/vnd.f-secure.mobile -application/vnd.fdf fdf -application/vnd.fdsn.mseed mseed -application/vnd.fdsn.seed seed dataless -# application/vnd.ffsns -# application/vnd.fints -application/vnd.flographit gph -application/vnd.fluxtime.clip ftc -# application/vnd.font-fontforge-sfd -application/vnd.framemaker fm frame maker book -application/vnd.frogans.fnc fnc -application/vnd.frogans.ltf ltf -application/vnd.fsc.weblaunch fsc -application/vnd.fujitsu.oasys oas -application/vnd.fujitsu.oasys2 oa2 -application/vnd.fujitsu.oasys3 oa3 -application/vnd.fujitsu.oasysgp fg5 -application/vnd.fujitsu.oasysprs bh2 -# application/vnd.fujixerox.art-ex -# application/vnd.fujixerox.art4 -# application/vnd.fujixerox.hbpl -application/vnd.fujixerox.ddd ddd -application/vnd.fujixerox.docuworks xdw -application/vnd.fujixerox.docuworks.binder xbd -# application/vnd.fut-misnet -application/vnd.fuzzysheet fzs -application/vnd.genomatix.tuxedo txd -# application/vnd.geocube+xml -application/vnd.geogebra.file ggb -application/vnd.geogebra.tool ggt -application/vnd.geometry-explorer gex gre -application/vnd.geonext gxt -application/vnd.geoplan g2w -application/vnd.geospace g3w -# application/vnd.globalplatform.card-content-mgt -# application/vnd.globalplatform.card-content-mgt-response -application/vnd.gmx gmx -application/vnd.google-earth.kml+xml kml -application/vnd.google-earth.kmz kmz -application/vnd.grafeq gqf gqs -# application/vnd.gridmp -application/vnd.groove-account gac -application/vnd.groove-help ghf -application/vnd.groove-identity-message gim -application/vnd.groove-injector grv -application/vnd.groove-tool-message gtm -application/vnd.groove-tool-template tpl -application/vnd.groove-vcard vcg -# application/vnd.hal+json -application/vnd.hal+xml hal -application/vnd.handheld-entertainment+xml zmm -application/vnd.hbci hbci -# application/vnd.hcl-bireports -application/vnd.hhe.lesson-player les -application/vnd.hp-hpgl hpgl -application/vnd.hp-hpid hpid -application/vnd.hp-hps hps -application/vnd.hp-jlyt jlt -application/vnd.hp-pcl pcl -application/vnd.hp-pclxl pclxl -# application/vnd.httphone -application/vnd.hydrostatix.sof-data sfd-hdstx -application/vnd.hzn-3d-crossword x3d -# application/vnd.ibm.afplinedata -# application/vnd.ibm.electronic-media -application/vnd.ibm.minipay mpy -application/vnd.ibm.modcap afp listafp list3820 -application/vnd.ibm.rights-management irm -application/vnd.ibm.secure-container sc -application/vnd.iccprofile icc icm -application/vnd.igloader igl -application/vnd.immervision-ivp ivp -application/vnd.immervision-ivu ivu -# application/vnd.informedcontrol.rms+xml -# application/vnd.informix-visionary -# application/vnd.infotech.project -# application/vnd.infotech.project+xml -application/vnd.insors.igm igm -application/vnd.intercon.formnet xpw xpx -application/vnd.intergeo i2g -# application/vnd.intertrust.digibox -# application/vnd.intertrust.nncp -application/vnd.intu.qbo qbo -application/vnd.intu.qfx qfx -# application/vnd.iptc.g2.conceptitem+xml -# application/vnd.iptc.g2.knowledgeitem+xml -# application/vnd.iptc.g2.newsitem+xml -# application/vnd.iptc.g2.packageitem+xml -application/vnd.ipunplugged.rcprofile rcprofile -application/vnd.irepository.package+xml irp -application/vnd.is-xpr xpr -application/vnd.isac.fcs fcs -application/vnd.jam jam -# application/vnd.japannet-directory-service -# application/vnd.japannet-jpnstore-wakeup -# application/vnd.japannet-payment-wakeup -# application/vnd.japannet-registration -# application/vnd.japannet-registration-wakeup -# application/vnd.japannet-setstore-wakeup -# application/vnd.japannet-verification -# application/vnd.japannet-verification-wakeup -application/vnd.jcp.javame.midlet-rms rms -application/vnd.jisp jisp -application/vnd.joost.joda-archive joda -application/vnd.kahootz ktz ktr -application/vnd.kde.karbon karbon -application/vnd.kde.kchart chrt -application/vnd.kde.kformula kfo -application/vnd.kde.kivio flw -application/vnd.kde.kontour kon -application/vnd.kde.kpresenter kpr kpt -application/vnd.kde.kspread ksp -application/vnd.kde.kword kwd kwt -application/vnd.kenameaapp htke -application/vnd.kidspiration kia -application/vnd.kinar kne knp -application/vnd.koan skp skd skt skm -application/vnd.kodak-descriptor sse -application/vnd.las.las+xml lasxml -# application/vnd.liberty-request+xml -application/vnd.llamagraphics.life-balance.desktop lbd -application/vnd.llamagraphics.life-balance.exchange+xml lbe -application/vnd.lotus-1-2-3 123 -application/vnd.lotus-approach apr -application/vnd.lotus-freelance pre -application/vnd.lotus-notes nsf -application/vnd.lotus-organizer org -application/vnd.lotus-screencam scm -application/vnd.lotus-wordpro lwp -application/vnd.macports.portpkg portpkg -# application/vnd.marlin.drm.actiontoken+xml -# application/vnd.marlin.drm.conftoken+xml -# application/vnd.marlin.drm.license+xml -# application/vnd.marlin.drm.mdcf -application/vnd.mcd mcd -application/vnd.medcalcdata mc1 -application/vnd.mediastation.cdkey cdkey -# application/vnd.meridian-slingshot -application/vnd.mfer mwf -application/vnd.mfmp mfm -application/vnd.micrografx.flo flo -application/vnd.micrografx.igx igx -application/vnd.mif mif -# application/vnd.minisoft-hp3000-save -# application/vnd.mitsubishi.misty-guard.trustweb -application/vnd.mobius.daf daf -application/vnd.mobius.dis dis -application/vnd.mobius.mbk mbk -application/vnd.mobius.mqy mqy -application/vnd.mobius.msl msl -application/vnd.mobius.plc plc -application/vnd.mobius.txf txf -application/vnd.mophun.application mpn -application/vnd.mophun.certificate mpc -# application/vnd.motorola.flexsuite -# application/vnd.motorola.flexsuite.adsi -# application/vnd.motorola.flexsuite.fis -# application/vnd.motorola.flexsuite.gotap -# application/vnd.motorola.flexsuite.kmr -# application/vnd.motorola.flexsuite.ttc -# application/vnd.motorola.flexsuite.wem -# application/vnd.motorola.iprm -application/vnd.mozilla.xul+xml xul -application/vnd.ms-artgalry cil -# application/vnd.ms-asf -application/vnd.ms-cab-compressed cab -application/vnd.ms-excel xls xlm xla xlc xlt xlw -application/vnd.ms-excel.addin.macroenabled.12 xlam -application/vnd.ms-excel.sheet.binary.macroenabled.12 xlsb -application/vnd.ms-excel.sheet.macroenabled.12 xlsm -application/vnd.ms-excel.template.macroenabled.12 xltm -application/vnd.ms-fontobject eot -application/vnd.ms-htmlhelp chm -application/vnd.ms-ims ims -application/vnd.ms-lrm lrm -# application/vnd.ms-office.activex+xml -application/vnd.ms-officetheme thmx -application/vnd.ms-pki.seccat cat -application/vnd.ms-pki.stl stl -# application/vnd.ms-playready.initiator+xml -application/vnd.ms-powerpoint ppt pps pot -application/vnd.ms-powerpoint.addin.macroenabled.12 ppam -application/vnd.ms-powerpoint.presentation.macroenabled.12 pptm -application/vnd.ms-powerpoint.slide.macroenabled.12 sldm -application/vnd.ms-powerpoint.slideshow.macroenabled.12 ppsm -application/vnd.ms-powerpoint.template.macroenabled.12 potm -application/vnd.ms-project mpp mpt -# application/vnd.ms-tnef -# application/vnd.ms-wmdrm.lic-chlg-req -# application/vnd.ms-wmdrm.lic-resp -# application/vnd.ms-wmdrm.meter-chlg-req -# application/vnd.ms-wmdrm.meter-resp -application/vnd.ms-word.document.macroenabled.12 docm -application/vnd.ms-word.template.macroenabled.12 dotm -application/vnd.ms-works wps wks wcm wdb -application/vnd.ms-wpl wpl -application/vnd.ms-xpsdocument xps -application/vnd.mseq mseq -# application/vnd.msign -# application/vnd.multiad.creator -# application/vnd.multiad.creator.cif -# application/vnd.music-niff -application/vnd.musician mus -application/vnd.muvee.style msty -application/vnd.mynfc taglet -# application/vnd.ncd.control -# application/vnd.ncd.reference -# application/vnd.nervana -# application/vnd.netfpx -application/vnd.neurolanguage.nlu nlu -application/vnd.noblenet-directory nnd -application/vnd.noblenet-sealer nns -application/vnd.noblenet-web nnw -# application/vnd.nokia.catalogs -# application/vnd.nokia.conml+wbxml -# application/vnd.nokia.conml+xml -# application/vnd.nokia.isds-radio-presets -# application/vnd.nokia.iptv.config+xml -# application/vnd.nokia.landmark+wbxml -# application/vnd.nokia.landmark+xml -# application/vnd.nokia.landmarkcollection+xml -# application/vnd.nokia.n-gage.ac+xml -application/vnd.nokia.n-gage.data ngdat -application/vnd.nokia.n-gage.symbian.install n-gage -# application/vnd.nokia.ncd -# application/vnd.nokia.pcd+wbxml -# application/vnd.nokia.pcd+xml -application/vnd.nokia.radio-preset rpst -application/vnd.nokia.radio-presets rpss -application/vnd.novadigm.edm edm -application/vnd.novadigm.edx edx -application/vnd.novadigm.ext ext -# application/vnd.ntt-local.file-transfer -# application/vnd.ntt-local.sip-ta_remote -# application/vnd.ntt-local.sip-ta_tcp_stream -application/vnd.oasis.opendocument.chart odc -application/vnd.oasis.opendocument.chart-template otc -application/vnd.oasis.opendocument.database odb -application/vnd.oasis.opendocument.formula odf -application/vnd.oasis.opendocument.formula-template odft -application/vnd.oasis.opendocument.graphics odg -application/vnd.oasis.opendocument.graphics-template otg -application/vnd.oasis.opendocument.image odi -application/vnd.oasis.opendocument.image-template oti -application/vnd.oasis.opendocument.presentation odp -application/vnd.oasis.opendocument.presentation-template otp -application/vnd.oasis.opendocument.spreadsheet ods -application/vnd.oasis.opendocument.spreadsheet-template ots -application/vnd.oasis.opendocument.text odt -application/vnd.oasis.opendocument.text-master odm -application/vnd.oasis.opendocument.text-template ott -application/vnd.oasis.opendocument.text-web oth -# application/vnd.obn -# application/vnd.oftn.l10n+json -# application/vnd.oipf.contentaccessdownload+xml -# application/vnd.oipf.contentaccessstreaming+xml -# application/vnd.oipf.cspg-hexbinary -# application/vnd.oipf.dae.svg+xml -# application/vnd.oipf.dae.xhtml+xml -# application/vnd.oipf.mippvcontrolmessage+xml -# application/vnd.oipf.pae.gem -# application/vnd.oipf.spdiscovery+xml -# application/vnd.oipf.spdlist+xml -# application/vnd.oipf.ueprofile+xml -# application/vnd.oipf.userprofile+xml -application/vnd.olpc-sugar xo -# application/vnd.oma-scws-config -# application/vnd.oma-scws-http-request -# application/vnd.oma-scws-http-response -# application/vnd.oma.bcast.associated-procedure-parameter+xml -# application/vnd.oma.bcast.drm-trigger+xml -# application/vnd.oma.bcast.imd+xml -# application/vnd.oma.bcast.ltkm -# application/vnd.oma.bcast.notification+xml -# application/vnd.oma.bcast.provisioningtrigger -# application/vnd.oma.bcast.sgboot -# application/vnd.oma.bcast.sgdd+xml -# application/vnd.oma.bcast.sgdu -# application/vnd.oma.bcast.simple-symbol-container -# application/vnd.oma.bcast.smartcard-trigger+xml -# application/vnd.oma.bcast.sprov+xml -# application/vnd.oma.bcast.stkm -# application/vnd.oma.cab-address-book+xml -# application/vnd.oma.cab-feature-handler+xml -# application/vnd.oma.cab-pcc+xml -# application/vnd.oma.cab-user-prefs+xml -# application/vnd.oma.dcd -# application/vnd.oma.dcdc -application/vnd.oma.dd2+xml dd2 -# application/vnd.oma.drm.risd+xml -# application/vnd.oma.group-usage-list+xml -# application/vnd.oma.pal+xml -# application/vnd.oma.poc.detailed-progress-report+xml -# application/vnd.oma.poc.final-report+xml -# application/vnd.oma.poc.groups+xml -# application/vnd.oma.poc.invocation-descriptor+xml -# application/vnd.oma.poc.optimized-progress-report+xml -# application/vnd.oma.push -# application/vnd.oma.scidm.messages+xml -# application/vnd.oma.xcap-directory+xml -# application/vnd.omads-email+xml -# application/vnd.omads-file+xml -# application/vnd.omads-folder+xml -# application/vnd.omaloc-supl-init -application/vnd.openofficeorg.extension oxt -# application/vnd.openxmlformats-officedocument.custom-properties+xml -# application/vnd.openxmlformats-officedocument.customxmlproperties+xml -# application/vnd.openxmlformats-officedocument.drawing+xml -# application/vnd.openxmlformats-officedocument.drawingml.chart+xml -# application/vnd.openxmlformats-officedocument.drawingml.chartshapes+xml -# application/vnd.openxmlformats-officedocument.drawingml.diagramcolors+xml -# application/vnd.openxmlformats-officedocument.drawingml.diagramdata+xml -# application/vnd.openxmlformats-officedocument.drawingml.diagramlayout+xml -# application/vnd.openxmlformats-officedocument.drawingml.diagramstyle+xml -# application/vnd.openxmlformats-officedocument.extended-properties+xml -# application/vnd.openxmlformats-officedocument.presentationml.commentauthors+xml -# application/vnd.openxmlformats-officedocument.presentationml.comments+xml -# application/vnd.openxmlformats-officedocument.presentationml.handoutmaster+xml -# application/vnd.openxmlformats-officedocument.presentationml.notesmaster+xml -# application/vnd.openxmlformats-officedocument.presentationml.notesslide+xml -application/vnd.openxmlformats-officedocument.presentationml.presentation pptx -# application/vnd.openxmlformats-officedocument.presentationml.presentation.main+xml -# application/vnd.openxmlformats-officedocument.presentationml.presprops+xml -application/vnd.openxmlformats-officedocument.presentationml.slide sldx -# application/vnd.openxmlformats-officedocument.presentationml.slide+xml -# application/vnd.openxmlformats-officedocument.presentationml.slidelayout+xml -# application/vnd.openxmlformats-officedocument.presentationml.slidemaster+xml -application/vnd.openxmlformats-officedocument.presentationml.slideshow ppsx -# application/vnd.openxmlformats-officedocument.presentationml.slideshow.main+xml -# application/vnd.openxmlformats-officedocument.presentationml.slideupdateinfo+xml -# application/vnd.openxmlformats-officedocument.presentationml.tablestyles+xml -# application/vnd.openxmlformats-officedocument.presentationml.tags+xml -application/vnd.openxmlformats-officedocument.presentationml.template potx -# application/vnd.openxmlformats-officedocument.presentationml.template.main+xml -# application/vnd.openxmlformats-officedocument.presentationml.viewprops+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.calcchain+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.chartsheet+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.comments+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.connections+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.dialogsheet+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.externallink+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.pivotcachedefinition+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.pivotcacherecords+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.pivottable+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.querytable+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.revisionheaders+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.revisionlog+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.sharedstrings+xml -application/vnd.openxmlformats-officedocument.spreadsheetml.sheet xlsx -# application/vnd.openxmlformats-officedocument.spreadsheetml.sheet.main+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.sheetmetadata+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.styles+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.table+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.tablesinglecells+xml -application/vnd.openxmlformats-officedocument.spreadsheetml.template xltx -# application/vnd.openxmlformats-officedocument.spreadsheetml.template.main+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.usernames+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.volatiledependencies+xml -# application/vnd.openxmlformats-officedocument.spreadsheetml.worksheet+xml -# application/vnd.openxmlformats-officedocument.theme+xml -# application/vnd.openxmlformats-officedocument.themeoverride+xml -# application/vnd.openxmlformats-officedocument.vmldrawing -# application/vnd.openxmlformats-officedocument.wordprocessingml.comments+xml -application/vnd.openxmlformats-officedocument.wordprocessingml.document docx -# application/vnd.openxmlformats-officedocument.wordprocessingml.document.glossary+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.document.main+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.endnotes+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.fonttable+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.footer+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.footnotes+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.numbering+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.settings+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.styles+xml -application/vnd.openxmlformats-officedocument.wordprocessingml.template dotx -# application/vnd.openxmlformats-officedocument.wordprocessingml.template.main+xml -# application/vnd.openxmlformats-officedocument.wordprocessingml.websettings+xml -# application/vnd.openxmlformats-package.core-properties+xml -# application/vnd.openxmlformats-package.digital-signature-xmlsignature+xml -# application/vnd.openxmlformats-package.relationships+xml -# application/vnd.quobject-quoxdocument -# application/vnd.osa.netdeploy -application/vnd.osgeo.mapguide.package mgp -# application/vnd.osgi.bundle -application/vnd.osgi.dp dp -# application/vnd.otps.ct-kip+xml -application/vnd.palm pdb pqa oprc -# application/vnd.paos.xml -application/vnd.pawaafile paw -application/vnd.pg.format str -application/vnd.pg.osasli ei6 -# application/vnd.piaccess.application-licence -application/vnd.picsel efif -application/vnd.pmi.widget wg -# application/vnd.poc.group-advertisement+xml -application/vnd.pocketlearn plf -application/vnd.powerbuilder6 pbd -# application/vnd.powerbuilder6-s -# application/vnd.powerbuilder7 -# application/vnd.powerbuilder7-s -# application/vnd.powerbuilder75 -# application/vnd.powerbuilder75-s -# application/vnd.preminet -application/vnd.previewsystems.box box -application/vnd.proteus.magazine mgz -application/vnd.publishare-delta-tree qps -application/vnd.pvi.ptid1 ptid -# application/vnd.pwg-multiplexed -# application/vnd.pwg-xhtml-print+xml -# application/vnd.qualcomm.brew-app-res -application/vnd.quark.quarkxpress qxd qxt qwd qwt qxl qxb -# application/vnd.radisys.moml+xml -# application/vnd.radisys.msml+xml -# application/vnd.radisys.msml-audit+xml -# application/vnd.radisys.msml-audit-conf+xml -# application/vnd.radisys.msml-audit-conn+xml -# application/vnd.radisys.msml-audit-dialog+xml -# application/vnd.radisys.msml-audit-stream+xml -# application/vnd.radisys.msml-conf+xml -# application/vnd.radisys.msml-dialog+xml -# application/vnd.radisys.msml-dialog-base+xml -# application/vnd.radisys.msml-dialog-fax-detect+xml -# application/vnd.radisys.msml-dialog-fax-sendrecv+xml -# application/vnd.radisys.msml-dialog-group+xml -# application/vnd.radisys.msml-dialog-speech+xml -# application/vnd.radisys.msml-dialog-transform+xml -# application/vnd.rainstor.data -# application/vnd.rapid -application/vnd.realvnc.bed bed -application/vnd.recordare.musicxml mxl -application/vnd.recordare.musicxml+xml musicxml -# application/vnd.renlearn.rlprint -application/vnd.rig.cryptonote cryptonote -application/vnd.rim.cod cod -application/vnd.rn-realmedia rm -application/vnd.route66.link66+xml link66 -# application/vnd.ruckus.download -# application/vnd.s3sms -application/vnd.sailingtracker.track st -# application/vnd.sbm.cid -# application/vnd.sbm.mid2 -# application/vnd.scribus -# application/vnd.sealed.3df -# application/vnd.sealed.csf -# application/vnd.sealed.doc -# application/vnd.sealed.eml -# application/vnd.sealed.mht -# application/vnd.sealed.net -# application/vnd.sealed.ppt -# application/vnd.sealed.tiff -# application/vnd.sealed.xls -# application/vnd.sealedmedia.softseal.html -# application/vnd.sealedmedia.softseal.pdf -application/vnd.seemail see -application/vnd.sema sema -application/vnd.semd semd -application/vnd.semf semf -application/vnd.shana.informed.formdata ifm -application/vnd.shana.informed.formtemplate itp -application/vnd.shana.informed.interchange iif -application/vnd.shana.informed.package ipk -application/vnd.simtech-mindmapper twd twds -application/vnd.smaf mmf -# application/vnd.smart.notebook -application/vnd.smart.teacher teacher -# application/vnd.software602.filler.form+xml -# application/vnd.software602.filler.form-xml-zip -application/vnd.solent.sdkm+xml sdkm sdkd -application/vnd.spotfire.dxp dxp -application/vnd.spotfire.sfs sfs -# application/vnd.sss-cod -# application/vnd.sss-dtf -# application/vnd.sss-ntf -application/vnd.stardivision.calc sdc -application/vnd.stardivision.draw sda -application/vnd.stardivision.impress sdd -application/vnd.stardivision.math smf -application/vnd.stardivision.writer sdw vor -application/vnd.stardivision.writer-global sgl -application/vnd.stepmania.package smzip -application/vnd.stepmania.stepchart sm -# application/vnd.street-stream -application/vnd.sun.xml.calc sxc -application/vnd.sun.xml.calc.template stc -application/vnd.sun.xml.draw sxd -application/vnd.sun.xml.draw.template std -application/vnd.sun.xml.impress sxi -application/vnd.sun.xml.impress.template sti -application/vnd.sun.xml.math sxm -application/vnd.sun.xml.writer sxw -application/vnd.sun.xml.writer.global sxg -application/vnd.sun.xml.writer.template stw -# application/vnd.sun.wadl+xml -application/vnd.sus-calendar sus susp -application/vnd.svd svd -# application/vnd.swiftview-ics -application/vnd.symbian.install sis sisx -application/vnd.syncml+xml xsm -application/vnd.syncml.dm+wbxml bdm -application/vnd.syncml.dm+xml xdm -# application/vnd.syncml.dm.notification -# application/vnd.syncml.ds.notification -application/vnd.tao.intent-module-archive tao -application/vnd.tcpdump.pcap pcap cap dmp -application/vnd.tmobile-livetv tmo -application/vnd.trid.tpt tpt -application/vnd.triscape.mxs mxs -application/vnd.trueapp tra -# application/vnd.truedoc -# application/vnd.ubisoft.webplayer -application/vnd.ufdl ufd ufdl -application/vnd.uiq.theme utz -application/vnd.umajin umj -application/vnd.unity unityweb -application/vnd.uoml+xml uoml -# application/vnd.uplanet.alert -# application/vnd.uplanet.alert-wbxml -# application/vnd.uplanet.bearer-choice -# application/vnd.uplanet.bearer-choice-wbxml -# application/vnd.uplanet.cacheop -# application/vnd.uplanet.cacheop-wbxml -# application/vnd.uplanet.channel -# application/vnd.uplanet.channel-wbxml -# application/vnd.uplanet.list -# application/vnd.uplanet.list-wbxml -# application/vnd.uplanet.listcmd -# application/vnd.uplanet.listcmd-wbxml -# application/vnd.uplanet.signal -application/vnd.vcx vcx -# application/vnd.vd-study -# application/vnd.vectorworks -# application/vnd.verimatrix.vcas -# application/vnd.vidsoft.vidconference -application/vnd.visio vsd vst vss vsw -application/vnd.visionary vis -# application/vnd.vividence.scriptfile -application/vnd.vsf vsf -# application/vnd.wap.sic -# application/vnd.wap.slc -application/vnd.wap.wbxml wbxml -application/vnd.wap.wmlc wmlc -application/vnd.wap.wmlscriptc wmlsc -application/vnd.webturbo wtb -# application/vnd.wfa.wsc -# application/vnd.wmc -# application/vnd.wmf.bootstrap -# application/vnd.wolfram.mathematica -# application/vnd.wolfram.mathematica.package -application/vnd.wolfram.player nbp -application/vnd.wordperfect wpd -application/vnd.wqd wqd -# application/vnd.wrq-hp3000-labelled -application/vnd.wt.stf stf -# application/vnd.wv.csp+wbxml -# application/vnd.wv.csp+xml -# application/vnd.wv.ssp+xml -application/vnd.xara xar -application/vnd.xfdl xfdl -# application/vnd.xfdl.webform -# application/vnd.xmi+xml -# application/vnd.xmpie.cpkg -# application/vnd.xmpie.dpkg -# application/vnd.xmpie.plan -# application/vnd.xmpie.ppkg -# application/vnd.xmpie.xlim -application/vnd.yamaha.hv-dic hvd -application/vnd.yamaha.hv-script hvs -application/vnd.yamaha.hv-voice hvp -application/vnd.yamaha.openscoreformat osf -application/vnd.yamaha.openscoreformat.osfpvg+xml osfpvg -# application/vnd.yamaha.remote-setup -application/vnd.yamaha.smaf-audio saf -application/vnd.yamaha.smaf-phrase spf -# application/vnd.yamaha.through-ngn -# application/vnd.yamaha.tunnel-udpencap -application/vnd.yellowriver-custom-menu cmp -application/vnd.zul zir zirz -application/vnd.zzazz.deck+xml zaz -application/voicexml+xml vxml -# application/vq-rtcpxr -# application/watcherinfo+xml -# application/whoispp-query -# application/whoispp-response -application/widget wgt -application/winhlp hlp -# application/wita -# application/wordperfect5.1 -application/wsdl+xml wsdl -application/wspolicy+xml wspolicy -application/x-7z-compressed 7z -application/x-abiword abw -application/x-ace-compressed ace -application/x-authorware-bin aab x32 u32 vox -application/x-authorware-map aam -application/x-authorware-seg aas -application/x-bcpio bcpio -application/x-bittorrent torrent -application/x-bzip bz -application/x-bzip2 bz2 boz -application/x-cdlink vcd -application/x-chat chat -application/x-chess-pgn pgn -# application/x-compress -application/x-cpio cpio -application/x-csh csh -application/x-debian-package deb udeb -application/x-director dir dcr dxr cst cct cxt w3d fgd swa -application/x-doom wad -application/x-dtbncx+xml ncx -application/x-dtbook+xml dtb -application/x-dtbresource+xml res -application/x-dvi dvi -application/x-font-bdf bdf -# application/x-font-dos -# application/x-font-framemaker -application/x-font-ghostscript gsf -# application/x-font-libgrx -application/x-font-linux-psf psf -application/x-font-otf otf -application/x-font-pcf pcf -application/x-font-snf snf -# application/x-font-speedo -# application/x-font-sunos-news -application/x-font-ttf ttf ttc -application/x-font-type1 pfa pfb pfm afm -application/x-font-woff woff -# application/x-font-vfont -application/x-futuresplash spl -application/x-gnumeric gnumeric -application/x-gtar gtar -# application/x-gzip -application/x-hdf hdf -application/x-java-jnlp-file jnlp -application/x-latex latex -application/x-mobipocket-ebook prc mobi -application/x-ms-application application -application/x-ms-wmd wmd -application/x-ms-wmz wmz -application/x-ms-xbap xbap -application/x-msaccess mdb -application/x-msbinder obd -application/x-mscardfile crd -application/x-msclip clp -application/x-msdownload exe dll com bat msi -application/x-msmediaview mvb m13 m14 -application/x-msmetafile wmf -application/x-msmoney mny -application/x-mspublisher pub -application/x-msschedule scd -application/x-msterminal trm -application/x-mswrite wri -application/x-netcdf nc cdf -application/x-pkcs12 p12 pfx -application/x-pkcs7-certificates p7b spc -application/x-pkcs7-certreqresp p7r -application/x-rar-compressed rar -application/x-sh sh -application/x-shar shar -application/x-shockwave-flash swf -application/x-silverlight-app xap -application/x-stuffit sit -application/x-stuffitx sitx -application/x-sv4cpio sv4cpio -application/x-sv4crc sv4crc -application/x-tar tar -application/x-tcl tcl -application/x-tex tex -application/x-tex-tfm tfm -application/x-texinfo texinfo texi -application/x-ustar ustar -application/x-wais-source src -application/x-x509-ca-cert der crt -application/x-xfig fig -application/x-xpinstall xpi -# application/x400-bp -# application/xcap-att+xml -# application/xcap-caps+xml -application/xcap-diff+xml xdf -# application/xcap-el+xml -# application/xcap-error+xml -# application/xcap-ns+xml -# application/xcon-conference-info-diff+xml -# application/xcon-conference-info+xml -application/xenc+xml xenc -application/xhtml+xml xhtml xht -# application/xhtml-voice+xml -application/xml xml xsl -application/xml-dtd dtd -# application/xml-external-parsed-entity -# application/xmpp+xml -application/xop+xml xop -application/xslt+xml xslt -application/xspf+xml xspf -application/xv+xml mxml xhvml xvml xvm -application/yang yang -application/yin+xml yin -application/zip zip -# audio/1d-interleaved-parityfec -# audio/32kadpcm -# audio/3gpp -# audio/3gpp2 -# audio/ac3 -audio/adpcm adp -# audio/amr -# audio/amr-wb -# audio/amr-wb+ -# audio/asc -# audio/atrac-advanced-lossless -# audio/atrac-x -# audio/atrac3 -audio/basic au snd -# audio/bv16 -# audio/bv32 -# audio/clearmode -# audio/cn -# audio/dat12 -# audio/dls -# audio/dsr-es201108 -# audio/dsr-es202050 -# audio/dsr-es202211 -# audio/dsr-es202212 -# audio/dv -# audio/dvi4 -# audio/eac3 -# audio/evrc -# audio/evrc-qcp -# audio/evrc0 -# audio/evrc1 -# audio/evrcb -# audio/evrcb0 -# audio/evrcb1 -# audio/evrcwb -# audio/evrcwb0 -# audio/evrcwb1 -# audio/example -# audio/fwdred -# audio/g719 -# audio/g722 -# audio/g7221 -# audio/g723 -# audio/g726-16 -# audio/g726-24 -# audio/g726-32 -# audio/g726-40 -# audio/g728 -# audio/g729 -# audio/g7291 -# audio/g729d -# audio/g729e -# audio/gsm -# audio/gsm-efr -# audio/gsm-hr-08 -# audio/ilbc -# audio/ip-mr_v2.5 -# audio/l16 -# audio/l20 -# audio/l24 -# audio/l8 -# audio/lpc -audio/midi mid midi kar rmi -# audio/mobile-xmf -audio/mp4 mp4a -# audio/mp4a-latm -# audio/mpa -# audio/mpa-robust -audio/mpeg mpga mp2 mp2a mp3 m2a m3a -# audio/mpeg4-generic -audio/ogg oga ogg spx -# audio/parityfec -# audio/pcma -# audio/pcma-wb -# audio/pcmu-wb -# audio/pcmu -# audio/prs.sid -# audio/qcelp -# audio/red -# audio/rtp-enc-aescm128 -# audio/rtp-midi -# audio/rtx -# audio/smv -# audio/smv0 -# audio/smv-qcp -# audio/sp-midi -# audio/speex -# audio/t140c -# audio/t38 -# audio/telephone-event -# audio/tone -# audio/uemclip -# audio/ulpfec -# audio/vdvi -# audio/vmr-wb -# audio/vnd.3gpp.iufp -# audio/vnd.4sb -# audio/vnd.audiokoz -# audio/vnd.celp -# audio/vnd.cisco.nse -# audio/vnd.cmles.radio-events -# audio/vnd.cns.anp1 -# audio/vnd.cns.inf1 -audio/vnd.dece.audio uva uvva -audio/vnd.digital-winds eol -# audio/vnd.dlna.adts -# audio/vnd.dolby.heaac.1 -# audio/vnd.dolby.heaac.2 -# audio/vnd.dolby.mlp -# audio/vnd.dolby.mps -# audio/vnd.dolby.pl2 -# audio/vnd.dolby.pl2x -# audio/vnd.dolby.pl2z -# audio/vnd.dolby.pulse.1 -audio/vnd.dra dra -audio/vnd.dts dts -audio/vnd.dts.hd dtshd -# audio/vnd.dvb.file dvb -# audio/vnd.everad.plj -# audio/vnd.hns.audio -audio/vnd.lucent.voice lvp -audio/vnd.ms-playready.media.pya pya -# audio/vnd.nokia.mobile-xmf -# audio/vnd.nortel.vbk -audio/vnd.nuera.ecelp4800 ecelp4800 -audio/vnd.nuera.ecelp7470 ecelp7470 -audio/vnd.nuera.ecelp9600 ecelp9600 -# audio/vnd.octel.sbc -# audio/vnd.qcelp -# audio/vnd.rhetorex.32kadpcm -audio/vnd.rip rip -# audio/vnd.sealedmedia.softseal.mpeg -# audio/vnd.vmx.cvsd -# audio/vorbis -# audio/vorbis-config -audio/webm weba -audio/x-aac aac -audio/x-aiff aif aiff aifc -audio/x-mpegurl m3u -audio/x-ms-wax wax -audio/x-ms-wma wma -audio/x-pn-realaudio ram ra -audio/x-pn-realaudio-plugin rmp -audio/x-wav wav -chemical/x-cdx cdx -chemical/x-cif cif -chemical/x-cmdf cmdf -chemical/x-cml cml -chemical/x-csml csml -# chemical/x-pdb -chemical/x-xyz xyz -image/bmp bmp -image/cgm cgm -# image/example -# image/fits -image/g3fax g3 -image/gif gif -image/ief ief -# image/jp2 -image/jpeg jpeg jpg jpe -# image/jpm -# image/jpx -image/ktx ktx -# image/naplps -image/png png -image/prs.btif btif -# image/prs.pti -image/svg+xml svg svgz -# image/t38 -image/tiff tiff tif -# image/tiff-fx -image/vnd.adobe.photoshop psd -# image/vnd.cns.inf2 -image/vnd.dece.graphic uvi uvvi uvg uvvg -image/vnd.dvb.subtitle sub -image/vnd.djvu djvu djv -image/vnd.dwg dwg -image/vnd.dxf dxf -image/vnd.fastbidsheet fbs -image/vnd.fpx fpx -image/vnd.fst fst -image/vnd.fujixerox.edmics-mmr mmr -image/vnd.fujixerox.edmics-rlc rlc -# image/vnd.globalgraphics.pgb -# image/vnd.microsoft.icon -# image/vnd.mix -image/vnd.ms-modi mdi -image/vnd.net-fpx npx -# image/vnd.radiance -# image/vnd.sealed.png -# image/vnd.sealedmedia.softseal.gif -# image/vnd.sealedmedia.softseal.jpg -# image/vnd.svf -image/vnd.wap.wbmp wbmp -image/vnd.xiff xif -image/webp webp -image/x-cmu-raster ras -image/x-cmx cmx -image/x-freehand fh fhc fh4 fh5 fh7 -image/x-icon ico -image/x-pcx pcx -image/x-pict pic pct -image/x-portable-anymap pnm -image/x-portable-bitmap pbm -image/x-portable-graymap pgm -image/x-portable-pixmap ppm -image/x-rgb rgb -image/x-xbitmap xbm -image/x-xpixmap xpm -image/x-xwindowdump xwd -# message/cpim -# message/delivery-status -# message/disposition-notification -# message/example -# message/external-body -# message/feedback-report -# message/global -# message/global-delivery-status -# message/global-disposition-notification -# message/global-headers -# message/http -# message/imdn+xml -# message/news -# message/partial -message/rfc822 eml mime -# message/s-http -# message/sip -# message/sipfrag -# message/tracking-status -# message/vnd.si.simp -# model/example -model/iges igs iges -model/mesh msh mesh silo -model/vnd.collada+xml dae -model/vnd.dwf dwf -# model/vnd.flatland.3dml -model/vnd.gdl gdl -# model/vnd.gs-gdl -# model/vnd.gs.gdl -model/vnd.gtw gtw -# model/vnd.moml+xml -model/vnd.mts mts -# model/vnd.parasolid.transmit.binary -# model/vnd.parasolid.transmit.text -model/vnd.vtu vtu -model/vrml wrl vrml -# multipart/alternative -# multipart/appledouble -# multipart/byteranges -# multipart/digest -# multipart/encrypted -# multipart/example -# multipart/form-data -# multipart/header-set -# multipart/mixed -# multipart/parallel -# multipart/related -# multipart/report -# multipart/signed -# multipart/voice-message -# text/1d-interleaved-parityfec -text/calendar ics ifb -text/css css -text/csv csv -# text/directory -# text/dns -# text/ecmascript -# text/enriched -# text/example -# text/fwdred -text/html html htm -# text/javascript -text/n3 n3 -# text/parityfec -text/plain txt text conf def list log in -# text/prs.fallenstein.rst -text/prs.lines.tag dsc -# text/vnd.radisys.msml-basic-layout -# text/red -# text/rfc822-headers -text/richtext rtx -# text/rtf -# text/rtp-enc-aescm128 -# text/rtx -text/sgml sgml sgm -# text/t140 -text/tab-separated-values tsv -text/troff t tr roff man me ms -text/turtle ttl -# text/ulpfec -text/uri-list uri uris urls -text/vcard vcard -# text/vnd.abc -text/vnd.curl curl -text/vnd.curl.dcurl dcurl -text/vnd.curl.scurl scurl -text/vnd.curl.mcurl mcurl -# text/vnd.dmclientscript -text/vnd.dvb.subtitle sub -# text/vnd.esmertec.theme-descriptor -text/vnd.fly fly -text/vnd.fmi.flexstor flx -text/vnd.graphviz gv -text/vnd.in3d.3dml 3dml -text/vnd.in3d.spot spot -# text/vnd.iptc.newsml -# text/vnd.iptc.nitf -# text/vnd.latex-z -# text/vnd.motorola.reflex -# text/vnd.ms-mediapackage -# text/vnd.net2phone.commcenter.command -# text/vnd.si.uricatalogue -text/vnd.sun.j2me.app-descriptor jad -# text/vnd.trolltech.linguist -# text/vnd.wap.si -# text/vnd.wap.sl -text/vnd.wap.wml wml -text/vnd.wap.wmlscript wmls -text/x-asm s asm -text/x-c c cc cxx cpp h hh dic -text/x-fortran f for f77 f90 -text/x-pascal p pas -text/x-java-source java -text/x-setext etx -text/x-uuencode uu -text/x-vcalendar vcs -text/x-vcard vcf -# text/xml -# text/xml-external-parsed-entity -# video/1d-interleaved-parityfec -video/3gpp 3gp -# video/3gpp-tt -video/3gpp2 3g2 -# video/bmpeg -# video/bt656 -# video/celb -# video/dv -# video/example -video/h261 h261 -video/h263 h263 -# video/h263-1998 -# video/h263-2000 -video/h264 h264 -# video/h264-rcdo -# video/h264-svc -video/jpeg jpgv -# video/jpeg2000 -video/jpm jpm jpgm -video/mj2 mj2 mjp2 -# video/mp1s -# video/mp2p -# video/mp2t -video/mp4 mp4 mp4v mpg4 -# video/mp4v-es -video/mpeg mpeg mpg mpe m1v m2v -# video/mpeg4-generic -# video/mpv -# video/nv -video/ogg ogv -# video/parityfec -# video/pointer -video/quicktime qt mov -# video/raw -# video/rtp-enc-aescm128 -# video/rtx -# video/smpte292m -# video/ulpfec -# video/vc1 -# video/vnd.cctv -video/vnd.dece.hd uvh uvvh -video/vnd.dece.mobile uvm uvvm -# video/vnd.dece.mp4 -video/vnd.dece.pd uvp uvvp -video/vnd.dece.sd uvs uvvs -video/vnd.dece.video uvv uvvv -# video/vnd.directv.mpeg -# video/vnd.directv.mpeg-tts -# video/vnd.dlna.mpeg-tts -video/vnd.dvb.file dvb -video/vnd.fvt fvt -# video/vnd.hns.video -# video/vnd.iptvforum.1dparityfec-1010 -# video/vnd.iptvforum.1dparityfec-2005 -# video/vnd.iptvforum.2dparityfec-1010 -# video/vnd.iptvforum.2dparityfec-2005 -# video/vnd.iptvforum.ttsavc -# video/vnd.iptvforum.ttsmpeg2 -# video/vnd.motorola.video -# video/vnd.motorola.videop -video/vnd.mpegurl mxu m4u -video/vnd.ms-playready.media.pyv pyv -# video/vnd.nokia.interleaved-multimedia -# video/vnd.nokia.videovoip -# video/vnd.objectvideo -# video/vnd.sealed.mpeg1 -# video/vnd.sealed.mpeg4 -# video/vnd.sealed.swf -# video/vnd.sealedmedia.softseal.mov -video/vnd.uvvu.mp4 uvu uvvu -video/vnd.vivo viv -video/webm webm -video/x-f4v f4v -video/x-fli fli -video/x-flv flv -video/x-m4v m4v -video/x-ms-asf asf asx -video/x-ms-wm wm -video/x-ms-wmv wmv -video/x-ms-wmx wmx -video/x-ms-wvx wvx -video/x-msvideo avi -video/x-sgi-movie movie -x-conference/x-cooltalk ice diff --git a/node_modules/express/node_modules/send/node_modules/mime/types/node.types b/node_modules/express/node_modules/send/node_modules/mime/types/node.types deleted file mode 100644 index b7fe8c0..0000000 --- a/node_modules/express/node_modules/send/node_modules/mime/types/node.types +++ /dev/null @@ -1,65 +0,0 @@ -# What: Google Chrome Extension -# Why: To allow apps to (work) be served with the right content type header. -# http://codereview.chromium.org/2830017 -# Added by: niftylettuce -application/x-chrome-extension crx - -# What: OTF Message Silencer -# Why: To silence the "Resource interpreted as font but transferred with MIME -# type font/otf" message that occurs in Google Chrome -# Added by: niftylettuce -font/opentype otf - -# What: HTC support -# Why: To properly render .htc files such as CSS3PIE -# Added by: niftylettuce -text/x-component htc - -# What: HTML5 application cache manifest -# Why: De-facto standard. Required by Mozilla browser when serving HTML5 apps -# per https://developer.mozilla.org/en/offline_resources_in_firefox -# Added by: louisremi -text/cache-manifest appcache manifest - -# What: node binary buffer format -# Why: semi-standard extension w/in the node community -# Added by: tootallnate -application/octet-stream buffer - -# What: The "protected" MP-4 formats used by iTunes. -# Why: Required for streaming music to browsers (?) -# Added by: broofa -application/mp4 m4p -audio/mp4 m4a - -# What: Music playlist format (http://en.wikipedia.org/wiki/M3U) -# Why: See https://github.com/bentomas/node-mime/pull/6 -# Added by: mjrusso -application/x-mpegURL m3u8 - -# What: Video format, Part of RFC1890 -# Why: See https://github.com/bentomas/node-mime/pull/6 -# Added by: mjrusso -video/MP2T ts - -# What: The FLAC lossless codec format -# Why: Streaming and serving FLAC audio -# Added by: jacobrask -audio/flac flac - -# What: EventSource mime type -# Why: mime type of Server-Sent Events stream -# http://www.w3.org/TR/eventsource/#text-event-stream -# Added by: francois2metz -text/event-stream event-stream - -# What: Mozilla App manifest mime type -# Why: https://developer.mozilla.org/en/Apps/Manifest#Serving_manifests -# Added by: ednapiranha -application/x-web-app-manifest+json webapp - -# What: Matroska Mime Types -# Why: http://en.wikipedia.org/wiki/Matroska -# Added by: aduncan88 -video/x-matroska mkv -audio/x-matroska mka diff --git a/node_modules/express/node_modules/send/package.json b/node_modules/express/node_modules/send/package.json deleted file mode 100644 index 5e936f1..0000000 --- a/node_modules/express/node_modules/send/package.json +++ /dev/null @@ -1,34 +0,0 @@ -{ - "name": "send", - "version": "0.1.0", - "description": "Better streaming static file server with Range and conditional-GET support", - "keywords": [ - "static", - "file", - "server" - ], - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca" - }, - "dependencies": { - "debug": "*", - "mime": "1.2.6", - "fresh": "0.1.0", - "range-parser": "0.0.4" - }, - "devDependencies": { - "mocha": "*", - "should": "*", - "supertest": "0.0.1", - "connect": "2.x" - }, - "scripts": { - "test": "make test" - }, - "main": "index", - "readme": "\n# send\n\n Send is Connect's `static()` extracted for generalized use, a streaming static file\n server supporting partial responses (Ranges), conditional-GET negotiation, high test coverage, and granular events which may be leveraged to take appropriate actions in your application or framework.\n\n## Installation\n\n $ npm install send\n\n## Examples\n\n Small:\n\n```js\nvar http = require('http');\nvar send = require('send');\n\nvar app = http.createServer(function(req, res){\n send(req, req.url).pipe(res);\n});\n```\n\n Serving from a root directory with custom error-handling:\n\n```js\nvar http = require('http');\nvar send = require('send');\n\nvar app = http.createServer(function(req, res){\n // your custom error-handling logic:\n function error(err) {\n res.statusCode = err.status || 500;\n res.end(err.message);\n }\n\n // your custom directory handling logic:\n function redirect() {\n res.statusCode = 301;\n res.setHeader('Location', req.url + '/');\n res.end('Redirecting to ' + req.url + '/');\n }\n\n // transfer arbitrary files from within\n // /www/example.com/public/*\n send(req, url.parse(req.url).pathname)\n .root('/www/example.com/public')\n .on('error', error)\n .on('directory', redirect)\n .pipe(res);\n});\n```\n\n## API\n\n### Events\n\n - `error` an error occurred `(err)`\n - `directory` a directory was requested\n - `stream` file streaming has started `(stream)`\n - `end` streaming has completed\n\n### .root(dir)\n\n Serve files relative to `path`. Aliased as `.from(dir)`.\n\n### .index(path)\n\n By default send supports \"index.html\" files, to disable this\n invoke `.index(false)` or to supply a new index pass a string.\n\n### .maxage(ms)\n\n Provide a max-age in milliseconds for http caching, defaults to 0.\n\n## Error-handling\n\n By default when no `error` listeners are present an automatic response will be made, otherwise you have full control over the response, aka you may show a 5xx page etc.\n\n## Caching\n\n It does _not_ perform internal caching, you should use a reverse proxy cache such\n as Varnish for this, or those fancy things called CDNs. If your application is small enough that it would benefit from single-node memory caching, it's small enough that it does not need caching at all ;).\n\n## Debugging\n\n To enable `debug()` instrumentation output export __DEBUG__:\n\n```\n$ DEBUG=send node app\n```\n\n## Running tests\n\n```\n$ npm install\n$ make test\n```\n\n## License \n\n(The MIT License)\n\nCopyright (c) 2012 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.", - "readmeFilename": "Readme.md", - "_id": "send@0.1.0", - "_from": "send@0.1.0" -} diff --git a/node_modules/express/package.json b/node_modules/express/package.json deleted file mode 100644 index fb41815..0000000 --- a/node_modules/express/package.json +++ /dev/null @@ -1,84 +0,0 @@ -{ - "name": "express", - "description": "Sinatra inspired web development framework", - "version": "3.1.2", - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca" - }, - "contributors": [ - { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca" - }, - { - "name": "Aaron Heckmann", - "email": "aaron.heckmann+github@gmail.com" - }, - { - "name": "Ciaran Jessup", - "email": "ciaranj@gmail.com" - }, - { - "name": "Guillermo Rauch", - "email": "rauchg@gmail.com" - } - ], - "dependencies": { - "connect": "2.7.5", - "commander": "0.6.1", - "range-parser": "0.0.4", - "mkdirp": "~0.3.4", - "cookie": "0.0.5", - "buffer-crc32": "~0.2.1", - "fresh": "0.1.0", - "methods": "0.0.1", - "send": "0.1.0", - "cookie-signature": "1.0.0", - "debug": "*" - }, - "devDependencies": { - "ejs": "*", - "mocha": "*", - "jade": "*", - "hjs": "*", - "stylus": "*", - "should": "*", - "connect-redis": "*", - "github-flavored-markdown": "*", - "supertest": "0.0.1" - }, - "keywords": [ - "express", - "framework", - "sinatra", - "web", - "rest", - "restful", - "router", - "app", - "api" - ], - "repository": { - "type": "git", - "url": "git://github.com/visionmedia/express" - }, - "main": "index", - "bin": { - "express": "./bin/express" - }, - "scripts": { - "prepublish": "npm prune", - "test": "make test" - }, - "engines": { - "node": "*" - }, - "readme": "![express logo](http://f.cl.ly/items/0V2S1n0K1i3y1c122g04/Screen%20Shot%202012-04-11%20at%209.59.42%20AM.png)\n\n Fast, unopinionated, minimalist web framework for [node](http://nodejs.org). [![Build Status](https://secure.travis-ci.org/visionmedia/express.png)](http://travis-ci.org/visionmedia/express) [![Dependency Status](https://gemnasium.com/visionmedia/express.png)](https://gemnasium.com/visionmedia/express)\n\n```js\nvar express = require('express');\nvar app = express();\n\napp.get('/', function(req, res){\n res.send('Hello World');\n});\n\napp.listen(3000);\n```\n\n## Installation\n\n $ npm install -g express\n\n## Quick Start\n\n The quickest way to get started with express is to utilize the executable `express(1)` to generate an application as shown below:\n\n Create the app:\n\n $ npm install -g express\n $ express /tmp/foo && cd /tmp/foo\n\n Install dependencies:\n\n $ npm install\n\n Start the server:\n\n $ node app\n\n## Features\n\n * Built on [Connect](http://github.com/senchalabs/connect)\n * Robust routing\n * HTTP helpers (redirection, caching, etc)\n * View system supporting 14+ template engines\n * Content negotiation\n * Focus on high performance\n * Environment based configuration\n * Executable for generating applications quickly\n * High test coverage\n\n## Philosophy\n\n The Express philosophy is to provide small, robust tooling for HTTP servers. Making\n it a great solution for single page applications, web sites, hybrids, or public\n HTTP APIs.\n\n Built on Connect you can use _only_ what you need, and nothing more, applications\n can be as big or as small as you like, even a single file. Express does\n not force you to use any specific ORM or template engine. With support for over\n 14 template engines via [Consolidate.js](http://github.com/visionmedia/consolidate.js)\n you can quickly craft your perfect framework.\n\n## More Information\n\n * Join #express on freenode\n * [Google Group](http://groups.google.com/group/express-js) for discussion\n * Follow [tjholowaychuk](http://twitter.com/tjholowaychuk) on twitter for updates\n * Visit the [Wiki](http://github.com/visionmedia/express/wiki)\n * [日本語ドキュメンテーション](http://hideyukisaito.com/doc/expressjs/) by [hideyukisaito](https://github.com/hideyukisaito)\n * [Русскоязычная документация](http://jsman.ru/express/)\n * Run express examples [online](https://runnable.com/express)\n\n## Viewing Examples\n\nClone the Express repo, then install the dev dependencies to install all the example / test suite deps:\n\n $ git clone git://github.com/visionmedia/express.git --depth 1\n $ cd express\n $ npm install\n\nthen run whichever tests you want:\n\n $ node examples/content-negotiation\n\n## Running Tests\n\nTo run the test suite first invoke the following command within the repo, installing the development dependencies:\n\n $ npm install\n\nthen run the tests:\n\n $ make test\n\n## Contributors\n\n```\nproject: express\ncommits: 3559\nactive : 468 days\nfiles : 237\nauthors:\n 1891\tTj Holowaychuk 53.1%\n 1285\tvisionmedia 36.1%\n 182\tTJ Holowaychuk 5.1%\n 54\tAaron Heckmann 1.5%\n 34\tcsausdev 1.0%\n 26\tciaranj 0.7%\n 21\tRobert Sköld 0.6%\n 6\tGuillermo Rauch 0.2%\n 3\tDav Glass 0.1%\n 3\tNick Poulden 0.1%\n 2\tRandy Merrill 0.1%\n 2\tBenny Wong 0.1%\n 2\tHunter Loftis 0.1%\n 2\tJake Gordon 0.1%\n 2\tBrian McKinney 0.1%\n 2\tRoman Shtylman 0.1%\n 2\tBen Weaver 0.1%\n 2\tDave Hoover 0.1%\n 2\tEivind Fjeldstad 0.1%\n 2\tDaniel Shaw 0.1%\n 1\tMatt Colyer 0.0%\n 1\tPau Ramon 0.0%\n 1\tPero Pejovic 0.0%\n 1\tPeter Rekdal Sunde 0.0%\n 1\tRaynos 0.0%\n 1\tTeng Siong Ong 0.0%\n 1\tViktor Kelemen 0.0%\n 1\tctide 0.0%\n 1\t8bitDesigner 0.0%\n 1\tisaacs 0.0%\n 1\tmgutz 0.0%\n 1\tpikeas 0.0%\n 1\tshuwatto 0.0%\n 1\ttstrimple 0.0%\n 1\tewoudj 0.0%\n 1\tAdam Sanderson 0.0%\n 1\tAndrii Kostenko 0.0%\n 1\tAndy Hiew 0.0%\n 1\tArpad Borsos 0.0%\n 1\tAshwin Purohit 0.0%\n 1\tBenjen 0.0%\n 1\tDarren Torpey 0.0%\n 1\tGreg Ritter 0.0%\n 1\tGregory Ritter 0.0%\n 1\tJames Herdman 0.0%\n 1\tJim Snodgrass 0.0%\n 1\tJoe McCann 0.0%\n 1\tJonathan Dumaine 0.0%\n 1\tJonathan Palardy 0.0%\n 1\tJonathan Zacsh 0.0%\n 1\tJustin Lilly 0.0%\n 1\tKen Sato 0.0%\n 1\tMaciej Małecki 0.0%\n 1\tMasahiro Hayashi 0.0%\n```\n\n## License\n\n(The MIT License)\n\nCopyright (c) 2009-2012 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n", - "readmeFilename": "Readme.md", - "bugs": { - "url": "https://github.com/visionmedia/express/issues" - }, - "_id": "express@3.1.2", - "_from": "express@~3.1.0" -} diff --git a/node_modules/express/test.js b/node_modules/express/test.js deleted file mode 100644 index 1ff7958..0000000 --- a/node_modules/express/test.js +++ /dev/null @@ -1,39 +0,0 @@ - -/** - * Module dependencies. - */ - -var express = require('./') - , app = express() - -var users = ['foo', 'bar', 'baz']; - -app.use(express.bodyParser()); -console.log(app.locals); - -app.get('/api/users', function(req, res){ - res.send(users); -}); - -app.del('/api/users', function(req, res){ - users = []; - res.send(200); -}); - -app.post('/api/users', function(req, res){ - users.push(req.body.name); - res.send(201); -}); - -app.get('/api/user/:id', function(req, res){ - var id = req.params.id; - res.send(users[id]); -}); - -app.use('/api', function(req, res, next){ - var err = new Error('Method Not Allowed'); - err.status = 405; -}); - -app.listen(5555); -console.log('listening on 5555'); diff --git a/node_modules/iced-coffee-script/.npmignore b/node_modules/iced-coffee-script/.npmignore deleted file mode 100644 index 21e430d..0000000 --- a/node_modules/iced-coffee-script/.npmignore +++ /dev/null @@ -1,11 +0,0 @@ -*.coffee -*.html -.DS_Store -.git* -Cakefile -documentation/ -examples/ -extras/coffee-script.js -raw/ -src/ -test/ diff --git a/node_modules/iced-coffee-script/CNAME b/node_modules/iced-coffee-script/CNAME deleted file mode 100644 index 9abc2e3..0000000 --- a/node_modules/iced-coffee-script/CNAME +++ /dev/null @@ -1 +0,0 @@ -icedcoffeescript.org diff --git a/node_modules/iced-coffee-script/CONTRIBUTING.md b/node_modules/iced-coffee-script/CONTRIBUTING.md deleted file mode 100644 index 6390c68..0000000 --- a/node_modules/iced-coffee-script/CONTRIBUTING.md +++ /dev/null @@ -1,9 +0,0 @@ -## How to contribute to CoffeeScript - -* Before you open a ticket or send a pull request, [search](https://github.com/jashkenas/coffee-script/issues) for previous discussions about the same feature or issue. Add to the earlier ticket if you find one. - -* Before sending a pull request for a feature, be sure to have [tests](https://github.com/jashkenas/coffee-script/tree/master/test). - -* Use the same coding style as the rest of the [codebase](https://github.com/jashkenas/coffee-script/tree/master/src). If you're just getting started with CoffeeScript, there's a nice [style guide](https://github.com/polarmobile/coffeescript-style-guide). - -* In your pull request, do not add documentation to `index.html` or re-build the minified `coffee-script.js` file. We'll do those things before cutting a new release. \ No newline at end of file diff --git a/node_modules/iced-coffee-script/LICENSE b/node_modules/iced-coffee-script/LICENSE deleted file mode 100644 index a396eae..0000000 --- a/node_modules/iced-coffee-script/LICENSE +++ /dev/null @@ -1,22 +0,0 @@ -Copyright (c) 2009-2013 Jeremy Ashkenas - -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/node_modules/iced-coffee-script/README b/node_modules/iced-coffee-script/README deleted file mode 100644 index bf4c284..0000000 --- a/node_modules/iced-coffee-script/README +++ /dev/null @@ -1,56 +0,0 @@ - - - ICED - - _____ __ __ - / ____| / _|/ _| - .- ----------- -. | | ___ | |_| |_ ___ ___ - ( (ice cubes) ) | | / _ \| _| _/ _ \/ _ \ - |`-..________ ..-'| | |___| (_) | | | || __/ __/ - | | \_____\___/|_| |_| \___|\___| - | ;--. - | (__ \ _____ _ _ - | | ) ) / ____| (_) | | - | |/ / | (___ ___ _ __ _ _ __ | |_ - | ( / \___ \ / __| '__| | '_ \| __| - | |/ ____) | (__| | | | |_) | |_ - | | |_____/ \___|_| |_| .__/ \__| - `-.._________..-' | | - |_| - - - CoffeeScript is a little language that compiles into JavaScript. - IcedCoffeeScript is a superset of CoffeeScript that adds two new - keywords: await and defer. - - Install Node.js, and then the CoffeeScript compiler: - sudo bin/cake install - - Or, if you have the Node Package Manager installed: - npm install -g iced-coffee-script - (Leave off the -g if you don't wish to install globally.) - - Execute a script: - iced /path/to/script.coffee - - Compile a script: - iced -c /path/to/script.coffee - - For documentation, usage, and examples, see: - http://maxtaco.github.com/coffee-script - - For iced-specific technical documentation, see: - https://github.com/maxtaco/coffee-script/blob/iced/iced.md - - To suggest a feature, report a bug, or general discussion: - https://github.com/maxtaco/coffee-script/issues/ - - DM or tweet at me with questions: @maxtaco - - Or better yet, tweet about how much you love IcedCoffeeScript. - - The source repository: - git://github.com/maxtaco/coffee-script.git - - All contributors are listed here: - http://github.com/maxtaco/coffee-script/contributors diff --git a/node_modules/iced-coffee-script/Rakefile b/node_modules/iced-coffee-script/Rakefile deleted file mode 100644 index d90cce3..0000000 --- a/node_modules/iced-coffee-script/Rakefile +++ /dev/null @@ -1,79 +0,0 @@ -require 'rubygems' -require 'erb' -require 'fileutils' -require 'rake/testtask' -require 'json' - -desc "Build the documentation page" -task :doc do - source = 'documentation/index.html.erb' - child = fork { exec "bin/coffee -bcw -o documentation/js documentation/coffee/*.coffee" } - at_exit { Process.kill("INT", child) } - Signal.trap("INT") { exit } - loop do - mtime = File.stat(source).mtime - if !@mtime || mtime > @mtime - rendered = ERB.new(File.read(source)).result(binding) - File.open('index.html', 'w+') {|f| f.write(rendered) } - end - @mtime = mtime - sleep 1 - end -end - -desc "Build coffee-script-source gem" -task :gem do - require 'rubygems' - require 'rubygems/package' - - gemspec = Gem::Specification.new do |s| - s.name = 'coffee-script-source' - s.version = JSON.parse(File.read('package.json'))["version"] - s.date = Time.now.strftime("%Y-%m-%d") - - s.homepage = "http://jashkenas.github.com/coffee-script/" - s.summary = "The CoffeeScript Compiler" - s.description = <<-EOS - CoffeeScript is a little language that compiles into JavaScript. - Underneath all of those embarrassing braces and semicolons, - JavaScript has always had a gorgeous object model at its heart. - CoffeeScript is an attempt to expose the good parts of JavaScript - in a simple way. - EOS - - s.files = [ - 'lib/coffee_script/coffee-script.js', - 'lib/coffee_script/source.rb' - ] - - s.authors = ['Jeremy Ashkenas'] - s.email = 'jashkenas@gmail.com' - s.rubyforge_project = 'coffee-script-source' - s.license = "MIT" - end - - file = File.open("coffee-script-source.gem", "w") - Gem::Package.open(file, 'w') do |pkg| - pkg.metadata = gemspec.to_yaml - - path = "lib/coffee_script/source.rb" - contents = <<-ERUBY -module CoffeeScript - module Source - def self.bundled_path - File.expand_path("../coffee-script.js", __FILE__) - end - end -end - ERUBY - pkg.add_file_simple(path, 0644, contents.size) do |tar_io| - tar_io.write(contents) - end - - contents = File.read("extras/coffee-script.js") - path = "lib/coffee_script/coffee-script.js" - pkg.add_file_simple(path, 0644, contents.size) do |tar_io| - tar_io.write(contents) - end - end -end diff --git a/node_modules/iced-coffee-script/UPDATING.md b/node_modules/iced-coffee-script/UPDATING.md deleted file mode 100644 index 1d12153..0000000 --- a/node_modules/iced-coffee-script/UPDATING.md +++ /dev/null @@ -1,167 +0,0 @@ -# How to Keep IcedCoffeeScript Up-to-Date with Mainline CoffeeScript - -## Source - -### Current Method: Fork/Merge - -Here is the current system: - -1. Add a remote upstream repo to pull in the mainline changes: -```sh -git remote add upstream git@github.com:jashkenas/coffee-script -``` - -1. Pull from the upstream into our master branch; push to our own origin/master while we are at it. -```sh -git checkout master -git pull upstream master -git push origin master -``` - -1. Make sure the local `iced2` branch is up-to-date: -```sh -git checkout iced2 -git pull origin iced2 -``` - -1. Then do the merge: -```sh -git merge master -``` - -1. Then, once the rebase has succeeded: - a. Update the version number of `package.json` - a. Update the version number in `src/coffee-script.coffee` - -1. Then build a million different times: -```sh -./bin/cake build -./bin/cake build:parser -./bin/cake build -./bin/cake build -./bin/cake build:browser -./bin/cake test -``` - -1. You're good to push if it all looks good. First commit all the new changes post-rebase with: -```sh -git commit -a -``` - -1. Then do a push to the *iced2* branch. -```sh -git push origin iced2 -``` - -1. Make and push a new tag (supplying whatever your new version is): -``` -git tag -a -m vbump 1.6.2c -git push --tags -``` - -1. Finally, publish to npm. It's usually worth cloning out a fresh -copy from github, and then running: -```sh -npm publish -``` - -## Documentation - -The documentation system is totally separate. Here is the general idea: - -1. Make sure the *iced* branch is up-to-date: -```sh -git checkout iced2 -git pull origin iced2 -``` - -1. Checkout and update the *gh-pages* branch: -```sh -git checkout gh-pages -git pull origin gh-pages -``` - -1. Then do the merge: -```sh -git merge iced2 -``` - -1. This will destroy you with conflicts, but most of them can be worked -through. Basically, you want to call `theirs`, as below, on everything -*but* the `documentation/index.html.erb` file. - -1. Edit `documentation/index.html.erb` by hand, changing the current version -number to whatever it is nowadays. - -1. Rebuild the documentation like `index.html` and the -embedded examples. -```sh -rake doc -``` - -1. Run `docco` on the source files: -```sh -icake doc:source -``` - -1. Commit everything: -```sh -git commit -a -``` - -1. And push -```sh -git push origin gh-pages -``` - -1. Test that the `run` button still works on the front page, and that the -sandbox is still operational. - -## How Do I Build Docs for the first time on a machine? - -You are **fucked**! But here is an attempt to guide you: - -1. First of all, there's the original reference from CoffeeScript, which isn't great, but it's worth a look. Find it [here](https://github.com/jashkenas/coffee-script/wiki/%5BHowto%5D-Hacking-on-the-CoffeeScript-Compiler) - -1. Boiling it down, here's how I just did it on *Linux*. Note that due to -ruby, it's going to be a bit different on MacOS X: - -```sh -npm install -g docco # jashkenas write docco as a node module, this is easy enough -sudo apt-get install libonig2 # the ruby stuff needs this as an external lib dependency -sudo gem install ultraviolet # this installs some dependencies, etc -icake build:ultraviolet # this is likely to fail due to hardcoded paths and hacks -rake doc # give it a shot, and now you're back to where we were above -``` - -There are several hard parts about dealing with this setup. The first is -generating the appropriate `coffeescript.yaml` file from Jeremy's TexMate -bundle. That's failed on me before and plus we want to make a bunch of ICS -additions for `await` and `defer`. What I have in the `Cakefile` is my best -attempt to automate it, but it's likely to break. - -The bigger problem is figuring out where to dump `coffeescript.yaml` (or -better yet the `ics.yaml` that I make from it) when it's ready to go. This -moves around depending on the latest shenanigans in the `ultraviolet` library. -The hardcoded paths in the `Cakefile` is what worked on 8 Sep 2013, but who -knows for the future. The upshot is that these syntax files now reside in the -`textpow` library and not the `ultraviolet` library as they used to. And they -have new names --- `source.coffeescript.syntax`, for example. Good luck! - -#### To Accept Their Changes - -I wrote a little shell script called `theirs`: - -```bash -#!/bin/sh - -git checkout --theirs $* -git add $* -``` - -I run this on any *autogenerated* file that has a -conflict. For instance, those files in `lib/coffee-script/` -or `documentation/js`. For conflicts in `package.json`, `Cakefile` -or `src/`, you probably need to do a hand-merge. - - diff --git a/node_modules/iced-coffee-script/bin/cake b/node_modules/iced-coffee-script/bin/cake deleted file mode 100755 index 5965f4e..0000000 --- a/node_modules/iced-coffee-script/bin/cake +++ /dev/null @@ -1,7 +0,0 @@ -#!/usr/bin/env node - -var path = require('path'); -var fs = require('fs'); -var lib = path.join(path.dirname(fs.realpathSync(__filename)), '../lib'); - -require(lib + '/coffee-script/cake').run(); diff --git a/node_modules/iced-coffee-script/bin/coffee b/node_modules/iced-coffee-script/bin/coffee deleted file mode 100755 index 3d1d71c..0000000 --- a/node_modules/iced-coffee-script/bin/coffee +++ /dev/null @@ -1,7 +0,0 @@ -#!/usr/bin/env node - -var path = require('path'); -var fs = require('fs'); -var lib = path.join(path.dirname(fs.realpathSync(__filename)), '../lib'); - -require(lib + '/coffee-script/command').run(); diff --git a/node_modules/iced-coffee-script/extras/coffee-script-iced-large.js b/node_modules/iced-coffee-script/extras/coffee-script-iced-large.js deleted file mode 100644 index 972bacb..0000000 --- a/node_modules/iced-coffee-script/extras/coffee-script-iced-large.js +++ /dev/null @@ -1,8 +0,0 @@ -/** - * IcedCoffeeScript Compiler v1.6.3-g - * http://iced-coffee-script.github.io/iced-coffee-script - * - * Copyright 2011, Jeremy Ashkenas, Maxwell Krohn - * Released under the MIT License - */ -!function(n){var e=function(){function n(e){return n[e]}return n["./iced"]=function(){var n={},e={exports:n};return function(){var e,t=[].slice;n.generator=e=function(n,e,r){var i,u,o,f,c,l;return e.transform=function(n,e){return n.icedTransform(e)},e["const"]=i={k:"__iced_k",k_noop:"__iced_k_noop",param:"__iced_p_",ns:"iced",runtime:"runtime",Deferrals:"Deferrals",deferrals:"__iced_deferrals",fulfill:"_fulfill",b_while:"_break",t_while:"_while",c_while:"_continue",n_while:"_next",n_arg:"__iced_next_arg",context:"context",defer_method:"defer",slot:"__slot",assign_fn:"assign_fn",autocb:"autocb",retslot:"ret",trace:"__iced_trace",passed_deferral:"__iced_passed_deferral",findDeferral:"findDeferral",lineno:"lineno",parent:"parent",filename:"filename",funcname:"funcname",catchExceptions:"catchExceptions",runtime_modes:["node","inline","window","none","browserify"],trampoline:"trampoline"},n.makeDeferReturn=function(e,r,u,o,f){var c,l,a,s;a={};for(c in o)s=o[c],a[c]=s;return a[i.lineno]=null!=r?r[i.lineno]:void 0,l=function(){var i,o,c;return i=1<=arguments.length?t.call(arguments,0):[],null!=r&&null!=(c=r.assign_fn)&&c.apply(null,i),e?(o=e,f||(e=null),o._fulfill(u,a)):n._warn("overused deferral at "+n._trace_to_string(a))},l[i.trace]=a,l},n.__c=0,n.tickCounter=function(e){return n.__c++,0===n.__c%e?(n.__c=0,!0):!1},n.__active_trace=null,n._trace_to_string=function(n){var e;return e=n[i.funcname]||"",""+e+" ("+n[i.filename]+":"+(n[i.lineno]+1)+")"},n._warn=function(n){return"undefined"!=typeof console&&null!==console?console.log("ICED warning: "+n):void 0},r.trampoline=function(e){return n.tickCounter(500)?"undefined"!=typeof process&&null!==process?process.nextTick(e):setTimeout(e):e()},r.Deferrals=u=function(){function e(n,e){this.trace=e,this.continuation=n,this.count=1,this.ret=null}return e.prototype._call=function(e){var t;return this.continuation?(n.__active_trace=e,t=this.continuation,this.continuation=null,t(this.ret)):n._warn("Entered dead await at "+n._trace_to_string(e))},e.prototype._fulfill=function(n,e){var t=this;return--this.count>0?void 0:r.trampoline(function(){return t._call(e)})},e.prototype.defer=function(e){var t;return this.count++,t=this,n.makeDeferReturn(t,e,null,this.trace)},e.prototype._defer=function(n){return this.defer(n)},e}(),r.findDeferral=c=function(n){var e,t,r;for(t=0,r=n.length;r>t;t++)if(e=n[t],null!=e?e[i.trace]:void 0)return e;return null},r.Rendezvous=o=function(){function e(){this.completed=[],this.waiters=[],this.defer_id=0}var t;return t=function(){function n(n,e,t){this.rv=n,this.id=e,this.multi=t}return n.prototype.defer=function(n){return this.rv._deferWithId(this.id,n,this.multi)},n}(),e.prototype.wait=function(n){var e;return this.completed.length?(e=this.completed.shift(),n(e)):this.waiters.push(n)},e.prototype.defer=function(n){var e;return e=this.defer_id++,this.deferWithId(e,n)},e.prototype.id=function(n,e){return null==e&&(e=!1),new t(this,n,e)},e.prototype._fulfill=function(n){var e;return this.waiters.length?(e=this.waiters.shift(),e(n)):this.completed.push(n)},e.prototype._deferWithId=function(e,t,r){return this.count++,n.makeDeferReturn(this,t,e,{},r)},e}(),r.stackWalk=l=function(e){var t,r,u,o;for(r=[],u=e?e[i.trace]:n.__active_trace;u;)t=" at "+n._trace_to_string(u),r.push(t),u=null!=u?null!=(o=u[i.parent])?o[i.trace]:void 0:void 0;return r},r.exceptionHandler=f=function(n,e){var t;return e||(e=console.log),e(n.stack),t=l(),t.length?(e("Iced callback trace:"),e(t.join("\n"))):void 0},r.catchExceptions=function(n){return"undefined"!=typeof process&&null!==process?process.on("uncaughtException",function(e){return f(e,n),process.exit(1)}):void 0}},n.runtime={},e(this,n,n.runtime)}.call(this),e.exports}(),n["./icedlib"]=function(){var e={},t={exports:e};return function(){var t,r,i,u,o,f,c,l,a=[].slice;u=o=function(){},i=n("./iced"),e.iced=r=i.runtime,l=function(n,e,t,i){var u,f,c,l,s,d;d=o,l=r.findDeferral(arguments),f=new r.Rendezvous,i[0]=f.id(!0).defer({assign_fn:function(){return function(){return u=a.call(arguments,0)}}(),lineno:17,context:s}),setTimeout(f.id(!1).defer({lineno:18,context:s}),e),function(n){s=new r.Deferrals(n,{parent:l,filename:"src/icedlib.coffee",funcname:"_timeout"}),f.wait(s.defer({assign_fn:function(){return function(){return c=arguments[0]}}(),lineno:19})),s._fulfill()}(function(){return t&&(t[0]=c),n.apply(null,u)})},e.timeout=function(n,e,t){var r;return r=[],l(n,e,t,r),r[0]},f=function(n,e,t){var i,u,f,c;c=o,u=r.findDeferral(arguments),function(n){f=new r.Deferrals(n,{parent:u,filename:"src/icedlib.coffee",funcname:"_iand"}),t[0]=f.defer({assign_fn:function(){return function(){return i=arguments[0]}}(),lineno:34}),f._fulfill()}(function(){return i||(e[0]=!1),n()})},e.iand=function(n,e){var t;return t=[],f(n,e,t),t[0]},c=function(n,e,t){var i,u,f,c;c=o,u=r.findDeferral(arguments),function(n){f=new r.Deferrals(n,{parent:u,filename:"src/icedlib.coffee",funcname:"_ior"}),t[0]=f.defer({assign_fn:function(){return function(){return i=arguments[0]}}(),lineno:51}),f._fulfill()}(function(){return i&&(e[0]=!0),n()})},e.ior=function(n,e){var t;return t=[],c(n,e,t),t[0]},e.Pipeliner=t=function(){function n(n,e){this.window=n||1,this.delay=e||0,this.queue=[],this.n_out=0,this.cb=null,this[i["const"].deferrals]=this,this.defer=this._defer}return n.prototype.waitInQueue=function(n){var e,t,i,u=this;i=o,e=r.findDeferral(arguments),function(n){var i,o;i=[],o=function(n){var f,c,l;return f=function(){return n(i)},c=function(){return r.trampoline(function(){return o(n)})},l=function(n){return i.push(n),c()},u.n_out>=u.window?(!function(n){t=new r.Deferrals(n,{parent:e,filename:"src/icedlib.coffee",funcname:"Pipeliner.waitInQueue"}),u.cb=t.defer({lineno:88}),t._fulfill()}(l),void 0):f()},o(n)}(function(){u.n_out++,function(n){return u.delay?(!function(n){t=new r.Deferrals(n,{parent:e,filename:"src/icedlib.coffee",funcname:"Pipeliner.waitInQueue"}),setTimeout(t.defer({lineno:96}),u.delay),t._fulfill()}(n),void 0):n()}(function(){return n()})})},n.prototype.__defer=function(n,e){var t,i,u,f,c,l=this;c=o,u=r.findDeferral(arguments),function(t){f=new r.Deferrals(t,{parent:u,filename:"src/icedlib.coffee",funcname:"Pipeliner.__defer"}),i=f.defer({lineno:109}),n[0]=function(){var n,t;return n=1<=arguments.length?a.call(arguments,0):[],null!=(t=e.assign_fn)&&t.apply(null,n),i()},f._fulfill()}(function(){return l.n_out--,l.cb?(t=l.cb,l.cb=null,t()):void 0})},n.prototype._defer=function(n){var e;return e=[],this.__defer(e,n),e[0]},n.prototype.flush=function(n){var e,t,i,u,o,f=this;i=n,e=r.findDeferral(arguments),u=[],o=function(n){var i,c,l;return i=function(){return n(u)},c=function(){return r.trampoline(function(){return o(n)})},l=function(n){return u.push(n),c()},f.n_out?(!function(n){t=new r.Deferrals(n,{parent:e,filename:"src/icedlib.coffee",funcname:"Pipeliner.flush"}),f.cb=t.defer({lineno:136}),t._fulfill()}(l),void 0):i()},o(i)},n}()}.call(this),t.exports}(),n["./icedlib"]}();"function"==typeof define&&define.amd?(define(function(){return e}),define(function(){return e.iced})):(n.icedlib=e,n.iced=e.iced)}(this); \ No newline at end of file diff --git a/node_modules/iced-coffee-script/extras/coffee-script-iced.js b/node_modules/iced-coffee-script/extras/coffee-script-iced.js deleted file mode 100644 index 709e414..0000000 --- a/node_modules/iced-coffee-script/extras/coffee-script-iced.js +++ /dev/null @@ -1,8 +0,0 @@ -/** - * IcedCoffeeScript Compiler v1.6.3-g - * http://iced-coffee-script.github.io/iced-coffee-script - * - * Copyright 2011, Jeremy Ashkenas, Maxwell Krohn - * Released under the MIT License - */ -!function(t){var e=function(){function t(e){return t[e]}return t["./iced"]=function(){var t={},e={exports:t};return function(){var e,n=[].slice;t.generator=e=function(t,e,r){var i,o,u,c,a,l;return e.transform=function(t,e){return t.icedTransform(e)},e["const"]=i={k:"__iced_k",k_noop:"__iced_k_noop",param:"__iced_p_",ns:"iced",runtime:"runtime",Deferrals:"Deferrals",deferrals:"__iced_deferrals",fulfill:"_fulfill",b_while:"_break",t_while:"_while",c_while:"_continue",n_while:"_next",n_arg:"__iced_next_arg",context:"context",defer_method:"defer",slot:"__slot",assign_fn:"assign_fn",autocb:"autocb",retslot:"ret",trace:"__iced_trace",passed_deferral:"__iced_passed_deferral",findDeferral:"findDeferral",lineno:"lineno",parent:"parent",filename:"filename",funcname:"funcname",catchExceptions:"catchExceptions",runtime_modes:["node","inline","window","none","browserify"],trampoline:"trampoline"},t.makeDeferReturn=function(e,r,o,u,c){var a,l,s,f;s={};for(a in u)f=u[a],s[a]=f;return s[i.lineno]=null!=r?r[i.lineno]:void 0,l=function(){var i,u,a;return i=1<=arguments.length?n.call(arguments,0):[],null!=r&&null!=(a=r.assign_fn)&&a.apply(null,i),e?(u=e,c||(e=null),u._fulfill(o,s)):t._warn("overused deferral at "+t._trace_to_string(s))},l[i.trace]=s,l},t.__c=0,t.tickCounter=function(e){return t.__c++,0===t.__c%e?(t.__c=0,!0):!1},t.__active_trace=null,t._trace_to_string=function(t){var e;return e=t[i.funcname]||"",""+e+" ("+t[i.filename]+":"+(t[i.lineno]+1)+")"},t._warn=function(t){return"undefined"!=typeof console&&null!==console?console.log("ICED warning: "+t):void 0},r.trampoline=function(e){return t.tickCounter(500)?"undefined"!=typeof process&&null!==process?process.nextTick(e):setTimeout(e):e()},r.Deferrals=o=function(){function e(t,e){this.trace=e,this.continuation=t,this.count=1,this.ret=null}return e.prototype._call=function(e){var n;return this.continuation?(t.__active_trace=e,n=this.continuation,this.continuation=null,n(this.ret)):t._warn("Entered dead await at "+t._trace_to_string(e))},e.prototype._fulfill=function(t,e){var n=this;return--this.count>0?void 0:r.trampoline(function(){return n._call(e)})},e.prototype.defer=function(e){var n;return this.count++,n=this,t.makeDeferReturn(n,e,null,this.trace)},e.prototype._defer=function(t){return this.defer(t)},e}(),r.findDeferral=a=function(t){var e,n,r;for(n=0,r=t.length;r>n;n++)if(e=t[n],null!=e?e[i.trace]:void 0)return e;return null},r.Rendezvous=u=function(){function e(){this.completed=[],this.waiters=[],this.defer_id=0}var n;return n=function(){function t(t,e,n){this.rv=t,this.id=e,this.multi=n}return t.prototype.defer=function(t){return this.rv._deferWithId(this.id,t,this.multi)},t}(),e.prototype.wait=function(t){var e;return this.completed.length?(e=this.completed.shift(),t(e)):this.waiters.push(t)},e.prototype.defer=function(t){var e;return e=this.defer_id++,this.deferWithId(e,t)},e.prototype.id=function(t,e){return null==e&&(e=!1),new n(this,t,e)},e.prototype._fulfill=function(t){var e;return this.waiters.length?(e=this.waiters.shift(),e(t)):this.completed.push(t)},e.prototype._deferWithId=function(e,n,r){return this.count++,t.makeDeferReturn(this,n,e,{},r)},e}(),r.stackWalk=l=function(e){var n,r,o,u;for(r=[],o=e?e[i.trace]:t.__active_trace;o;)n=" at "+t._trace_to_string(o),r.push(n),o=null!=o?null!=(u=o[i.parent])?u[i.trace]:void 0:void 0;return r},r.exceptionHandler=c=function(t,e){var n;return e||(e=console.log),e(t.stack),n=l(),n.length?(e("Iced callback trace:"),e(n.join("\n"))):void 0},r.catchExceptions=function(t){return"undefined"!=typeof process&&null!==process?process.on("uncaughtException",function(e){return c(e,t),process.exit(1)}):void 0}},t.runtime={},e(this,t,t.runtime)}.call(this),e.exports}(),t["./iced"]}();"function"==typeof define&&define.amd?define(function(){return e.runtime}):t.iced=e.runtime}(this); \ No newline at end of file diff --git a/node_modules/iced-coffee-script/iced.md b/node_modules/iced-coffee-script/iced.md deleted file mode 100644 index 7061ca8..0000000 --- a/node_modules/iced-coffee-script/iced.md +++ /dev/null @@ -1,709 +0,0 @@ -# What Is IcedCoffeeScript? - -IcedCoffeeScript (ICS) is a system for handling callbacks in event-based code. -There were two existing implementations, one in [the sfslite library for -C++](https://github.com/maxtaco/sfslite), and another in the [tamejs translator -for JavaScript](https://github.com/maxtaco/tamejs). This extension to -CoffeeScript is a third implementation. The code and translation techniques -are derived from experience with JS, but with some new Coffee-style -flavoring. - -This document first presents a "Iced" tutorial (adapted from the JavaScript -version), and then discusses the specifics of the CoffeeScript implementation. - -# Installing and Running ICS - -ICS is available as an npm package: - - npm install -g iced-coffee-script - -You can alternatively checkout ICS and install from source: - - git clone https://github.com/maxtaco/coffee-script - ./bin/cake install - -This will give you libraries under `iced-coffee-script` and -the binaries `iced` and `icake`, which are replacements -for `coffee` and `cake` respectively. In almost all cases, -`iced` should serve as a drop-in replacement for `coffee`, -since the ICS language is a superset of CoffeeScript. - -For more information about CS and ICS, you can also see -our brochure page. - -# Quick Tutorial and Examples - -Here is a simple example that prints "hello" 10 times, with 100ms -delay slots in between: - -```coffeescript -# A basic serial loop -for i in [0..10] - await setTimeout(defer(), 100) - console.log "hello" -``` - -There is one new language addition here, the `await ... ` block (or -expression), and also one new primitive function, `defer`. The two of -them work in concert. A function must "wait" at the close of a -`await` block until all `defer`rals made in that `await` block are -fulfilled. The function `defer` returns a callback, and a callee in -an `await` block can fulfill a deferral by simply calling the callback -it was given. In the code above, there is only one deferral produced -in each iteration of the loop, so after it's fulfilled by `setTimer` -in 100ms, control continues past the `await` block, onto the log line, -and back to the next iteration of the loop. The code looks and feels -like threaded code, but is still in the asynchronous idiom (if you -look at the rewritten code output by the *coffee* compiler). - -This next example does the same, while showcasing power of the -`await..` language addition. In the example below, the two timers -are fired in parallel, and only when both have fulfilled their deferrals -(after 100ms), does progress continue... - -```coffeescript -for i in [0..10] - await - setTimeout defer(), 100 - setTimeout defer(), 10 - console.log ("hello"); -``` - -Now for something more useful. Here is a parallel DNS resolver that -will exit as soon as the last of your resolutions completes: - -```coffeescript -dns = require("dns"); - -do_one = (cb, host) -> - await dns.resolve host, "A", defer(err, ip) - msg = if err then "ERROR! #{err}" else "#{host} -> #{ip}" - console.log msg - cb() - -do_all = (lst) -> - await - for h in lst - do_one defer(), h - -do_all process.argv[2...] -``` - -You can run this on the command line like so: - - iced examples/iced/dns.coffee yahoo.com google.com nytimes.com okcupid.com tinyurl.com - -And you will get a response: - - yahoo.com -> 72.30.2.43,98.137.149.56,209.191.122.70,67.195.160.76,69.147.125.65 - google.com -> 74.125.93.105,74.125.93.99,74.125.93.104,74.125.93.147,74.125.93.106,74.125.93.103 - nytimes.com -> 199.239.136.200 - okcupid.com -> 66.59.66.6 - tinyurl.com -> 195.66.135.140,195.66.135.139 - -If you want to run these DNS resolutions in serial (rather than -parallel), then the change from above is trivial: just switch the -order of the `await` and `for` statements above: - -```coffeescript -do_all = (lst) -> - for h in lst - await - do_one defer(), h -``` - -### Slightly More Advanced Example - -We've shown parallel and serial work flows, what about something in -between? For instance, we might want to make progress in parallel on -our DNS lookups, but not smash the server all at once. A compromise is -windowing, which can be achieved in IcedCoffeeScript conveniently in a -number of different ways. The [2007 academic paper on -tame](http://pdos.csail.mit.edu/~max/docs/tame.pdf) suggests a -technique called a *rendezvous*. A rendezvous is implemented in -CoffeeScript as a pure CS construct (no rewriting involved), which -allows a program to continue as soon as the first deferral is -fulfilled (rather than the last): - -```coffeescript -do_all = (lst, windowsz) -> - rv = new iced.Rendezvous - nsent = 0 - nrecv = 0 - - while nrecv < lst.length - if nsent - nrecv < windowsz and nsent < n - do_one rv.id(nsent).defer(), lst[nsent] - nsent++ - else - await rv.wait defer evid - console.log "got back lookup nsent=#{evid}" - nrecv++ -``` - -This code maintains two counters: the number of requests sent, and the -number received. It keeps looping until the last lookup is received. -Inside the loop, if there is room in the window and there are more to -send, then send; otherwise, wait and harvest. `Rendezvous.defer` -makes a deferral much like the `defer` primitive, but it can be -labeled with an identifier. This way, the waiter can know which -deferral has fulfilled. In this case we use the variable `nsent` as the -defer ID --- it's the ID of this deferral in launch order. When we -harvest the deferral, `rv.wait` fires its callback with the ID of the -deferral that's harvested. - -Note that with windowing, the arrival order might not be the same as -the issue order. In this example, a slower DNS lookup might arrive -after faster ones, even if issued before them. - -### Composing Serial And Parallel Patterns - -In IcedCoffeeScript, arbitrary composition of serial and parallel control flows is -possible with just normal functional decomposition. Therefore, we -don't allow direct `await` nesting. With inline anonymous CoffeeScript -functions, you can concisely achieve interesting patterns. The code -below launches 10 parallel computations, each of which must complete -two serial actions before finishing: - -```coffeescript -f = (n,cb) -> - await - for i in [0..n] - ((cb) -> - await setTimeout defer(), 5 * Math.random() - await setTimeout defer(), 4 * Math.random() - cb() - )(defer()) - cb() -``` - -### autocb - -Most of the time, an iced function will call its callback and return -at the same time. To get this behavior "for free", you can simply -name this callback `autocb` and it will fire whenever your iced function -returns. For instance, the above example could be equivalently written as: - -```coffeescript -f = (n,autocb) -> - await - for i in [0..n] - ((autocb) -> - setTimeout defer(), 5 * Math.random() - setTimeout defer(), 4 * Math.random() - )(defer()) -``` -In the first example, recall, you call `cb()` explicitly. In this -example, because the callback is named `autocb`, it's fired -automatically when the iced function returns. - -If your callback needs to fulfill with a value, then you can pass -that value via `return`. Consider the following function, that waits -for a random number of seconds between 0 and 4. After waiting, it -then fulfills its callback `cb` with the amount of time it waited: - -```coffeescript -rand_wait = (cb) -> - time = Math.floor Math.random() * 5 - if time is 0 - cb(0) - return - await setTimeout defer(), time - cb(time) # return here, implicitly..... -``` - -This function can written equivalently with `autocb` as: - -```coffeescript -rand_wait = (autocb) -> - time = Math.floor Math.random() * 5 - return 0 if time is 0 - await setTimeout defer(), time - return time -``` - -Implicitly, `return 0;` is mapped by the CoffeeScript compiler to `autocb(0); return`. - -## Language Design Considerations - -In sum, the iced additions to CoffeeScript consist of three new keywords: - -* **await**, marking off a block or a single statement. -* **defer**, which is quite similar to a normal function call, but is compiled specially -to accommodate argument passing. - -Finally, `autocb` isn't a bona-fide keyword, but the compiler searches -for it in parameters to CoffeeScript functions, and updates the -behavior of the `Code` block accordingly. - -These keywords represent the potential for these iced additions to -break existing CoffeeScript code --- any preexisting use of these -keywords as regular function, variable or class names will cause -headaches. - -### Debugging and Stack Traces -- Now Greatly Improved! - -An oft-cited problem with async-style programming, with ICS or -hand-rolled, is that stack traces are often incomplete or -incomprehensible. If an exception is caught in a Iced function, the -stack trace will only show the "bottom half" of the call stack, or all -of those functions that are descendents of the main event loop. The -"top half" of the call stack, telling you "who _really_ called this -function," is probably long gone. - -ICS has a workaround to this problem. When an iced function is -entered, the runtime will find the first argument to the function that -was output by `defer()`. Such callbacks are annotated to contain the -file, line and function where they were created. They also are -annotated to hold a refernce to `defer()`-generated callback passed to -the function in which they were created. This chaining creates an -implicit stack that can be walked when an exception is thrown. - -Consider this example: - -```coffeescript -iced.catchExceptions() - -foo = (y) -> - await setTimeout defer(), 10 - throw new Error "oh no!" - y(10) - -bar = (x) -> - await foo defer() - x() - -baz = () -> - await bar defer() - -baz() -``` - -The function `iced.catchExceptions` sets the `uncaughtException` -handler in Node to print out the standard callstack, and also the Iced -"callstack", and then to exit. The callback generated by `defer()` -in the function `bar` holds a reference to `x`. Similarly, -the callback generated in `foo` holds a reference to `y`. -Here's what happens when this program is run: - -``` -Error: oh no! - at Deferrals.continuation (/Users/max/src/coffee-script/prog.iced:24:13) - at Deferrals._call (/Users/max/src/coffee-script/lib/coffee-script/iced.js:86:19) - at Deferrals._fulfill (/Users/max/src/coffee-script/lib/coffee-script/iced.js:97:23) - at Object._onTimeout (/Users/max/src/coffee-script/lib/coffee-script/iced.js:53:18) - at Timer.ontimeout (timers.js:84:39) -Iced 'stack' trace (w/ real line numbers): - at foo (prog.iced:4) - at bar (prog.iced:9) - at baz (prog.iced:13) -``` - -The first stack trace is the standard Node stacktrace. It is -inscrutable, since it mainly covers node internals, and has line -numbering relative to the translated file (I still haven't fixed this -bug, sorry). The second stack trace is much better. It tells the -sequence of Iced calls the lead to this exception. Line numbers are -relative to the original input file. - -The relavant API is as follows: - -#### iced.stackWalk cb - -Start from the given `cb`, or use the currently active callback -if none was given, and walk up the Iced-generated stack. Return -a list of call site descriptions. You can call this from your -own exception-handling code. - -#### iced.catchExceptions() - -Tell the runtime to catch uncaught exceptions, and to print -a Iced-aware stack dump as above. - - -### The Lowdown on defer - -The implementation of `defer` is interesting --- it's trying to -emulate ``call by reference'' in languages like C++ or Java. Here is an -example that shows off the four different cases required to make this -happen: - -```coffeescript -cb = defer x, obj.field, arr[i], rest... -``` - -And here is the output from the iced `coffee` compiler: - -```javascript -cb = __iced_deferrals.defer({ - assign_fn: (function(__slot_1, __slot_2, __slot_3) { - return function() { - x = arguments[0]; - __slot_1.field = arguments[1]; - __slot_2[__slot_3] = arguments[2]; - return rest = __slice.call(arguments, 3); - }; - })(obj, arr, i) - }); -``` - -The `__iced_deferrals` object is an internal object of type `Deferrals` -that's collecting all calls to `defer` in the current `await` block. -The one in question should fulfill with 3 or more values. When it does, -it will call into the innermost anonymous function to perform the -appropriate assignments in the original scope. The four cases are: - -1. **Simple assignment** --- seen in `x = arguments[0]`. Here, the -`x` variable is in the scope of the original `defer` call. - -1. **Object slot assignment** --- seen in `__slot_1.field = arguments[1]`. -Here, the reference `obj` must be captured at the time of the `defer` call, -and `obj.field` is filled in later. - -1. **Array cell assignment** --- seen in `__slot_2[__slot_3] = arguments[2]`. -This of course will work on an array or an object. Here, the reference -to the array, and the value of the index must be captured when `defer` -is called, and the cell is assigned later. - -1. **Splat assignment** --- seen in `res = __slice.call(arguments,3)`. -This is much like a simple assignment, but allows a ``splat'' meaning -assignment of multiple values at once, accessed as an array. - -These specifics are also detailed in the code in the `Defer` class, -file `nodes.coffee`. - -### Awaits No Longer Work as Expressions - -The following do not work and will generate syntax errors at compile time: - -```coffeescript -y = (await foo defer x) -``` - -```coffeescript -x = if true - await foo defer y - y -else 10 -``` - -```coffescript -my_func 10, ( - await foo defer y - y -) -``` - -That is, you can't treat `await` statements as expressions. -And recursively speaking, you can't treat any blocks that -contain `await` statements as expressions. Previous versions of -IcedCoffeeScript supported this arcane feature, but it was extremely -difficult to implement properly, and unnecessarily obscured the -control flow of iced programs. - -## Translation Technique - -The IcedCoffeeScript addition uses a similar continuation-passing translation -to *tamejs*, but it's been refined to generate cleaner code, and to translate -only when necessary. Here are the general steps involved: - -* **1** Run the standard CoffeeScript lexer, rewriter, and parser, with a -few small additions (for `await` and `defer`), yielding -a standard CoffeeScript-style abstract syntax tree (AST). - -* **2** Apply *iced annotations*: - - * **2.1** Find all `await` nodes in the AST. Mark these nodes and their - ancestors with an **A** flag. - - * **2.2** Find all `for`, `while`, `until`, or `loop` nodes marked with - **A**. Flood them and their descendants with an **L** flag. Stop - flooding when the first loop without an **A** flag is hit. - - * **2.3** Find all `continue` or `break` nodes marked with an **L** flag. - Mark them and their descendants with a **P** flag. - -* **3** ``Rotate'' all those nodes marked with **A** or **P**: - - * **3.1** For each `Block` node _b_ in the `AST` marked **A** or **P**: - - * **3.1.1** Find _b_'s first child _c_ marked with **A** or **P**. - - * **3.1.2** Cut _b_'s list of expressions after _c_, and move those - expressions on the right of the cut into a new block, called - _d_. This block is _c_'s continuation block and becomes _c_'s - child in the AST. This is the actual ``rotation.'' - - * **3.1.3** Call the rotation recursively on the child block _d_. - - * **3.1.4** Add an additional code to _c_'s body, which is to call the - continuation represented by _d_. For `if` statements this means - calling the continuation in both branches; for `switch` - statements, this means calling the continuation from all - branches; for loops, this means calling `continue` at the end of - the loop body; for blocks, this means just calling the - continuation as the last statement in the block. See - `callContinuation` in `nodes.coffee.` - -* **4** Output preamble/boilerplate; for the case of JavaScript output to -browsers, inline the small class `Deferrals` needed during runtime; -for node-based server-side JavaScript, a `require` statement suffices -here. Only do this if the source file has a `defer` statement -in it. - -* **5** Compile as normal. The effect of the above is to mutate the original -CoffeeScript AST into another valid CoffeeScript AST. This AST is then -compiled with the normal rules. - - -Translation Example ------------------- - -For an example translation, consider the following block of code: - -```coffeescript - -while x1 - f1() - -while x2 - if y - f2() - continue - f3() - await - f4(defer()) - if z - f5() - break - f6() - -while x3 - f7() -``` - -* Here is schematic diagram for this AST: - - - -* After Step 2.1, nodes in blue are marked with **A**. Recall, Step 2.1 traces -upwards from all `await` blocks. - - - -* After Step 2.2, nodes in purple are marked with **L**. Recall, Step 2.2 floods -downwards from any any loops marked with **A**. - - - -* After Step 2.3, nodes in yellow are marked with **P**. Recall, Step 2.3 -traces upwards from any jumps marked with **L**. - - - -* The green nodes are those marked with **A** or **P**. They are "marked" -for rotations in the next step. - - - -* In Step 3, rotate all marked nodes AST nodes. This rotation -introduces the new orange `block` nodes in the graph, and attaches -them to pivot nodes as _continuation_ blocks. - - - - -* In translated code, the general format of a _pivot_ node is: - -```javascript -(function (k) { - // the body - k(); -})(function () { - // the continuation block. -} -``` - -To see how pivots and continuations are output in our example, look -at this portion of the AST, introduced after Step 3: - - ![detail](/media/detail.png) - -Here is the translated output (slightly hand-edited for clarity): - -```javascript -(function() { - // await block f4() - (function(k) { - var __deferrals = new iced.Deferrals(k); - f4(__deferrals.defer({})); - __deferrals._fulfill(); - })(function() { - // The continuation block, starting at 'if z' - (function(k) { - if (z) { - f5(); - (function(k) { - // 'break' throws away the current continuation 'k' - // and just calls _break() - _break(); - })(function() { - // A continuation block, after 'break', up to 'f6()' - // This code will never be reached - f6(); - return k(); - }); - } else { - return k(); - } - })(function() { - // end of the loop, call _continue() to start at the top - return _continue(); - }); - }); -}); -``` - -## API and Library Documentation - -### iced.Rendezvous - -The `Rendezvous` is a not a core feature, meaning it's written as a -straight-ahead CoffeeScript library. It's quite useful for more advanced -control flows, so we've included it in the main runtime library. - -The `Rendezvous` is similar to a blocking condition variable (or a -"Hoare style monitor") in threaded programming. - -#### iced.Rendezvous.id(i,[multi]).defer slots... - -Associate a new deferral with the given Rendezvous, whose deferral ID -is `i`, and whose callbacks slots are supplied as `slots`. Those -slots can take the two forms of `defer` return as above. As with -standard `defer`, the return value of the `Rendezvous`'s `defer` is -fed to a function expecting a callback. As soon as that callback -fires (and the deferral is fulfilled), the provided slots will be -filled with the arguments to that callback. - -Also, note the optional boolean flag `multi`. By default, a function -generated by `defer` can be called only once, and will generate an -error on subsequent calls. Only with the `multi` flag set to `true` -(and only in the case of a `Rendezvous`), can this restriction be -relaxed. - -#### iced.Rendezvous.defer slots... - -You don't need to explicitly assign an ID to a deferral generated from a -Rendezvous. If you don't, one will automatically be assigned, in -ascending order starting from `0`. - -#### iced.Rendezvous.wait cb - -Wait until the next deferral on this rendezvous is fulfilled. When it -is, callback `cb` with the ID of the fulfilled deferral. If an -unclaimed deferral fulfilled before `wait` was called, then `cb` is fired -immediately. - -Though `wait` would work with any hand-rolled JS function expecting -a callback, it's meant to work particularly well with *tamejs*'s -`await` function. - -#### Example - -Here is an example that shows off the different inputs and -outputs of a `Rendezvous`. It does two parallel DNS lookups, -and reports only when the first returns: - -```coffeescript -hosts = [ "okcupid.com", "google.com" ]; -ips = errs = [] -rv = new iced.Rendezvous -for h,i in hosts - dns.resolve hosts[i], rv.id(i).defer errs[i], ips[i] - -await rv.wait defer which -console.log "#{hosts[which]} -> #{ips[which]}" -``` - -### connectors - -A *connector* is a function that takes as input -a callback, and outputs another callback. The best example -is a `timeout`, given here: - -#### iced.timeout(cb, time, res = []) - -Timeout an arbitrary async operation. - -Given a callback `cb`, a time to wait `time`, and an array to output a -result `res`, return another callback. This connector will set up a -race between the callback returned to the caller, and the timer that -fires after `time` milliseconds. If the callback returned to the -caller fires first, then fill `res[0] = true;`. If the timer won -(i.e., if there was a timeout), then fill `res[0] = false;`. - -In the following example, we timeout a DNS lookup after 100ms: - -```coffeescript -{timeout} = require 'icedlib' -info = []; -host = "pirateWarezSite.ru"; -await dns.lookup host, timeout(defer(err, ip), 100, info) -if not info[0] - console.log "#{host}: timed out!" -else if (err) - console.log "#{host}: error: #{err}" -else - console.log "#{host} -> #{ip}" -``` - -### The Pipeliner library - -There's another way to do the windowed DNS lookups we saw earlier --- -you can use the control flow library called `Pipeliner`, which -manages the common pattern of having "m calls total, with only -n of them in flight at once, where m > n." - -The Pipeliner class is available in the `icedlib` library: - -```coffeescript -{Pipeliner} = require 'icedlib' -pipeliner = new Pipeliner w,s -``` - -Using the pipeliner, we can rewrite our earlier windowed DNS lookups -as follows: - -```coffescript -do_all = (lst, windowsz) -> - pipeliner = new Pipeliner windowsz - for x in list - await pipeliner.waitInQueue defer() - do_one pipeliner.defer(), x - await pipeliner.flush defer() -``` - -The API is as follows: - -#### new Pipeliner w, s - -Create a new Pipeliner controller, with a window of at most `w` calls -out at once, and waiting `s` seconds before launching each call. The -default values are `w = 10` and `s = 0`. - -#### Pipeliner.waitInQueue c - -Wait in a queue until there's room in the window to launch a new call. -The callback `c` will be fulfilled when there is room. - -#### Pipeliner.defer args... - -Create a new `defer`al for this pipeline, and pass it to whatever -function is doing the actual work. When the work completes, fulfill -this `defer`al --- that will update the accounting in the pipeliner -class, allowing queued actions to proceed. - -#### Pipeliner.flush c - -Wait for the pipeline to clear out. Fulfills the callback `c` -when the last action in the pipeline is done. diff --git a/node_modules/iced-coffee-script/lib/coffee-script/browser.js b/node_modules/iced-coffee-script/lib/coffee-script/browser.js deleted file mode 100644 index 1383a60..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/browser.js +++ /dev/null @@ -1,120 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var CoffeeScript, compile, runScripts, - __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; }; - - - - CoffeeScript = require('./coffee-script'); - - CoffeeScript.require = require; - - compile = CoffeeScript.compile; - - CoffeeScript["eval"] = function(code, options) { - if (options == null) { - options = {}; - } - if (options.bare == null) { - options.bare = true; - } - return eval(compile(code, options)); - }; - - CoffeeScript.run = function(code, options) { - if (options == null) { - options = {}; - } - options.bare = true; - options.shiftLine = true; - return Function(compile(code, options))(); - }; - - if (typeof window === "undefined" || window === null) { - return; - } - - if ((typeof btoa !== "undefined" && btoa !== null) && (typeof JSON !== "undefined" && JSON !== null) && (typeof unescape !== "undefined" && unescape !== null) && (typeof encodeURIComponent !== "undefined" && encodeURIComponent !== null)) { - compile = function(code, options) { - var js, v3SourceMap, _ref; - if (options == null) { - options = {}; - } - options.sourceMap = true; - options.inline = true; - _ref = CoffeeScript.compile(code, options), js = _ref.js, v3SourceMap = _ref.v3SourceMap; - return "" + js + "\n//# sourceMappingURL=data:application/json;base64," + (btoa(unescape(encodeURIComponent(v3SourceMap)))) + "\n//# sourceURL=coffeescript"; - }; - } - - CoffeeScript.load = function(url, callback, options) { - var xhr; - if (options == null) { - options = {}; - } - options.sourceFiles = [url]; - xhr = window.ActiveXObject ? new window.ActiveXObject('Microsoft.XMLHTTP') : new window.XMLHttpRequest(); - xhr.open('GET', url, true); - if ('overrideMimeType' in xhr) { - xhr.overrideMimeType('text/plain'); - } - xhr.onreadystatechange = function() { - var _ref; - if (xhr.readyState === 4) { - if ((_ref = xhr.status) === 0 || _ref === 200) { - CoffeeScript.run(xhr.responseText, options); - } else { - throw new Error("Could not load " + url); - } - if (callback) { - return callback(); - } - } - }; - return xhr.send(null); - }; - - runScripts = function() { - var coffees, coffeetypes, execute, index, length, s, scripts; - scripts = window.document.getElementsByTagName('script'); - coffeetypes = ['text/coffeescript', 'text/literate-coffeescript']; - coffees = (function() { - var _i, _len, _ref, _results; - _results = []; - for (_i = 0, _len = scripts.length; _i < _len; _i++) { - s = scripts[_i]; - if (_ref = s.type, __indexOf.call(coffeetypes, _ref) >= 0) { - _results.push(s); - } - } - return _results; - })(); - index = 0; - length = coffees.length; - (execute = function() { - var mediatype, options, script; - script = coffees[index++]; - mediatype = script != null ? script.type : void 0; - if (__indexOf.call(coffeetypes, mediatype) >= 0) { - options = { - literate: mediatype === 'text/literate-coffeescript' - }; - if (script.src) { - return CoffeeScript.load(script.src, execute, options); - } else { - options.sourceFiles = ['embedded']; - CoffeeScript.run(script.innerHTML, options); - return execute(); - } - } - })(); - return null; - }; - - if (window.addEventListener) { - window.addEventListener('DOMContentLoaded', runScripts, false); - } else { - window.attachEvent('onload', runScripts); - } - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/cake.js b/node_modules/iced-coffee-script/lib/coffee-script/cake.js deleted file mode 100644 index ea53548..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/cake.js +++ /dev/null @@ -1,116 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var CoffeeScript, cakefileDirectory, existsSync, fatalError, fs, helpers, missingTask, oparse, options, optparse, path, printTasks, switches, tasks; - - - - fs = require('fs'); - - path = require('path'); - - helpers = require('./helpers'); - - optparse = require('./optparse'); - - CoffeeScript = require('./coffee-script'); - - existsSync = fs.existsSync || path.existsSync; - - tasks = {}; - - options = {}; - - switches = []; - - oparse = null; - - helpers.extend(global, { - task: function(name, description, action) { - var _ref; - if (!action) { - _ref = [description, action], action = _ref[0], description = _ref[1]; - } - return tasks[name] = { - name: name, - description: description, - action: action - }; - }, - option: function(letter, flag, description) { - return switches.push([letter, flag, description]); - }, - invoke: function(name, cb) { - if (!tasks[name]) { - missingTask(name); - } - return tasks[name].action(options); - } - }); - - exports.run = function(cb) { - var arg, args, e, _i, _len, _ref, _results; - global.__originalDirname = fs.realpathSync('.'); - process.chdir(cakefileDirectory(__originalDirname)); - args = process.argv.slice(2); - CoffeeScript.run(fs.readFileSync('Cakefile').toString(), { - filename: 'Cakefile' - }); - oparse = new optparse.OptionParser(switches); - if (!args.length) { - return printTasks(); - } - try { - options = oparse.parse(args); - } catch (_error) { - e = _error; - return fatalError("" + e); - } - _ref = options["arguments"]; - _results = []; - for (_i = 0, _len = _ref.length; _i < _len; _i++) { - arg = _ref[_i]; - _results.push(invoke(arg)); - } - return _results; - }; - - printTasks = function() { - var cakefilePath, desc, name, relative, spaces, task; - relative = path.relative || path.resolve; - cakefilePath = path.join(relative(__originalDirname, process.cwd()), 'Cakefile'); - console.log("" + cakefilePath + " defines the following tasks:\n"); - for (name in tasks) { - task = tasks[name]; - spaces = 20 - name.length; - spaces = spaces > 0 ? Array(spaces + 1).join(' ') : ''; - desc = task.description ? "# " + task.description : ''; - console.log("cake " + name + spaces + " " + desc); - } - if (switches.length) { - return console.log(oparse.help()); - } - }; - - fatalError = function(message) { - console.error(message + '\n'); - console.log('To see a list of all tasks/options, run "cake"'); - return process.exit(1); - }; - - missingTask = function(task) { - return fatalError("No such task: " + task); - }; - - cakefileDirectory = function(dir) { - var parent; - if (existsSync(path.join(dir, 'Cakefile'))) { - return dir; - } - parent = path.normalize(path.join(dir, '..')); - if (parent !== dir) { - return cakefileDirectory(parent); - } - throw new Error("Cakefile not found in " + (process.cwd())); - }; - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/coffee-script.js b/node_modules/iced-coffee-script/lib/coffee-script/coffee-script.js deleted file mode 100644 index 763483f..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/coffee-script.js +++ /dev/null @@ -1,373 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var Lexer, Module, SourceMap, child_process, compile, compileFile, ext, fileExtensions, findExtension, fork, formatSourcePosition, fs, getSourceMap, helpers, iced, lexer, loadFile, parser, path, sourceMaps, vm, _i, _len, - __hasProp = {}.hasOwnProperty, - __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; }; - - - - fs = require('fs'); - - vm = require('vm'); - - path = require('path'); - - child_process = require('child_process'); - - Lexer = require('./lexer').Lexer; - - parser = require('./parser').parser; - - helpers = require('./helpers'); - - SourceMap = require('./sourcemap').SourceMap; - - iced = require('./iced'); - - exports.VERSION = '1.6.3-g'; - - fileExtensions = ['.coffee', '.litcoffee', '.coffee.md', '.iced']; - - exports.helpers = helpers; - - exports.compile = compile = function(code, options) { - var answer, currentColumn, currentLine, fragment, fragments, header, js, map, merge, newLines, _i, _len; - if (options == null) { - options = {}; - } - merge = helpers.merge; - if (options.sourceMap) { - map = new SourceMap; - } - fragments = (iced.transform(parser.parse(lexer.tokenize(code, options)))).compileToFragments(options); - currentLine = 0; - if (options.header) { - currentLine += 1; - } - if (options.shiftLine) { - currentLine += 1; - } - currentColumn = 0; - js = ""; - for (_i = 0, _len = fragments.length; _i < _len; _i++) { - fragment = fragments[_i]; - if (options.sourceMap) { - if (fragment.locationData) { - map.add([fragment.locationData.first_line, fragment.locationData.first_column], [currentLine, currentColumn], { - noReplace: true - }); - } - newLines = helpers.count(fragment.code, "\n"); - currentLine += newLines; - currentColumn = fragment.code.length - (newLines ? fragment.code.lastIndexOf("\n") : 0); - } - js += fragment.code; - } - if (options.header) { - header = "Generated by IcedCoffeeScript " + this.VERSION; - js = "// " + header + "\n" + js; - } - if (options.sourceMap) { - answer = { - js: js - }; - answer.sourceMap = map; - answer.v3SourceMap = map.generate(options, code); - return answer; - } else { - return js; - } - }; - - exports.tokens = function(code, options) { - return lexer.tokenize(code, options); - }; - - exports.nodes = function(source, options) { - if (typeof source === 'string') { - return iced.transform(parser.parse(lexer.tokenize(source, options)), options); - } else { - return iced.transform(parser.parse(source), options); - } - }; - - exports.run = function(code, options) { - var answer, mainModule, _ref; - if (options == null) { - options = {}; - } - mainModule = require.main; - mainModule.filename = process.argv[1] = options.filename ? fs.realpathSync(options.filename) : '.'; - mainModule.moduleCache && (mainModule.moduleCache = {}); - mainModule.paths = require('module')._nodeModulePaths(path.dirname(fs.realpathSync(options.filename || '.'))); - if (!helpers.isCoffee(mainModule.filename) || require.extensions) { - answer = compile(code, options); - code = (_ref = answer.js) != null ? _ref : answer; - } - return mainModule._compile(code, mainModule.filename); - }; - - exports["eval"] = function(code, options) { - var Module, Script, js, k, o, r, sandbox, v, _i, _len, _module, _ref, _ref1, _require; - if (options == null) { - options = {}; - } - if (!(code = code.trim())) { - return; - } - Script = vm.Script; - if (Script) { - if (options.sandbox != null) { - if (options.sandbox instanceof Script.createContext().constructor) { - sandbox = options.sandbox; - } else { - sandbox = Script.createContext(); - _ref = options.sandbox; - for (k in _ref) { - if (!__hasProp.call(_ref, k)) continue; - v = _ref[k]; - sandbox[k] = v; - } - } - sandbox.global = sandbox.root = sandbox.GLOBAL = sandbox; - } else { - sandbox = global; - } - sandbox.__filename = options.filename || 'eval'; - sandbox.__dirname = path.dirname(sandbox.__filename); - if (!(sandbox !== global || sandbox.module || sandbox.require)) { - Module = require('module'); - sandbox.module = _module = new Module(options.modulename || 'eval'); - sandbox.require = _require = function(path) { - return Module._load(path, _module, true); - }; - _module.filename = sandbox.__filename; - _ref1 = Object.getOwnPropertyNames(require); - for (_i = 0, _len = _ref1.length; _i < _len; _i++) { - r = _ref1[_i]; - if (r !== 'paths') { - _require[r] = require[r]; - } - } - _require.paths = _module.paths = Module._nodeModulePaths(process.cwd()); - _require.resolve = function(request) { - return Module._resolveFilename(request, _module); - }; - } - } - o = {}; - for (k in options) { - if (!__hasProp.call(options, k)) continue; - v = options[k]; - o[k] = v; - } - o.bare = true; - js = compile(code, o); - if (sandbox === global) { - return vm.runInThisContext(js); - } else { - return vm.runInContext(js, sandbox); - } - }; - - compileFile = function(filename, sourceMap) { - var answer, err, raw, stripped; - raw = fs.readFileSync(filename, 'utf8'); - stripped = raw.charCodeAt(0) === 0xFEFF ? raw.substring(1) : raw; - try { - answer = compile(stripped, { - filename: filename, - sourceMap: sourceMap, - literate: helpers.isLiterate(filename) - }); - } catch (_error) { - err = _error; - err.filename = filename; - err.code = stripped; - throw err; - } - return answer; - }; - - loadFile = function(module, filename) { - var answer; - answer = compileFile(filename, false); - return module._compile(answer, filename); - }; - - if (require.extensions) { - for (_i = 0, _len = fileExtensions.length; _i < _len; _i++) { - ext = fileExtensions[_i]; - require.extensions[ext] = loadFile; - } - Module = require('module'); - findExtension = function(filename) { - var curExtension, extensions; - extensions = path.basename(filename).split('.'); - if (extensions[0] === '') { - extensions.shift(); - } - while (extensions.shift()) { - curExtension = '.' + extensions.join('.'); - if (Module._extensions[curExtension]) { - return curExtension; - } - } - return '.js'; - }; - Module.prototype.load = function(filename) { - var extension; - this.filename = filename; - this.paths = Module._nodeModulePaths(path.dirname(filename)); - extension = findExtension(filename); - Module._extensions[extension](this, filename); - return this.loaded = true; - }; - } - - if (child_process) { - fork = child_process.fork; - child_process.fork = function(path, args, options) { - var execPath; - if (args == null) { - args = []; - } - if (options == null) { - options = {}; - } - execPath = helpers.isCoffee(path) ? 'coffee' : null; - if (!Array.isArray(args)) { - args = []; - options = args || {}; - } - options.execPath || (options.execPath = execPath); - return fork(path, args, options); - }; - } - - lexer = new Lexer; - - parser.lexer = { - lex: function() { - var tag, token; - token = this.tokens[this.pos++]; - if (token) { - tag = token[0], this.yytext = token[1], this.yylloc = token[2]; - this.yylineno = this.yylloc.first_line; - } else { - tag = ''; - } - return tag; - }, - setInput: function(tokens) { - this.tokens = tokens; - return this.pos = 0; - }, - upcomingInput: function() { - return ""; - } - }; - - parser.yy = require('./nodes'); - - exports.iced = iced.runtime; - - parser.yy.parseError = function(message, _arg) { - var token; - token = _arg.token; - message = "unexpected " + (token === 1 ? 'end of input' : token); - return helpers.throwSyntaxError(message, parser.lexer.yylloc); - }; - - formatSourcePosition = function(frame, getSourceMapping) { - var as, column, fileLocation, fileName, functionName, isConstructor, isMethodCall, line, methodName, source, tp, typeName; - fileName = void 0; - fileLocation = ''; - if (frame.isNative()) { - fileLocation = "native"; - } else { - if (frame.isEval()) { - fileName = frame.getScriptNameOrSourceURL(); - if (!fileName) { - fileLocation = "" + (frame.getEvalOrigin()) + ", "; - } - } else { - fileName = frame.getFileName(); - } - fileName || (fileName = ""); - line = frame.getLineNumber(); - column = frame.getColumnNumber(); - source = getSourceMapping(fileName, line, column); - fileLocation = source ? "" + fileName + ":" + source[0] + ":" + source[1] : "" + fileName + ":" + line + ":" + column; - } - functionName = frame.getFunctionName(); - isConstructor = frame.isConstructor(); - isMethodCall = !(frame.isToplevel() || isConstructor); - if (isMethodCall) { - methodName = frame.getMethodName(); - typeName = frame.getTypeName(); - if (functionName) { - tp = as = ''; - if (typeName && functionName.indexOf(typeName)) { - tp = "" + typeName + "."; - } - if (methodName && functionName.indexOf("." + methodName) !== functionName.length - methodName.length - 1) { - as = " [as " + methodName + "]"; - } - return "" + tp + functionName + as + " (" + fileLocation + ")"; - } else { - return "" + typeName + "." + (methodName || '') + " (" + fileLocation + ")"; - } - } else if (isConstructor) { - return "new " + (functionName || '') + " (" + fileLocation + ")"; - } else if (functionName) { - return "" + functionName + " (" + fileLocation + ")"; - } else { - return fileLocation; - } - }; - - sourceMaps = {}; - - getSourceMap = function(filename) { - var answer, _ref; - if (sourceMaps[filename]) { - return sourceMaps[filename]; - } - if (_ref = path != null ? path.extname(filename) : void 0, __indexOf.call(fileExtensions, _ref) < 0) { - return; - } - answer = compileFile(filename, true); - return sourceMaps[filename] = answer.sourceMap; - }; - - Error.prepareStackTrace = function(err, stack) { - var frame, frames, getSourceMapping, _ref; - getSourceMapping = function(filename, line, column) { - var answer, sourceMap; - sourceMap = getSourceMap(filename); - if (sourceMap) { - answer = sourceMap.sourceLocation([line - 1, column - 1]); - } - if (answer) { - return [answer[0] + 1, answer[1] + 1]; - } else { - return null; - } - }; - frames = (function() { - var _j, _len1, _results; - _results = []; - for (_j = 0, _len1 = stack.length; _j < _len1; _j++) { - frame = stack[_j]; - if (frame.getFunction() === exports.run) { - break; - } - _results.push(" at " + (formatSourcePosition(frame, getSourceMapping))); - } - return _results; - })(); - return "" + err.name + ": " + ((_ref = err.message) != null ? _ref : '') + "\n" + (frames.join('\n')) + "\n"; - }; - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/command.js b/node_modules/iced-coffee-script/lib/coffee-script/command.js deleted file mode 100644 index ccb2f89..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/command.js +++ /dev/null @@ -1,546 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var BANNER, CoffeeScript, EventEmitter, SWITCHES, compileJoin, compileOptions, compilePath, compileScript, compileStdio, exec, exists, forkNode, fs, handleIcedOptions, helpers, hidden, iced, joinTimeout, notSources, optionParser, optparse, opts, outputPath, parseOptions, path, printLine, printTokens, printWarn, removeSource, runtime_modes_str, sourceCode, sources, spawn, timeLog, unwatchDir, usage, useWinPathSep, version, wait, watch, watchDir, watchers, writeJs, _ref, - __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; }; - - - - fs = require('fs'); - - path = require('path'); - - helpers = require('./helpers'); - - optparse = require('./optparse'); - - CoffeeScript = require('./coffee-script'); - - _ref = require('child_process'), spawn = _ref.spawn, exec = _ref.exec; - - EventEmitter = require('events').EventEmitter; - - iced = require('./iced'); - - runtime_modes_str = "{" + (iced["const"].runtime_modes.join(", ")) + "}"; - - exists = fs.exists || path.exists; - - useWinPathSep = path.sep === '\\'; - - helpers.extend(CoffeeScript, new EventEmitter); - - printLine = function(line) { - return process.stdout.write(line + '\n'); - }; - - printWarn = function(line) { - return process.stderr.write(line + '\n'); - }; - - hidden = function(file) { - return /^\.|~$/.test(file); - }; - - BANNER = 'Usage: iced [options] path/to/script.iced -- [args]\n\nIf called without options, `iced` will run your script.'; - - SWITCHES = [['-b', '--bare', 'compile without a top-level function wrapper'], ['-c', '--compile', 'compile to JavaScript and save as .js files'], ['-e', '--eval', 'pass a string from the command line as input'], ['-h', '--help', 'display this help message'], ['-i', '--interactive', 'run an interactive CoffeeScript REPL'], ['-j', '--join [FILE]', 'concatenate the source CoffeeScript before compiling'], ['-m', '--map', 'generate source map and save as .map files'], ['-n', '--nodes', 'print out the parse tree that the parser produces'], ['--nodejs [ARGS]', 'pass options directly to the "node" binary'], ['-o', '--output [DIR]', 'set the output directory for compiled JavaScript'], ['-p', '--print', 'print out the compiled JavaScript'], ['-s', '--stdio', 'listen for and compile scripts over stdio'], ['-l', '--literate', 'treat stdio as literate style coffee-script'], ['-t', '--tokens', 'print out the tokens that the lexer/rewriter produce'], ['-v', '--version', 'display the version number'], ['-w', '--watch', 'watch scripts for changes and rerun commands'], ['-I', '--runtime [WHICH]', "how to include the iced runtime, one of " + runtime_modes_str + "; default is 'node'"], ['-F', '--runforce', 'output an Iced runtime even if not needed']]; - - opts = {}; - - sources = []; - - sourceCode = []; - - notSources = {}; - - watchers = {}; - - optionParser = null; - - exports.run = function() { - var literals, source, _i, _len, _results; - parseOptions(); - if (opts.nodejs) { - return forkNode(); - } - if (opts.help) { - return usage(); - } - if (opts.version) { - return version(); - } - if (opts.interactive) { - return require('./repl').start(); - } - if (opts.watch && !fs.watch) { - return printWarn("The --watch feature depends on Node v0.6.0+. You are running " + process.version + "."); - } - if (opts.stdio) { - return compileStdio(); - } - if (opts["eval"]) { - return compileScript(null, sources[0]); - } - if (!sources.length) { - return require('./repl').start(); - } - literals = opts.run ? sources.splice(1) : []; - process.argv = process.argv.slice(0, 2).concat(literals); - process.argv[0] = 'coffee'; - _results = []; - for (_i = 0, _len = sources.length; _i < _len; _i++) { - source = sources[_i]; - _results.push(compilePath(source, true, path.normalize(source))); - } - return _results; - }; - - compilePath = function(source, topLevel, base) { - return fs.stat(source, function(err, stats) { - if (err && err.code !== 'ENOENT') { - throw err; - } - if ((err != null ? err.code : void 0) === 'ENOENT') { - console.error("File not found: " + source); - process.exit(1); - } - if (stats.isDirectory() && path.dirname(source) !== 'node_modules') { - if (opts.watch) { - watchDir(source, base); - } - return fs.readdir(source, function(err, files) { - var file, index, _ref1, _ref2; - if (err && err.code !== 'ENOENT') { - throw err; - } - if ((err != null ? err.code : void 0) === 'ENOENT') { - return; - } - index = sources.indexOf(source); - files = files.filter(function(file) { - return !hidden(file); - }); - [].splice.apply(sources, [index, index - index + 1].concat(_ref1 = (function() { - var _i, _len, _results; - _results = []; - for (_i = 0, _len = files.length; _i < _len; _i++) { - file = files[_i]; - _results.push(path.join(source, file)); - } - return _results; - })())), _ref1; - [].splice.apply(sourceCode, [index, index - index + 1].concat(_ref2 = files.map(function() { - return null; - }))), _ref2; - return files.forEach(function(file) { - return compilePath(path.join(source, file), false, base); - }); - }); - } else if (topLevel || helpers.isCoffee(source)) { - if (opts.watch) { - watch(source, base); - } - return fs.readFile(source, function(err, code) { - if (err && err.code !== 'ENOENT') { - throw err; - } - if ((err != null ? err.code : void 0) === 'ENOENT') { - return; - } - return compileScript(source, code.toString(), base); - }); - } else { - notSources[source] = true; - return removeSource(source, base); - } - }); - }; - - compileScript = function(file, input, base) { - var compiled, err, message, o, options, t, task, useColors; - if (base == null) { - base = null; - } - o = opts; - options = compileOptions(file, base); - try { - t = task = { - file: file, - input: input, - options: options - }; - CoffeeScript.emit('compile', task); - if (o.tokens) { - return printTokens(CoffeeScript.tokens(t.input, t.options)); - } else if (o.nodes) { - return printLine(CoffeeScript.nodes(t.input, t.options).toString().trim()); - } else if (o.run) { - return CoffeeScript.run(t.input, t.options); - } else if (o.join && t.file !== o.join) { - if (helpers.isLiterate(file)) { - t.input = helpers.invertLiterate(t.input); - } - sourceCode[sources.indexOf(t.file)] = t.input; - return compileJoin(); - } else { - compiled = CoffeeScript.compile(t.input, t.options); - t.output = compiled; - if (o.map) { - t.output = compiled.js; - t.sourceMap = compiled.v3SourceMap; - } - CoffeeScript.emit('success', task); - if (o.print) { - return printLine(t.output.trim()); - } else if (o.compile || o.map) { - return writeJs(base, t.file, t.output, options.jsPath, t.sourceMap); - } - } - } catch (_error) { - err = _error; - CoffeeScript.emit('failure', err, task); - if (CoffeeScript.listeners('failure').length) { - return; - } - useColors = process.stdout.isTTY && !process.env.NODE_DISABLE_COLORS; - message = helpers.prettyErrorMessage(err, file || '[stdin]', input, useColors); - if (o.watch) { - return printLine(message + '\x07'); - } else { - printWarn(message); - return process.exit(1); - } - } - }; - - compileStdio = function() { - var code, stdin; - code = ''; - stdin = process.openStdin(); - stdin.on('data', function(buffer) { - if (buffer) { - return code += buffer.toString(); - } - }); - return stdin.on('end', function() { - return compileScript(null, code); - }); - }; - - joinTimeout = null; - - compileJoin = function() { - if (!opts.join) { - return; - } - if (!sourceCode.some(function(code) { - return code === null; - })) { - clearTimeout(joinTimeout); - return joinTimeout = wait(100, function() { - return compileScript(opts.join, sourceCode.join('\n'), opts.join); - }); - } - }; - - watch = function(source, base) { - var compile, compileTimeout, e, prevStats, rewatch, watchErr, watcher; - prevStats = null; - compileTimeout = null; - watchErr = function(e) { - if (e.code === 'ENOENT') { - if (sources.indexOf(source) === -1) { - return; - } - try { - rewatch(); - return compile(); - } catch (_error) { - e = _error; - removeSource(source, base, true); - return compileJoin(); - } - } else { - throw e; - } - }; - compile = function() { - clearTimeout(compileTimeout); - return compileTimeout = wait(25, function() { - return fs.stat(source, function(err, stats) { - if (err) { - return watchErr(err); - } - if (prevStats && stats.size === prevStats.size && stats.mtime.getTime() === prevStats.mtime.getTime()) { - return rewatch(); - } - prevStats = stats; - return fs.readFile(source, function(err, code) { - if (err) { - return watchErr(err); - } - compileScript(source, code.toString(), base); - return rewatch(); - }); - }); - }); - }; - try { - watcher = fs.watch(source, compile); - } catch (_error) { - e = _error; - watchErr(e); - } - return rewatch = function() { - if (watcher != null) { - watcher.close(); - } - return watcher = fs.watch(source, compile); - }; - }; - - watchDir = function(source, base) { - var e, readdirTimeout, watcher; - readdirTimeout = null; - try { - return watcher = fs.watch(source, function() { - clearTimeout(readdirTimeout); - return readdirTimeout = wait(25, function() { - return fs.readdir(source, function(err, files) { - var file, _i, _len, _results; - if (err) { - if (err.code !== 'ENOENT') { - throw err; - } - watcher.close(); - return unwatchDir(source, base); - } - _results = []; - for (_i = 0, _len = files.length; _i < _len; _i++) { - file = files[_i]; - if (!(!hidden(file) && !notSources[file])) { - continue; - } - file = path.join(source, file); - if (sources.some(function(s) { - return s.indexOf(file) >= 0; - })) { - continue; - } - sources.push(file); - sourceCode.push(null); - _results.push(compilePath(file, false, base)); - } - return _results; - }); - }); - }); - } catch (_error) { - e = _error; - if (e.code !== 'ENOENT') { - throw e; - } - } - }; - - unwatchDir = function(source, base) { - var file, prevSources, toRemove, _i, _len; - prevSources = sources.slice(0); - toRemove = (function() { - var _i, _len, _results; - _results = []; - for (_i = 0, _len = sources.length; _i < _len; _i++) { - file = sources[_i]; - if (file.indexOf(source) >= 0) { - _results.push(file); - } - } - return _results; - })(); - for (_i = 0, _len = toRemove.length; _i < _len; _i++) { - file = toRemove[_i]; - removeSource(file, base, true); - } - if (!sources.some(function(s, i) { - return prevSources[i] !== s; - })) { - return; - } - return compileJoin(); - }; - - removeSource = function(source, base, removeJs) { - var index, jsPath; - index = sources.indexOf(source); - sources.splice(index, 1); - sourceCode.splice(index, 1); - if (removeJs && !opts.join) { - jsPath = outputPath(source, base); - return exists(jsPath, function(itExists) { - if (itExists) { - return fs.unlink(jsPath, function(err) { - if (err && err.code !== 'ENOENT') { - throw err; - } - return timeLog("removed " + source); - }); - } - }); - } - }; - - outputPath = function(source, base, extension) { - var baseDir, basename, dir, srcDir; - if (extension == null) { - extension = ".js"; - } - basename = helpers.baseFileName(source, true, useWinPathSep); - srcDir = path.dirname(source); - baseDir = base === '.' || base === './' ? srcDir : srcDir.substring(base.length); - dir = opts.output ? path.join(opts.output, baseDir) : srcDir; - return path.join(dir, basename + extension); - }; - - writeJs = function(base, sourcePath, js, jsPath, generatedSourceMap) { - var compile, jsDir, sourceMapPath; - if (generatedSourceMap == null) { - generatedSourceMap = null; - } - sourceMapPath = outputPath(sourcePath, base, ".map"); - jsDir = path.dirname(jsPath); - compile = function() { - if (opts.compile) { - if (js.length <= 0) { - js = ' '; - } - if (generatedSourceMap) { - js = "" + js + "\n//# sourceMappingURL=" + (helpers.baseFileName(sourceMapPath, false, useWinPathSep)) + "\n"; - } - fs.writeFile(jsPath, js, function(err) { - if (err) { - return printLine(err.message); - } else if (opts.compile && opts.watch) { - return timeLog("compiled " + sourcePath); - } - }); - } - if (generatedSourceMap) { - return fs.writeFile(sourceMapPath, generatedSourceMap, function(err) { - if (err) { - return printLine("Could not write source map: " + err.message); - } - }); - } - }; - return exists(jsDir, function(itExists) { - if (itExists) { - return compile(); - } else { - return exec("mkdir -p " + jsDir, compile); - } - }); - }; - - wait = function(milliseconds, func) { - return setTimeout(func, milliseconds); - }; - - timeLog = function(message) { - return console.log("" + ((new Date).toLocaleTimeString()) + " - " + message); - }; - - printTokens = function(tokens) { - var strings, tag, token, value; - strings = (function() { - var _i, _len, _results; - _results = []; - for (_i = 0, _len = tokens.length; _i < _len; _i++) { - token = tokens[_i]; - tag = token[0]; - value = token[1].toString().replace(/\n/, '\\n'); - _results.push("[" + tag + " " + value + "]"); - } - return _results; - })(); - return printLine(strings.join(' ')); - }; - - handleIcedOptions = function(o) { - var v, val; - if (!o.runtime && ((v = process.env.ICED_RUNTIME) != null)) { - o.runtime = v; - } - if (((val = o.runtime) != null) && __indexOf.call(iced["const"].runtime_modes, val) < 0) { - throw new Error("Option -I/--runtime has to be one of " + runtime_modes_str + ", got '" + val + "'"); - } - }; - - parseOptions = function() { - var i, o, source, _i, _len; - optionParser = new optparse.OptionParser(SWITCHES, BANNER); - o = opts = optionParser.parse(process.argv.slice(2)); - handleIcedOptions(o); - o.compile || (o.compile = !!o.output); - o.run = !(o.compile || o.print || o.map); - o.print = !!(o.print || (o["eval"] || o.stdio && o.compile)); - sources = o["arguments"]; - for (i = _i = 0, _len = sources.length; _i < _len; i = ++_i) { - source = sources[i]; - sourceCode[i] = null; - } - }; - - compileOptions = function(filename, base) { - var answer, cwd, jsDir, jsPath; - answer = { - filename: filename, - literate: opts.literate || helpers.isLiterate(filename), - bare: opts.bare, - header: opts.compile, - sourceMap: opts.map, - runtime: opts.runtime, - runforce: opts.runforce - }; - if (filename) { - if (base) { - cwd = process.cwd(); - jsPath = outputPath(filename, base); - jsDir = path.dirname(jsPath); - answer = helpers.merge(answer, { - jsPath: jsPath, - sourceRoot: path.relative(jsDir, cwd), - sourceFiles: [path.relative(cwd, filename)], - generatedFile: helpers.baseFileName(jsPath, false, useWinPathSep) - }); - } else { - answer = helpers.merge(answer, { - sourceRoot: "", - sourceFiles: [helpers.baseFileName(filename, false, useWinPathSep)], - generatedFile: helpers.baseFileName(filename, true, useWinPathSep) + ".js" - }); - } - } - return answer; - }; - - forkNode = function() { - var args, nodeArgs; - nodeArgs = opts.nodejs.split(/\s+/); - args = process.argv.slice(1); - args.splice(args.indexOf('--nodejs'), 2); - return spawn(process.execPath, nodeArgs.concat(args), { - cwd: process.cwd(), - env: process.env, - customFds: [0, 1, 2] - }); - }; - - usage = function() { - return printLine((new optparse.OptionParser(SWITCHES, BANNER)).help()); - }; - - version = function() { - return printLine("IcedCoffeeScript version " + CoffeeScript.VERSION); - }; - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/grammar.js b/node_modules/iced-coffee-script/lib/coffee-script/grammar.js deleted file mode 100644 index fd297d4..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/grammar.js +++ /dev/null @@ -1,641 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var Parser, alt, alternatives, grammar, name, o, operators, token, tokens, unwrap; - - - - Parser = require('jison').Parser; - - unwrap = /^function\s*\(\)\s*\{\s*return\s*([\s\S]*);\s*\}/; - - o = function(patternString, action, options) { - var addLocationDataFn, match, patternCount; - patternString = patternString.replace(/\s{2,}/g, ' '); - patternCount = patternString.split(' ').length; - if (!action) { - return [patternString, '$$ = $1;', options]; - } - action = (match = unwrap.exec(action)) ? match[1] : "(" + action + "())"; - action = action.replace(/\bnew /g, '$&yy.'); - action = action.replace(/\b(?:Block\.wrap|extend)\b/g, 'yy.$&'); - addLocationDataFn = function(first, last) { - if (!last) { - return "yy.addLocationDataFn(@" + first + ")"; - } else { - return "yy.addLocationDataFn(@" + first + ", @" + last + ")"; - } - }; - action = action.replace(/LOC\(([0-9]*)\)/g, addLocationDataFn('$1')); - action = action.replace(/LOC\(([0-9]*),\s*([0-9]*)\)/g, addLocationDataFn('$1', '$2')); - return [patternString, "$$ = " + (addLocationDataFn(1, patternCount)) + "(" + action + ");", options]; - }; - - grammar = { - Root: [ - o('', function() { - return new Block; - }), o('Body') - ], - Body: [ - o('Line', function() { - return Block.wrap([$1]); - }), o('Body TERMINATOR Line', function() { - return $1.push($3); - }), o('Body TERMINATOR') - ], - Line: [o('Expression'), o('Statement')], - Statement: [ - o('Return'), o('Comment'), o('STATEMENT', function() { - return new Literal($1); - }), o('Await') - ], - Await: [ - o('AWAIT Block', function() { - return new Await($2); - }), o('AWAIT Expression', function() { - return new Await(Block.wrap([$2])); - }) - ], - Expression: [o('Value'), o('Invocation'), o('Code'), o('Operation'), o('Assign'), o('If'), o('Try'), o('While'), o('For'), o('Switch'), o('Class'), o('Throw'), o('Defer')], - Block: [ - o('INDENT OUTDENT', function() { - return new Block; - }), o('INDENT Body OUTDENT', function() { - return $2; - }) - ], - Identifier: [ - o('IDENTIFIER', function() { - return new Literal($1); - }) - ], - AlphaNumeric: [ - o('NUMBER', function() { - return new Literal($1); - }), o('STRING', function() { - return new Literal($1); - }) - ], - Literal: [ - o('AlphaNumeric'), o('JS', function() { - return new Literal($1); - }), o('REGEX', function() { - return new Literal($1); - }), o('DEBUGGER', function() { - return new Literal($1); - }), o('UNDEFINED', function() { - return new Undefined; - }), o('NULL', function() { - return new Null; - }), o('BOOL', function() { - return new Bool($1); - }) - ], - Assign: [ - o('Assignable = Expression', function() { - return new Assign($1, $3); - }), o('Assignable = TERMINATOR Expression', function() { - return new Assign($1, $4); - }), o('Assignable = INDENT Expression OUTDENT', function() { - return new Assign($1, $4); - }) - ], - AssignObj: [ - o('ObjAssignable', function() { - return new Value($1); - }), o('ObjAssignable : Expression', function() { - return new Assign(LOC(1)(new Value($1)), $3, 'object'); - }), o('ObjAssignable :\ - INDENT Expression OUTDENT', function() { - return new Assign(LOC(1)(new Value($1)), $4, 'object'); - }), o('Comment') - ], - ObjAssignable: [o('Identifier'), o('AlphaNumeric'), o('ThisProperty')], - Return: [ - o('RETURN Expression', function() { - return new Return($2); - }), o('RETURN', function() { - return new Return; - }) - ], - Comment: [ - o('HERECOMMENT', function() { - return new Comment($1); - }) - ], - Code: [ - o('PARAM_START ParamList PARAM_END FuncGlyph Block', function() { - return new Code($2, $5, $4); - }), o('FuncGlyph Block', function() { - return new Code([], $2, $1); - }) - ], - FuncGlyph: [ - o('->', function() { - return 'func'; - }), o('=>', function() { - return 'boundfunc'; - }) - ], - OptComma: [o(''), o(',')], - ParamList: [ - o('', function() { - return []; - }), o('Param', function() { - return [$1]; - }), o('ParamList , Param', function() { - return $1.concat($3); - }), o('ParamList OptComma TERMINATOR Param', function() { - return $1.concat($4); - }), o('ParamList OptComma INDENT ParamList OptComma OUTDENT', function() { - return $1.concat($4); - }) - ], - Param: [ - o('ParamVar', function() { - return new Param($1); - }), o('ParamVar ...', function() { - return new Param($1, null, true); - }), o('ParamVar = Expression', function() { - return new Param($1, $3); - }) - ], - ParamVar: [o('Identifier'), o('ThisProperty'), o('Array'), o('Object')], - Splat: [ - o('Expression ...', function() { - return new Splat($1); - }) - ], - SimpleAssignable: [ - o('Identifier', function() { - return new Value($1); - }), o('Value Accessor', function() { - return $1.add($2); - }), o('Invocation Accessor', function() { - return new Value($1, [].concat($2)); - }), o('ThisProperty') - ], - Assignable: [ - o('SimpleAssignable'), o('Array', function() { - return new Value($1); - }), o('Object', function() { - return new Value($1); - }) - ], - Value: [ - o('Assignable'), o('Literal', function() { - return new Value($1); - }), o('Parenthetical', function() { - return new Value($1); - }), o('Range', function() { - return new Value($1); - }), o('This') - ], - Accessor: [ - o('. Identifier', function() { - return new Access($2); - }), o('. Defer', function() { - return new Access($2.setCustom()); - }), o('?. Identifier', function() { - return new Access($2, 'soak'); - }), o(':: Identifier', function() { - return [LOC(1)(new Access(new Literal('prototype'))), LOC(2)(new Access($2))]; - }), o('?:: Identifier', function() { - return [LOC(1)(new Access(new Literal('prototype'), 'soak')), LOC(2)(new Access($2))]; - }), o('::', function() { - return new Access(new Literal('prototype')); - }), o('Index') - ], - Index: [ - o('INDEX_START IndexValue INDEX_END', function() { - return $2; - }), o('INDEX_SOAK Index', function() { - return extend($2, { - soak: true - }); - }) - ], - IndexValue: [ - o('Expression', function() { - return new Index($1); - }), o('Slice', function() { - return new Slice($1); - }) - ], - Object: [ - o('{ AssignList OptComma }', function() { - return new Obj($2, $1.generated); - }) - ], - AssignList: [ - o('', function() { - return []; - }), o('AssignObj', function() { - return [$1]; - }), o('AssignList , AssignObj', function() { - return $1.concat($3); - }), o('AssignList OptComma TERMINATOR AssignObj', function() { - return $1.concat($4); - }), o('AssignList OptComma INDENT AssignList OptComma OUTDENT', function() { - return $1.concat($4); - }) - ], - Class: [ - o('CLASS', function() { - return new Class; - }), o('CLASS Block', function() { - return new Class(null, null, $2); - }), o('CLASS EXTENDS Expression', function() { - return new Class(null, $3); - }), o('CLASS EXTENDS Expression Block', function() { - return new Class(null, $3, $4); - }), o('CLASS SimpleAssignable', function() { - return new Class($2); - }), o('CLASS SimpleAssignable Block', function() { - return new Class($2, null, $3); - }), o('CLASS SimpleAssignable EXTENDS Expression', function() { - return new Class($2, $4); - }), o('CLASS SimpleAssignable EXTENDS Expression Block', function() { - return new Class($2, $4, $5); - }) - ], - Invocation: [ - o('Value OptFuncExist Arguments', function() { - return new Call($1, $3, $2); - }), o('Invocation OptFuncExist Arguments', function() { - return new Call($1, $3, $2); - }), o('SUPER', function() { - return new Call('super', [new Splat(new Literal('arguments'))]); - }), o('SUPER Arguments', function() { - return new Call('super', $2); - }) - ], - Defer: [ - o('DEFER Arguments', function() { - return new Defer($2, yylineno); - }) - ], - OptFuncExist: [ - o('', function() { - return false; - }), o('FUNC_EXIST', function() { - return true; - }) - ], - Arguments: [ - o('CALL_START CALL_END', function() { - return []; - }), o('CALL_START ArgList OptComma CALL_END', function() { - return $2; - }) - ], - This: [ - o('THIS', function() { - return new Value(new Literal('this')); - }), o('@', function() { - return new Value(new Literal('this')); - }) - ], - ThisProperty: [ - o('@ Identifier', function() { - return new Value(LOC(1)(new Literal('this')), [LOC(2)(new Access($2))], 'this'); - }) - ], - Array: [ - o('[ ]', function() { - return new Arr([]); - }), o('[ ArgList OptComma ]', function() { - return new Arr($2); - }) - ], - RangeDots: [ - o('..', function() { - return 'inclusive'; - }), o('...', function() { - return 'exclusive'; - }) - ], - Range: [ - o('[ Expression RangeDots Expression ]', function() { - return new Range($2, $4, $3); - }) - ], - Slice: [ - o('Expression RangeDots Expression', function() { - return new Range($1, $3, $2); - }), o('Expression RangeDots', function() { - return new Range($1, null, $2); - }), o('RangeDots Expression', function() { - return new Range(null, $2, $1); - }), o('RangeDots', function() { - return new Range(null, null, $1); - }) - ], - ArgList: [ - o('Arg', function() { - return [$1]; - }), o('ArgList , Arg', function() { - return $1.concat($3); - }), o('ArgList OptComma TERMINATOR Arg', function() { - return $1.concat($4); - }), o('INDENT ArgList OptComma OUTDENT', function() { - return $2; - }), o('ArgList OptComma INDENT ArgList OptComma OUTDENT', function() { - return $1.concat($4); - }) - ], - Arg: [o('Expression'), o('Splat')], - SimpleArgs: [ - o('Expression'), o('SimpleArgs , Expression', function() { - return [].concat($1, $3); - }) - ], - Try: [ - o('TRY Block', function() { - return new Try($2); - }), o('TRY Block Catch', function() { - return new Try($2, $3[0], $3[1]); - }), o('TRY Block FINALLY Block', function() { - return new Try($2, null, null, $4); - }), o('TRY Block Catch FINALLY Block', function() { - return new Try($2, $3[0], $3[1], $5); - }) - ], - Catch: [ - o('CATCH Identifier Block', function() { - return [$2, $3]; - }), o('CATCH Object Block', function() { - return [LOC(2)(new Value($2)), $3]; - }), o('CATCH Block', function() { - return [null, $2]; - }) - ], - Throw: [ - o('THROW Expression', function() { - return new Throw($2); - }) - ], - Parenthetical: [ - o('( Body )', function() { - return new Parens($2); - }), o('( INDENT Body OUTDENT )', function() { - return new Parens($3); - }) - ], - WhileSource: [ - o('WHILE Expression', function() { - return new While($2); - }), o('WHILE Expression WHEN Expression', function() { - return new While($2, { - guard: $4 - }); - }), o('UNTIL Expression', function() { - return new While($2, { - invert: true - }); - }), o('UNTIL Expression WHEN Expression', function() { - return new While($2, { - invert: true, - guard: $4 - }); - }) - ], - While: [ - o('WhileSource Block', function() { - return $1.addBody($2); - }), o('Statement WhileSource', function() { - return $2.addBody(LOC(1)(Block.wrap([$1]))); - }), o('Expression WhileSource', function() { - return $2.addBody(LOC(1)(Block.wrap([$1]))); - }), o('Loop', function() { - return $1; - }) - ], - Loop: [ - o('LOOP Block', function() { - return new While(LOC(1)(new Literal('true'))).addBody($2); - }), o('LOOP Expression', function() { - return new While(LOC(1)(new Literal('true'))).addBody(LOC(2)(Block.wrap([$2]))); - }) - ], - For: [ - o('Statement ForBody', function() { - return new For($1, $2); - }), o('Expression ForBody', function() { - return new For($1, $2); - }), o('ForBody Block', function() { - return new For($2, $1); - }) - ], - ForBody: [ - o('FOR Range', function() { - return { - source: LOC(2)(new Value($2)) - }; - }), o('ForStart ForSource', function() { - $2.own = $1.own; - $2.name = $1[0]; - $2.index = $1[1]; - return $2; - }) - ], - ForStart: [ - o('FOR ForVariables', function() { - return $2; - }), o('FOR OWN ForVariables', function() { - $3.own = true; - return $3; - }) - ], - ForValue: [ - o('Identifier'), o('ThisProperty'), o('Array', function() { - return new Value($1); - }), o('Object', function() { - return new Value($1); - }) - ], - ForVariables: [ - o('ForValue', function() { - return [$1]; - }), o('ForValue , ForValue', function() { - return [$1, $3]; - }) - ], - ForSource: [ - o('FORIN Expression', function() { - return { - source: $2 - }; - }), o('FOROF Expression', function() { - return { - source: $2, - object: true - }; - }), o('FORIN Expression WHEN Expression', function() { - return { - source: $2, - guard: $4 - }; - }), o('FOROF Expression WHEN Expression', function() { - return { - source: $2, - guard: $4, - object: true - }; - }), o('FORIN Expression BY Expression', function() { - return { - source: $2, - step: $4 - }; - }), o('FORIN Expression WHEN Expression BY Expression', function() { - return { - source: $2, - guard: $4, - step: $6 - }; - }), o('FORIN Expression BY Expression WHEN Expression', function() { - return { - source: $2, - step: $4, - guard: $6 - }; - }) - ], - Switch: [ - o('SWITCH Expression INDENT Whens OUTDENT', function() { - return new Switch($2, $4); - }), o('SWITCH Expression INDENT Whens ELSE Block OUTDENT', function() { - return new Switch($2, $4, $6); - }), o('SWITCH INDENT Whens OUTDENT', function() { - return new Switch(null, $3); - }), o('SWITCH INDENT Whens ELSE Block OUTDENT', function() { - return new Switch(null, $3, $5); - }) - ], - Whens: [ - o('When'), o('Whens When', function() { - return $1.concat($2); - }) - ], - When: [ - o('LEADING_WHEN SimpleArgs Block', function() { - return [[$2, $3]]; - }), o('LEADING_WHEN SimpleArgs Block TERMINATOR', function() { - return [[$2, $3]]; - }) - ], - IfBlock: [ - o('IF Expression Block', function() { - return new If($2, $3, { - type: $1 - }); - }), o('IfBlock ELSE IF Expression Block', function() { - return $1.addElse(LOC(3, 5)(new If($4, $5, { - type: $3 - }))); - }) - ], - If: [ - o('IfBlock'), o('IfBlock ELSE Block', function() { - return $1.addElse($3); - }), o('Statement POST_IF Expression', function() { - return new If($3, LOC(1)(Block.wrap([$1])), { - type: $2, - statement: true - }); - }), o('Expression POST_IF Expression', function() { - return new If($3, LOC(1)(Block.wrap([$1])), { - type: $2, - statement: true - }); - }) - ], - Operation: [ - o('UNARY Expression', function() { - return new Op($1, $2); - }), o('- Expression', (function() { - return new Op('-', $2); - }), { - prec: 'UNARY' - }), o('+ Expression', (function() { - return new Op('+', $2); - }), { - prec: 'UNARY' - }), o('-- SimpleAssignable', function() { - return new Op('--', $2); - }), o('++ SimpleAssignable', function() { - return new Op('++', $2); - }), o('SimpleAssignable --', function() { - return new Op('--', $1, null, true); - }), o('SimpleAssignable ++', function() { - return new Op('++', $1, null, true); - }), o('Expression ?', function() { - return new Existence($1); - }), o('Expression + Expression', function() { - return new Op('+', $1, $3); - }), o('Expression - Expression', function() { - return new Op('-', $1, $3); - }), o('Expression MATH Expression', function() { - return new Op($2, $1, $3); - }), o('Expression SHIFT Expression', function() { - return new Op($2, $1, $3); - }), o('Expression COMPARE Expression', function() { - return new Op($2, $1, $3); - }), o('Expression LOGIC Expression', function() { - return new Op($2, $1, $3); - }), o('Expression RELATION Expression', function() { - if ($2.charAt(0) === '!') { - return new Op($2.slice(1), $1, $3).invert(); - } else { - return new Op($2, $1, $3); - } - }), o('SimpleAssignable COMPOUND_ASSIGN\ - Expression', function() { - return new Assign($1, $3, $2); - }), o('SimpleAssignable COMPOUND_ASSIGN\ - INDENT Expression OUTDENT', function() { - return new Assign($1, $4, $2); - }), o('SimpleAssignable COMPOUND_ASSIGN TERMINATOR\ - Expression', function() { - return new Assign($1, $4, $2); - }), o('SimpleAssignable EXTENDS Expression', function() { - return new Extends($1, $3); - }) - ] - }; - - operators = [['left', '.', '?.', '::', '?::'], ['left', 'CALL_START', 'CALL_END'], ['nonassoc', '++', '--'], ['left', '?'], ['right', 'UNARY'], ['left', 'MATH'], ['left', '+', '-'], ['left', 'SHIFT'], ['left', 'RELATION'], ['left', 'COMPARE'], ['left', 'LOGIC'], ['nonassoc', 'INDENT', 'OUTDENT'], ['right', '=', ':', 'COMPOUND_ASSIGN', 'RETURN', 'THROW', 'EXTENDS'], ['right', 'FORIN', 'FOROF', 'BY', 'WHEN'], ['right', 'IF', 'ELSE', 'FOR', 'WHILE', 'UNTIL', 'LOOP', 'SUPER', 'CLASS', 'AWAIT'], ['right', 'POST_IF']]; - - tokens = []; - - for (name in grammar) { - alternatives = grammar[name]; - grammar[name] = (function() { - var _i, _j, _len, _len1, _ref, _results; - _results = []; - for (_i = 0, _len = alternatives.length; _i < _len; _i++) { - alt = alternatives[_i]; - _ref = alt[0].split(' '); - for (_j = 0, _len1 = _ref.length; _j < _len1; _j++) { - token = _ref[_j]; - if (!grammar[token]) { - tokens.push(token); - } - } - if (name === 'Root') { - alt[1] = "return " + alt[1]; - } - _results.push(alt); - } - return _results; - })(); - } - - exports.parser = new Parser({ - tokens: tokens.join(' '), - bnf: grammar, - operators: operators.reverse(), - startSymbol: 'Root' - }); - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/helpers.js b/node_modules/iced-coffee-script/lib/coffee-script/helpers.js deleted file mode 100644 index 209779f..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/helpers.js +++ /dev/null @@ -1,227 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var buildLocationData, extend, flatten, last, repeat, _ref; - - - - exports.starts = function(string, literal, start) { - return literal === string.substr(start, literal.length); - }; - - exports.ends = function(string, literal, back) { - var len; - len = literal.length; - return literal === string.substr(string.length - len - (back || 0), len); - }; - - exports.repeat = repeat = function(str, n) { - var res; - res = ''; - while (n > 0) { - if (n & 1) { - res += str; - } - n >>>= 1; - str += str; - } - return res; - }; - - exports.compact = function(array) { - var item, _i, _len, _results; - _results = []; - for (_i = 0, _len = array.length; _i < _len; _i++) { - item = array[_i]; - if (item) { - _results.push(item); - } - } - return _results; - }; - - exports.count = function(string, substr) { - var num, pos; - num = pos = 0; - if (!substr.length) { - return 1 / 0; - } - while (pos = 1 + string.indexOf(substr, pos)) { - num++; - } - return num; - }; - - exports.merge = function(options, overrides) { - return extend(extend({}, options), overrides); - }; - - extend = exports.extend = function(object, properties) { - var key, val; - for (key in properties) { - val = properties[key]; - object[key] = val; - } - return object; - }; - - exports.flatten = flatten = function(array) { - var element, flattened, _i, _len; - flattened = []; - for (_i = 0, _len = array.length; _i < _len; _i++) { - element = array[_i]; - if (element instanceof Array) { - flattened = flattened.concat(flatten(element)); - } else { - flattened.push(element); - } - } - return flattened; - }; - - exports.del = function(obj, key) { - var val; - val = obj[key]; - delete obj[key]; - return val; - }; - - exports.last = last = function(array, back) { - return array[array.length - (back || 0) - 1]; - }; - - exports.some = (_ref = Array.prototype.some) != null ? _ref : function(fn) { - var e, _i, _len; - for (_i = 0, _len = this.length; _i < _len; _i++) { - e = this[_i]; - if (fn(e)) { - return true; - } - } - return false; - }; - - exports.invertLiterate = function(code) { - var line, lines, maybe_code; - maybe_code = true; - lines = (function() { - var _i, _len, _ref1, _results; - _ref1 = code.split('\n'); - _results = []; - for (_i = 0, _len = _ref1.length; _i < _len; _i++) { - line = _ref1[_i]; - if (maybe_code && /^([ ]{4}|[ ]{0,3}\t)/.test(line)) { - _results.push(line); - } else if (maybe_code = /^\s*$/.test(line)) { - _results.push(line); - } else { - _results.push('# ' + line); - } - } - return _results; - })(); - return lines.join('\n'); - }; - - buildLocationData = function(first, last) { - if (!last) { - return first; - } else { - return { - first_line: first.first_line, - first_column: first.first_column, - last_line: last.last_line, - last_column: last.last_column - }; - } - }; - - exports.addLocationDataFn = function(first, last) { - return function(obj) { - if (((typeof obj) === 'object') && (!!obj['updateLocationDataIfMissing'])) { - obj.updateLocationDataIfMissing(buildLocationData(first, last)); - } - return obj; - }; - }; - - exports.locationDataToString = function(obj) { - var locationData; - if (("2" in obj) && ("first_line" in obj[2])) { - locationData = obj[2]; - } else if ("first_line" in obj) { - locationData = obj; - } - if (locationData) { - return ("" + (locationData.first_line + 1) + ":" + (locationData.first_column + 1) + "-") + ("" + (locationData.last_line + 1) + ":" + (locationData.last_column + 1)); - } else { - return "No location data"; - } - }; - - exports.baseFileName = function(file, stripExt, useWinPathSep) { - var parts, pathSep; - if (stripExt == null) { - stripExt = false; - } - if (useWinPathSep == null) { - useWinPathSep = false; - } - pathSep = useWinPathSep ? /\\|\// : /\//; - parts = file.split(pathSep); - file = parts[parts.length - 1]; - if (!stripExt) { - return file; - } - parts = file.split('.'); - parts.pop(); - if (parts[parts.length - 1] === 'coffee' && parts.length > 1) { - parts.pop(); - } - return parts.join('.'); - }; - - exports.isCoffee = function(file) { - return /\.((lit)?coffee|coffee\.md|iced)$/.test(file); - }; - - exports.isLiterate = function(file) { - return /\.(litcoffee|coffee\.md)$/.test(file); - }; - - exports.throwSyntaxError = function(message, location) { - var error; - if (location.last_line == null) { - location.last_line = location.first_line; - } - if (location.last_column == null) { - location.last_column = location.first_column; - } - error = new SyntaxError(message); - error.location = location; - throw error; - }; - - exports.prettyErrorMessage = function(error, filename, code, useColors) { - var codeLine, colorize, end, first_column, first_line, last_column, last_line, marker, message, start, _ref1; - if (!error.location) { - return error.stack || ("" + error); - } - filename = error.filename || filename; - code = error.code || code; - _ref1 = error.location, first_line = _ref1.first_line, first_column = _ref1.first_column, last_line = _ref1.last_line, last_column = _ref1.last_column; - codeLine = code.split('\n')[first_line]; - start = first_column; - end = first_line === last_line ? last_column + 1 : codeLine.length; - marker = repeat(' ', start) + repeat('^', end - start); - if (useColors) { - colorize = function(str) { - return "\x1B[1;31m" + str + "\x1B[0m"; - }; - codeLine = codeLine.slice(0, start) + colorize(codeLine.slice(start, end)) + codeLine.slice(end); - marker = colorize(marker); - } - message = "" + filename + ":" + (first_line + 1) + ":" + (first_column + 1) + ": error: " + error.message + "\n" + codeLine + "\n" + marker; - return message; - }; - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/iced.js b/node_modules/iced-coffee-script/lib/coffee-script/iced.js deleted file mode 100644 index 43031d5..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/iced.js +++ /dev/null @@ -1,256 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var generator, - __slice = [].slice; - - - - exports.generator = generator = function(intern, compiletime, runtime) { - var C, Deferrals, Rendezvous, exceptionHandler, findDeferral, stackWalk; - compiletime.transform = function(x, options) { - return x.icedTransform(options); - }; - compiletime["const"] = C = { - k: "__iced_k", - k_noop: "__iced_k_noop", - param: "__iced_p_", - ns: "iced", - runtime: "runtime", - Deferrals: "Deferrals", - deferrals: "__iced_deferrals", - fulfill: "_fulfill", - b_while: "_break", - t_while: "_while", - c_while: "_continue", - n_while: "_next", - n_arg: "__iced_next_arg", - context: "context", - defer_method: "defer", - slot: "__slot", - assign_fn: "assign_fn", - autocb: "autocb", - retslot: "ret", - trace: "__iced_trace", - passed_deferral: "__iced_passed_deferral", - findDeferral: "findDeferral", - lineno: "lineno", - parent: "parent", - filename: "filename", - funcname: "funcname", - catchExceptions: 'catchExceptions', - runtime_modes: ["node", "inline", "window", "none", "browserify"], - trampoline: "trampoline" - }; - intern.makeDeferReturn = function(obj, defer_args, id, trace_template, multi) { - var k, ret, trace, v; - trace = {}; - for (k in trace_template) { - v = trace_template[k]; - trace[k] = v; - } - trace[C.lineno] = defer_args != null ? defer_args[C.lineno] : void 0; - ret = function() { - var inner_args, o, _ref; - inner_args = 1 <= arguments.length ? __slice.call(arguments, 0) : []; - if (defer_args != null) { - if ((_ref = defer_args.assign_fn) != null) { - _ref.apply(null, inner_args); - } - } - if (obj) { - o = obj; - if (!multi) { - obj = null; - } - return o._fulfill(id, trace); - } else { - return intern._warn("overused deferral at " + (intern._trace_to_string(trace))); - } - }; - ret[C.trace] = trace; - return ret; - }; - intern.__c = 0; - intern.tickCounter = function(mod) { - intern.__c++; - if ((intern.__c % mod) === 0) { - intern.__c = 0; - return true; - } else { - return false; - } - }; - intern.__active_trace = null; - intern._trace_to_string = function(tr) { - var fn; - fn = tr[C.funcname] || ""; - return "" + fn + " (" + tr[C.filename] + ":" + (tr[C.lineno] + 1) + ")"; - }; - intern._warn = function(m) { - return typeof console !== "undefined" && console !== null ? console.log("ICED warning: " + m) : void 0; - }; - runtime.trampoline = function(fn) { - if (!intern.tickCounter(500)) { - return fn(); - } else if (typeof process !== "undefined" && process !== null) { - return process.nextTick(fn); - } else { - return setTimeout(fn); - } - }; - runtime.Deferrals = Deferrals = (function() { - function Deferrals(k, trace) { - this.trace = trace; - this.continuation = k; - this.count = 1; - this.ret = null; - } - - Deferrals.prototype._call = function(trace) { - var c; - if (this.continuation) { - intern.__active_trace = trace; - c = this.continuation; - this.continuation = null; - return c(this.ret); - } else { - return intern._warn("Entered dead await at " + (intern._trace_to_string(trace))); - } - }; - - Deferrals.prototype._fulfill = function(id, trace) { - var _this = this; - if (--this.count > 0) { - - } else { - return runtime.trampoline((function() { - return _this._call(trace); - })); - } - }; - - Deferrals.prototype.defer = function(args) { - var self; - this.count++; - self = this; - return intern.makeDeferReturn(self, args, null, this.trace); - }; - - Deferrals.prototype._defer = function(args) { - return this["defer"](args); - }; - - return Deferrals; - - })(); - runtime.findDeferral = findDeferral = function(args) { - var a, _i, _len; - for (_i = 0, _len = args.length; _i < _len; _i++) { - a = args[_i]; - if (a != null ? a[C.trace] : void 0) { - return a; - } - } - return null; - }; - runtime.Rendezvous = Rendezvous = (function() { - var RvId; - - function Rendezvous() { - this.completed = []; - this.waiters = []; - this.defer_id = 0; - } - - RvId = (function() { - function RvId(rv, id, multi) { - this.rv = rv; - this.id = id; - this.multi = multi; - } - - RvId.prototype.defer = function(defer_args) { - return this.rv._deferWithId(this.id, defer_args, this.multi); - }; - - return RvId; - - })(); - - Rendezvous.prototype.wait = function(cb) { - var x; - if (this.completed.length) { - x = this.completed.shift(); - return cb(x); - } else { - return this.waiters.push(cb); - } - }; - - Rendezvous.prototype.defer = function(defer_args) { - var id; - id = this.defer_id++; - return this.deferWithId(id, defer_args); - }; - - Rendezvous.prototype.id = function(i, multi) { - if (multi == null) { - multi = false; - } - return new RvId(this, i, multi); - }; - - Rendezvous.prototype._fulfill = function(id, trace) { - var cb; - if (this.waiters.length) { - cb = this.waiters.shift(); - return cb(id); - } else { - return this.completed.push(id); - } - }; - - Rendezvous.prototype._deferWithId = function(id, defer_args, multi) { - this.count++; - return intern.makeDeferReturn(this, defer_args, id, {}, multi); - }; - - return Rendezvous; - - })(); - runtime.stackWalk = stackWalk = function(cb) { - var line, ret, tr, _ref; - ret = []; - tr = cb ? cb[C.trace] : intern.__active_trace; - while (tr) { - line = " at " + (intern._trace_to_string(tr)); - ret.push(line); - tr = tr != null ? (_ref = tr[C.parent]) != null ? _ref[C.trace] : void 0 : void 0; - } - return ret; - }; - runtime.exceptionHandler = exceptionHandler = function(err, logger) { - var stack; - if (!logger) { - logger = console.log; - } - logger(err.stack); - stack = stackWalk(); - if (stack.length) { - logger("Iced callback trace:"); - return logger(stack.join("\n")); - } - }; - return runtime.catchExceptions = function(logger) { - return typeof process !== "undefined" && process !== null ? process.on('uncaughtException', function(err) { - exceptionHandler(err, logger); - return process.exit(1); - }) : void 0; - }; - }; - - exports.runtime = {}; - - generator(this, exports, exports.runtime); - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/icedlib.js b/node_modules/iced-coffee-script/lib/coffee-script/icedlib.js deleted file mode 100644 index 1da9cff..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/icedlib.js +++ /dev/null @@ -1,288 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var Pipeliner, iced, iced_internals, __iced_k, __iced_k_noop, _iand, _ior, _timeout, - __slice = [].slice; - - __iced_k = __iced_k_noop = function() {}; - - iced_internals = require('./iced'); - - exports.iced = iced = iced_internals.runtime; - - _timeout = function(cb, t, res, tmp) { - var arr, rv, which, ___iced_passed_deferral, __iced_deferrals, __iced_k, - _this = this; - __iced_k = __iced_k_noop; - ___iced_passed_deferral = iced.findDeferral(arguments); - rv = new iced.Rendezvous; - tmp[0] = rv.id(true).defer({ - assign_fn: (function() { - return function() { - return arr = __slice.call(arguments, 0); - }; - })(), - lineno: 17, - context: __iced_deferrals - }); - setTimeout(rv.id(false).defer({ - lineno: 18, - context: __iced_deferrals - }), t); - (function(__iced_k) { - __iced_deferrals = new iced.Deferrals(__iced_k, { - parent: ___iced_passed_deferral, - filename: "src/icedlib.coffee", - funcname: "_timeout" - }); - rv.wait(__iced_deferrals.defer({ - assign_fn: (function() { - return function() { - return which = arguments[0]; - }; - })(), - lineno: 19 - })); - __iced_deferrals._fulfill(); - })(function() { - if (res) { - res[0] = which; - } - return cb.apply(null, arr); - }); - }; - - exports.timeout = function(cb, t, res) { - var tmp; - tmp = []; - _timeout(cb, t, res, tmp); - return tmp[0]; - }; - - _iand = function(cb, res, tmp) { - var ok, ___iced_passed_deferral, __iced_deferrals, __iced_k, - _this = this; - __iced_k = __iced_k_noop; - ___iced_passed_deferral = iced.findDeferral(arguments); - (function(__iced_k) { - __iced_deferrals = new iced.Deferrals(__iced_k, { - parent: ___iced_passed_deferral, - filename: "src/icedlib.coffee", - funcname: "_iand" - }); - tmp[0] = __iced_deferrals.defer({ - assign_fn: (function() { - return function() { - return ok = arguments[0]; - }; - })(), - lineno: 34 - }); - __iced_deferrals._fulfill(); - })(function() { - if (!ok) { - res[0] = false; - } - return cb(); - }); - }; - - exports.iand = function(cb, res) { - var tmp; - tmp = []; - _iand(cb, res, tmp); - return tmp[0]; - }; - - _ior = function(cb, res, tmp) { - var ok, ___iced_passed_deferral, __iced_deferrals, __iced_k, - _this = this; - __iced_k = __iced_k_noop; - ___iced_passed_deferral = iced.findDeferral(arguments); - (function(__iced_k) { - __iced_deferrals = new iced.Deferrals(__iced_k, { - parent: ___iced_passed_deferral, - filename: "src/icedlib.coffee", - funcname: "_ior" - }); - tmp[0] = __iced_deferrals.defer({ - assign_fn: (function() { - return function() { - return ok = arguments[0]; - }; - })(), - lineno: 51 - }); - __iced_deferrals._fulfill(); - })(function() { - if (ok) { - res[0] = true; - } - return cb(); - }); - }; - - exports.ior = function(cb, res) { - var tmp; - tmp = []; - _ior(cb, res, tmp); - return tmp[0]; - }; - - exports.Pipeliner = Pipeliner = (function() { - function Pipeliner(window, delay) { - this.window = window || 1; - this.delay = delay || 0; - this.queue = []; - this.n_out = 0; - this.cb = null; - this[iced_internals["const"].deferrals] = this; - this["defer"] = this._defer; - } - - Pipeliner.prototype.waitInQueue = function(cb) { - var ___iced_passed_deferral, __iced_deferrals, __iced_k, - _this = this; - __iced_k = __iced_k_noop; - ___iced_passed_deferral = iced.findDeferral(arguments); - (function(__iced_k) { - var _results, _while; - _results = []; - _while = function(__iced_k) { - var _break, _continue, _next; - _break = function() { - return __iced_k(_results); - }; - _continue = function() { - return iced.trampoline(function() { - return _while(__iced_k); - }); - }; - _next = function(__iced_next_arg) { - _results.push(__iced_next_arg); - return _continue(); - }; - if (!(_this.n_out >= _this.window)) { - return _break(); - } else { - (function(__iced_k) { - __iced_deferrals = new iced.Deferrals(__iced_k, { - parent: ___iced_passed_deferral, - filename: "src/icedlib.coffee", - funcname: "Pipeliner.waitInQueue" - }); - _this.cb = __iced_deferrals.defer({ - lineno: 88 - }); - __iced_deferrals._fulfill(); - })(_next); - } - }; - _while(__iced_k); - })(function() { - _this.n_out++; - (function(__iced_k) { - if (_this.delay) { - (function(__iced_k) { - __iced_deferrals = new iced.Deferrals(__iced_k, { - parent: ___iced_passed_deferral, - filename: "src/icedlib.coffee", - funcname: "Pipeliner.waitInQueue" - }); - setTimeout(__iced_deferrals.defer({ - lineno: 96 - }), _this.delay); - __iced_deferrals._fulfill(); - })(__iced_k); - } else { - return __iced_k(); - } - })(function() { - return cb(); - }); - }); - }; - - Pipeliner.prototype.__defer = function(out, deferArgs) { - var tmp, voidCb, ___iced_passed_deferral, __iced_deferrals, __iced_k, - _this = this; - __iced_k = __iced_k_noop; - ___iced_passed_deferral = iced.findDeferral(arguments); - (function(__iced_k) { - __iced_deferrals = new iced.Deferrals(__iced_k, { - parent: ___iced_passed_deferral, - filename: "src/icedlib.coffee", - funcname: "Pipeliner.__defer" - }); - voidCb = __iced_deferrals.defer({ - lineno: 109 - }); - out[0] = function() { - var args, _ref; - args = 1 <= arguments.length ? __slice.call(arguments, 0) : []; - if ((_ref = deferArgs.assign_fn) != null) { - _ref.apply(null, args); - } - return voidCb(); - }; - __iced_deferrals._fulfill(); - })(function() { - _this.n_out--; - if (_this.cb) { - tmp = _this.cb; - _this.cb = null; - return tmp(); - } - }); - }; - - Pipeliner.prototype._defer = function(deferArgs) { - var tmp; - tmp = []; - this.__defer(tmp, deferArgs); - return tmp[0]; - }; - - Pipeliner.prototype.flush = function(autocb) { - var ___iced_passed_deferral, __iced_deferrals, __iced_k, _results, _while, - _this = this; - __iced_k = autocb; - ___iced_passed_deferral = iced.findDeferral(arguments); - _results = []; - _while = function(__iced_k) { - var _break, _continue, _next; - _break = function() { - return __iced_k(_results); - }; - _continue = function() { - return iced.trampoline(function() { - return _while(__iced_k); - }); - }; - _next = function(__iced_next_arg) { - _results.push(__iced_next_arg); - return _continue(); - }; - if (!_this.n_out) { - return _break(); - } else { - (function(__iced_k) { - __iced_deferrals = new iced.Deferrals(__iced_k, { - parent: ___iced_passed_deferral, - filename: "src/icedlib.coffee", - funcname: "Pipeliner.flush" - }); - _this.cb = __iced_deferrals.defer({ - lineno: 136 - }); - __iced_deferrals._fulfill(); - })(_next); - } - }; - _while(__iced_k); - }; - - return Pipeliner; - - })(); - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/index.js b/node_modules/iced-coffee-script/lib/coffee-script/index.js deleted file mode 100644 index f8f766b..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/index.js +++ /dev/null @@ -1,13 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var key, val, _ref; - - - - _ref = require('./coffee-script'); - for (key in _ref) { - val = _ref[key]; - exports[key] = val; - } - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/lexer.js b/node_modules/iced-coffee-script/lib/coffee-script/lexer.js deleted file mode 100644 index 5076904..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/lexer.js +++ /dev/null @@ -1,907 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var BOM, BOOL, CALLABLE, CODE, COFFEE_ALIASES, COFFEE_ALIAS_MAP, COFFEE_KEYWORDS, COMMENT, COMPARE, COMPOUND_ASSIGN, HEREDOC, HEREDOC_ILLEGAL, HEREDOC_INDENT, HEREGEX, HEREGEX_OMIT, IDENTIFIER, INDEXABLE, INVERSES, JSTOKEN, JS_FORBIDDEN, JS_KEYWORDS, LINE_BREAK, LINE_CONTINUER, LOGIC, Lexer, MATH, MULTILINER, MULTI_DENT, NOT_REGEX, NOT_SPACED_REGEX, NUMBER, OPERATOR, REGEX, RELATION, RESERVED, Rewriter, SHIFT, SIMPLESTR, STRICT_PROSCRIBED, TRAILING_SPACES, UNARY, WHITESPACE, compact, count, invertLiterate, key, last, locationDataToString, repeat, starts, throwSyntaxError, _ref, _ref1, - __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; }; - - - - _ref = require('./rewriter'), Rewriter = _ref.Rewriter, INVERSES = _ref.INVERSES; - - _ref1 = require('./helpers'), count = _ref1.count, starts = _ref1.starts, compact = _ref1.compact, last = _ref1.last, repeat = _ref1.repeat, invertLiterate = _ref1.invertLiterate, locationDataToString = _ref1.locationDataToString, throwSyntaxError = _ref1.throwSyntaxError; - - exports.Lexer = Lexer = (function() { - function Lexer() {} - - Lexer.prototype.tokenize = function(code, opts) { - var consumed, i, tag, _ref2; - if (opts == null) { - opts = {}; - } - this.literate = opts.literate; - this.indent = 0; - this.baseIndent = 0; - this.indebt = 0; - this.outdebt = 0; - this.indents = []; - this.ends = []; - this.tokens = []; - this.chunkLine = opts.line || 0; - this.chunkColumn = opts.column || 0; - code = this.clean(code); - i = 0; - while (this.chunk = code.slice(i)) { - consumed = this.identifierToken() || this.commentToken() || this.whitespaceToken() || this.lineToken() || this.heredocToken() || this.stringToken() || this.numberToken() || this.regexToken() || this.jsToken() || this.literalToken(); - _ref2 = this.getLineAndColumnFromChunk(consumed), this.chunkLine = _ref2[0], this.chunkColumn = _ref2[1]; - i += consumed; - } - this.closeIndentation(); - if (tag = this.ends.pop()) { - this.error("missing " + tag); - } - if (opts.rewrite === false) { - return this.tokens; - } - return (new Rewriter).rewrite(this.tokens); - }; - - Lexer.prototype.clean = function(code) { - if (code.charCodeAt(0) === BOM) { - code = code.slice(1); - } - code = code.replace(/\r/g, '').replace(TRAILING_SPACES, ''); - if (WHITESPACE.test(code)) { - code = "\n" + code; - this.chunkLine--; - } - if (this.literate) { - code = invertLiterate(code); - } - return code; - }; - - Lexer.prototype.identifierToken = function() { - var colon, colonOffset, forcedIdentifier, id, idLength, input, match, poppedToken, prev, tag, tagToken, _ref2, _ref3, _ref4; - if (!(match = IDENTIFIER.exec(this.chunk))) { - return 0; - } - input = match[0], id = match[1], colon = match[2]; - idLength = id.length; - poppedToken = void 0; - if (id === 'own' && this.tag() === 'FOR') { - this.token('OWN', id); - return id.length; - } - forcedIdentifier = colon || (prev = last(this.tokens)) && (((_ref2 = prev[0]) === '.' || _ref2 === '?.' || _ref2 === '::' || _ref2 === '?::') || !prev.spaced && prev[0] === '@') && id !== 'defer'; - tag = 'IDENTIFIER'; - if (!forcedIdentifier && (__indexOf.call(JS_KEYWORDS, id) >= 0 || __indexOf.call(COFFEE_KEYWORDS, id) >= 0)) { - tag = id.toUpperCase(); - if (tag === 'WHEN' && (_ref3 = this.tag(), __indexOf.call(LINE_BREAK, _ref3) >= 0)) { - tag = 'LEADING_WHEN'; - } else if (tag === 'FOR') { - this.seenFor = true; - } else if (tag === 'UNLESS') { - tag = 'IF'; - } else if (__indexOf.call(UNARY, tag) >= 0) { - tag = 'UNARY'; - } else if (__indexOf.call(RELATION, tag) >= 0) { - if (tag !== 'INSTANCEOF' && this.seenFor) { - tag = 'FOR' + tag; - this.seenFor = false; - } else { - tag = 'RELATION'; - if (this.value() === '!') { - poppedToken = this.tokens.pop(); - id = '!' + id; - } - } - } - } - if (__indexOf.call(JS_FORBIDDEN, id) >= 0) { - if (forcedIdentifier) { - tag = 'IDENTIFIER'; - id = new String(id); - id.reserved = true; - } else if (__indexOf.call(RESERVED, id) >= 0) { - this.error("reserved word \"" + id + "\""); - } - } - if (!forcedIdentifier) { - if (__indexOf.call(COFFEE_ALIASES, id) >= 0) { - id = COFFEE_ALIAS_MAP[id]; - } - tag = (function() { - switch (id) { - case '!': - return 'UNARY'; - case '==': - case '!=': - return 'COMPARE'; - case '&&': - case '||': - return 'LOGIC'; - case 'true': - case 'false': - return 'BOOL'; - case 'break': - case 'continue': - return 'STATEMENT'; - default: - return tag; - } - })(); - } - tagToken = this.token(tag, id, 0, idLength); - if (poppedToken) { - _ref4 = [poppedToken[2].first_line, poppedToken[2].first_column], tagToken[2].first_line = _ref4[0], tagToken[2].first_column = _ref4[1]; - } - if (colon) { - colonOffset = input.lastIndexOf(':'); - this.token(':', ':', colonOffset, colon.length); - } - return input.length; - }; - - Lexer.prototype.numberToken = function() { - var binaryLiteral, lexedLength, match, number, octalLiteral; - if (!(match = NUMBER.exec(this.chunk))) { - return 0; - } - number = match[0]; - if (/^0[BOX]/.test(number)) { - this.error("radix prefix '" + number + "' must be lowercase"); - } else if (/E/.test(number) && !/^0x/.test(number)) { - this.error("exponential notation '" + number + "' must be indicated with a lowercase 'e'"); - } else if (/^0\d*[89]/.test(number)) { - this.error("decimal literal '" + number + "' must not be prefixed with '0'"); - } else if (/^0\d+/.test(number)) { - this.error("octal literal '" + number + "' must be prefixed with '0o'"); - } - lexedLength = number.length; - if (octalLiteral = /^0o([0-7]+)/.exec(number)) { - number = '0x' + parseInt(octalLiteral[1], 8).toString(16); - } - if (binaryLiteral = /^0b([01]+)/.exec(number)) { - number = '0x' + parseInt(binaryLiteral[1], 2).toString(16); - } - this.token('NUMBER', number, 0, lexedLength); - return lexedLength; - }; - - Lexer.prototype.stringToken = function() { - var match, octalEsc, string; - switch (this.chunk.charAt(0)) { - case "'": - if (!(match = SIMPLESTR.exec(this.chunk))) { - return 0; - } - string = match[0]; - this.token('STRING', string.replace(MULTILINER, '\\\n'), 0, string.length); - break; - case '"': - if (!(string = this.balancedString(this.chunk, '"'))) { - return 0; - } - if (0 < string.indexOf('#{', 1)) { - this.interpolateString(string.slice(1, -1), { - strOffset: 1, - lexedLength: string.length - }); - } else { - this.token('STRING', this.escapeLines(string, 0, string.length)); - } - break; - default: - return 0; - } - if (octalEsc = /^(?:\\.|[^\\])*\\(?:0[0-7]|[1-7])/.test(string)) { - this.error("octal escape sequences " + string + " are not allowed"); - } - return string.length; - }; - - Lexer.prototype.heredocToken = function() { - var doc, heredoc, match, quote; - if (!(match = HEREDOC.exec(this.chunk))) { - return 0; - } - heredoc = match[0]; - quote = heredoc.charAt(0); - doc = this.sanitizeHeredoc(match[2], { - quote: quote, - indent: null - }); - if (quote === '"' && 0 <= doc.indexOf('#{')) { - this.interpolateString(doc, { - heredoc: true, - strOffset: 3, - lexedLength: heredoc.length - }); - } else { - this.token('STRING', this.makeString(doc, quote, true), 0, heredoc.length); - } - return heredoc.length; - }; - - Lexer.prototype.commentToken = function() { - var comment, here, match; - if (!(match = this.chunk.match(COMMENT))) { - return 0; - } - comment = match[0], here = match[1]; - if (here) { - this.token('HERECOMMENT', this.sanitizeHeredoc(here, { - herecomment: true, - indent: repeat(' ', this.indent) - }), 0, comment.length); - } - return comment.length; - }; - - Lexer.prototype.jsToken = function() { - var match, script; - if (!(this.chunk.charAt(0) === '`' && (match = JSTOKEN.exec(this.chunk)))) { - return 0; - } - this.token('JS', (script = match[0]).slice(1, -1), 0, script.length); - return script.length; - }; - - Lexer.prototype.regexToken = function() { - var flags, length, match, prev, regex, _ref2, _ref3; - if (this.chunk.charAt(0) !== '/') { - return 0; - } - if (match = HEREGEX.exec(this.chunk)) { - length = this.heregexToken(match); - return length; - } - prev = last(this.tokens); - if (prev && (_ref2 = prev[0], __indexOf.call((prev.spaced ? NOT_REGEX : NOT_SPACED_REGEX), _ref2) >= 0)) { - return 0; - } - if (!(match = REGEX.exec(this.chunk))) { - return 0; - } - _ref3 = match, match = _ref3[0], regex = _ref3[1], flags = _ref3[2]; - if (regex.slice(0, 2) === '/*') { - this.error('regular expressions cannot begin with `*`'); - } - if (regex === '//') { - regex = '/(?:)/'; - } - this.token('REGEX', "" + regex + flags, 0, match.length); - return match.length; - }; - - Lexer.prototype.heregexToken = function(match) { - var body, flags, flagsOffset, heregex, plusToken, prev, re, tag, token, tokens, value, _i, _len, _ref2, _ref3, _ref4; - heregex = match[0], body = match[1], flags = match[2]; - if (0 > body.indexOf('#{')) { - re = body.replace(HEREGEX_OMIT, '').replace(/\//g, '\\/'); - if (re.match(/^\*/)) { - this.error('regular expressions cannot begin with `*`'); - } - this.token('REGEX', "/" + (re || '(?:)') + "/" + flags, 0, heregex.length); - return heregex.length; - } - this.token('IDENTIFIER', 'RegExp', 0, 0); - this.token('CALL_START', '(', 0, 0); - tokens = []; - _ref2 = this.interpolateString(body, { - regex: true - }); - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - token = _ref2[_i]; - tag = token[0], value = token[1]; - if (tag === 'TOKENS') { - tokens.push.apply(tokens, value); - } else if (tag === 'NEOSTRING') { - if (!(value = value.replace(HEREGEX_OMIT, ''))) { - continue; - } - value = value.replace(/\\/g, '\\\\'); - token[0] = 'STRING'; - token[1] = this.makeString(value, '"', true); - tokens.push(token); - } else { - this.error("Unexpected " + tag); - } - prev = last(this.tokens); - plusToken = ['+', '+']; - plusToken[2] = prev[2]; - tokens.push(plusToken); - } - tokens.pop(); - if (((_ref3 = tokens[0]) != null ? _ref3[0] : void 0) !== 'STRING') { - this.token('STRING', '""', 0, 0); - this.token('+', '+', 0, 0); - } - (_ref4 = this.tokens).push.apply(_ref4, tokens); - if (flags) { - flagsOffset = heregex.lastIndexOf(flags); - this.token(',', ',', flagsOffset, 0); - this.token('STRING', '"' + flags + '"', flagsOffset, flags.length); - } - this.token(')', ')', heregex.length - 1, 0); - return heregex.length; - }; - - Lexer.prototype.lineToken = function() { - var diff, indent, match, noNewlines, size; - if (!(match = MULTI_DENT.exec(this.chunk))) { - return 0; - } - indent = match[0]; - this.seenFor = false; - size = indent.length - 1 - indent.lastIndexOf('\n'); - noNewlines = this.unfinished(); - if (size - this.indebt === this.indent) { - if (noNewlines) { - this.suppressNewlines(); - } else { - this.newlineToken(0); - } - return indent.length; - } - if (size > this.indent) { - if (noNewlines) { - this.indebt = size - this.indent; - this.suppressNewlines(); - return indent.length; - } - if (!this.tokens.length) { - this.baseIndent = this.indent = size; - return indent.length; - } - diff = size - this.indent + this.outdebt; - this.token('INDENT', diff, indent.length - size, size); - this.indents.push(diff); - this.ends.push('OUTDENT'); - this.outdebt = this.indebt = 0; - } else if (size < this.baseIndent) { - this.error('missing indentation', indent.length); - } else { - this.indebt = 0; - this.outdentToken(this.indent - size, noNewlines, indent.length); - } - this.indent = size; - return indent.length; - }; - - Lexer.prototype.outdentToken = function(moveOut, noNewlines, outdentLength) { - var dent, len; - while (moveOut > 0) { - len = this.indents.length - 1; - if (this.indents[len] === void 0) { - moveOut = 0; - } else if (this.indents[len] === this.outdebt) { - moveOut -= this.outdebt; - this.outdebt = 0; - } else if (this.indents[len] < this.outdebt) { - this.outdebt -= this.indents[len]; - moveOut -= this.indents[len]; - } else { - dent = this.indents.pop() + this.outdebt; - moveOut -= dent; - this.outdebt = 0; - this.pair('OUTDENT'); - this.token('OUTDENT', dent, 0, outdentLength); - } - } - if (dent) { - this.outdebt -= moveOut; - } - while (this.value() === ';') { - this.tokens.pop(); - } - if (!(this.tag() === 'TERMINATOR' || noNewlines)) { - this.token('TERMINATOR', '\n', outdentLength, 0); - } - return this; - }; - - Lexer.prototype.whitespaceToken = function() { - var match, nline, prev; - if (!((match = WHITESPACE.exec(this.chunk)) || (nline = this.chunk.charAt(0) === '\n'))) { - return 0; - } - prev = last(this.tokens); - if (prev) { - prev[match ? 'spaced' : 'newLine'] = true; - } - if (match) { - return match[0].length; - } else { - return 0; - } - }; - - Lexer.prototype.newlineToken = function(offset) { - while (this.value() === ';') { - this.tokens.pop(); - } - if (this.tag() !== 'TERMINATOR') { - this.token('TERMINATOR', '\n', offset, 0); - } - return this; - }; - - Lexer.prototype.suppressNewlines = function() { - if (this.value() === '\\') { - this.tokens.pop(); - } - return this; - }; - - Lexer.prototype.literalToken = function() { - var match, prev, tag, value, _ref2, _ref3, _ref4, _ref5; - if (match = OPERATOR.exec(this.chunk)) { - value = match[0]; - if (CODE.test(value)) { - this.tagParameters(); - } - } else { - value = this.chunk.charAt(0); - } - tag = value; - prev = last(this.tokens); - if (value === '=' && prev) { - if (!prev[1].reserved && (_ref2 = prev[1], __indexOf.call(JS_FORBIDDEN, _ref2) >= 0)) { - this.error("reserved word \"" + (this.value()) + "\" can't be assigned"); - } - if ((_ref3 = prev[1]) === '||' || _ref3 === '&&') { - prev[0] = 'COMPOUND_ASSIGN'; - prev[1] += '='; - return value.length; - } - } - if (value === ';') { - this.seenFor = false; - tag = 'TERMINATOR'; - } else if (__indexOf.call(MATH, value) >= 0) { - tag = 'MATH'; - } else if (__indexOf.call(COMPARE, value) >= 0) { - tag = 'COMPARE'; - } else if (__indexOf.call(COMPOUND_ASSIGN, value) >= 0) { - tag = 'COMPOUND_ASSIGN'; - } else if (__indexOf.call(UNARY, value) >= 0) { - tag = 'UNARY'; - } else if (__indexOf.call(SHIFT, value) >= 0) { - tag = 'SHIFT'; - } else if (__indexOf.call(LOGIC, value) >= 0 || value === '?' && (prev != null ? prev.spaced : void 0)) { - tag = 'LOGIC'; - } else if (prev && !prev.spaced) { - if (value === '(' && (_ref4 = prev[0], __indexOf.call(CALLABLE, _ref4) >= 0)) { - if (prev[0] === '?') { - prev[0] = 'FUNC_EXIST'; - } - tag = 'CALL_START'; - } else if (value === '[' && (_ref5 = prev[0], __indexOf.call(INDEXABLE, _ref5) >= 0)) { - tag = 'INDEX_START'; - switch (prev[0]) { - case '?': - prev[0] = 'INDEX_SOAK'; - } - } - } - switch (value) { - case '(': - case '{': - case '[': - this.ends.push(INVERSES[value]); - break; - case ')': - case '}': - case ']': - this.pair(value); - } - this.token(tag, value); - return value.length; - }; - - Lexer.prototype.sanitizeHeredoc = function(doc, options) { - var attempt, herecomment, indent, match, _ref2; - indent = options.indent, herecomment = options.herecomment; - if (herecomment) { - if (HEREDOC_ILLEGAL.test(doc)) { - this.error("block comment cannot contain \"*/\", starting"); - } - if (doc.indexOf('\n') < 0) { - return doc; - } - } else { - while (match = HEREDOC_INDENT.exec(doc)) { - attempt = match[1]; - if (indent === null || (0 < (_ref2 = attempt.length) && _ref2 < indent.length)) { - indent = attempt; - } - } - } - if (indent) { - doc = doc.replace(RegExp("\\n" + indent, "g"), '\n'); - } - if (!herecomment) { - doc = doc.replace(/^\n/, ''); - } - return doc; - }; - - Lexer.prototype.tagParameters = function() { - var i, stack, tok, tokens; - if (this.tag() !== ')') { - return this; - } - stack = []; - tokens = this.tokens; - i = tokens.length; - tokens[--i][0] = 'PARAM_END'; - while (tok = tokens[--i]) { - switch (tok[0]) { - case ')': - stack.push(tok); - break; - case '(': - case 'CALL_START': - if (stack.length) { - stack.pop(); - } else if (tok[0] === '(') { - tok[0] = 'PARAM_START'; - return this; - } else { - return this; - } - } - } - return this; - }; - - Lexer.prototype.closeIndentation = function() { - return this.outdentToken(this.indent); - }; - - Lexer.prototype.balancedString = function(str, end) { - var continueCount, i, letter, match, prev, stack, _i, _ref2; - continueCount = 0; - stack = [end]; - for (i = _i = 1, _ref2 = str.length; 1 <= _ref2 ? _i < _ref2 : _i > _ref2; i = 1 <= _ref2 ? ++_i : --_i) { - if (continueCount) { - --continueCount; - continue; - } - switch (letter = str.charAt(i)) { - case '\\': - ++continueCount; - continue; - case end: - stack.pop(); - if (!stack.length) { - return str.slice(0, +i + 1 || 9e9); - } - end = stack[stack.length - 1]; - continue; - } - if (end === '}' && (letter === '"' || letter === "'")) { - stack.push(end = letter); - } else if (end === '}' && letter === '/' && (match = HEREGEX.exec(str.slice(i)) || REGEX.exec(str.slice(i)))) { - continueCount += match[0].length - 1; - } else if (end === '}' && letter === '{') { - stack.push(end = '}'); - } else if (end === '"' && prev === '#' && letter === '{') { - stack.push(end = '}'); - } - prev = letter; - } - return this.error("missing " + (stack.pop()) + ", starting"); - }; - - Lexer.prototype.interpolateString = function(str, options) { - var column, expr, heredoc, i, inner, interpolated, len, letter, lexedLength, line, locationToken, nested, offsetInChunk, pi, plusToken, popped, regex, rparen, strOffset, tag, token, tokens, value, _i, _len, _ref2, _ref3, _ref4; - if (options == null) { - options = {}; - } - heredoc = options.heredoc, regex = options.regex, offsetInChunk = options.offsetInChunk, strOffset = options.strOffset, lexedLength = options.lexedLength; - offsetInChunk = offsetInChunk || 0; - strOffset = strOffset || 0; - lexedLength = lexedLength || str.length; - if (heredoc && str.length > 0 && str[0] === '\n') { - str = str.slice(1); - strOffset++; - } - tokens = []; - pi = 0; - i = -1; - while (letter = str.charAt(i += 1)) { - if (letter === '\\') { - i += 1; - continue; - } - if (!(letter === '#' && str.charAt(i + 1) === '{' && (expr = this.balancedString(str.slice(i + 1), '}')))) { - continue; - } - if (pi < i) { - tokens.push(this.makeToken('NEOSTRING', str.slice(pi, i), strOffset + pi)); - } - inner = expr.slice(1, -1); - if (inner.length) { - _ref2 = this.getLineAndColumnFromChunk(strOffset + i + 1), line = _ref2[0], column = _ref2[1]; - nested = new Lexer().tokenize(inner, { - line: line, - column: column, - rewrite: false - }); - popped = nested.pop(); - if (((_ref3 = nested[0]) != null ? _ref3[0] : void 0) === 'TERMINATOR') { - popped = nested.shift(); - } - if (len = nested.length) { - if (len > 1) { - nested.unshift(this.makeToken('(', '(', strOffset + i + 1, 0)); - nested.push(this.makeToken(')', ')', strOffset + i + 1 + inner.length, 0)); - } - tokens.push(['TOKENS', nested]); - } - } - i += expr.length; - pi = i + 1; - } - if ((i > pi && pi < str.length)) { - tokens.push(this.makeToken('NEOSTRING', str.slice(pi), strOffset + pi)); - } - if (regex) { - return tokens; - } - if (!tokens.length) { - return this.token('STRING', '""', offsetInChunk, lexedLength); - } - if (tokens[0][0] !== 'NEOSTRING') { - tokens.unshift(this.makeToken('NEOSTRING', '', offsetInChunk)); - } - if (interpolated = tokens.length > 1) { - this.token('(', '(', offsetInChunk, 0); - } - for (i = _i = 0, _len = tokens.length; _i < _len; i = ++_i) { - token = tokens[i]; - tag = token[0], value = token[1]; - if (i) { - if (i) { - plusToken = this.token('+', '+'); - } - locationToken = tag === 'TOKENS' ? value[0] : token; - plusToken[2] = { - first_line: locationToken[2].first_line, - first_column: locationToken[2].first_column, - last_line: locationToken[2].first_line, - last_column: locationToken[2].first_column - }; - } - if (tag === 'TOKENS') { - (_ref4 = this.tokens).push.apply(_ref4, value); - } else if (tag === 'NEOSTRING') { - token[0] = 'STRING'; - token[1] = this.makeString(value, '"', heredoc); - this.tokens.push(token); - } else { - this.error("Unexpected " + tag); - } - } - if (interpolated) { - rparen = this.makeToken(')', ')', offsetInChunk + lexedLength, 0); - rparen.stringEnd = true; - this.tokens.push(rparen); - } - return tokens; - }; - - Lexer.prototype.pair = function(tag) { - var size, wanted; - if (tag !== (wanted = last(this.ends))) { - if ('OUTDENT' !== wanted) { - this.error("unmatched " + tag); - } - this.indent -= size = last(this.indents); - this.outdentToken(size, true); - return this.pair(tag); - } - return this.ends.pop(); - }; - - Lexer.prototype.getLineAndColumnFromChunk = function(offset) { - var column, lineCount, lines, string; - if (offset === 0) { - return [this.chunkLine, this.chunkColumn]; - } - if (offset >= this.chunk.length) { - string = this.chunk; - } else { - string = this.chunk.slice(0, +(offset - 1) + 1 || 9e9); - } - lineCount = count(string, '\n'); - column = this.chunkColumn; - if (lineCount > 0) { - lines = string.split('\n'); - column = last(lines).length; - } else { - column += string.length; - } - return [this.chunkLine + lineCount, column]; - }; - - Lexer.prototype.makeToken = function(tag, value, offsetInChunk, length) { - var lastCharacter, locationData, token, _ref2, _ref3; - if (offsetInChunk == null) { - offsetInChunk = 0; - } - if (length == null) { - length = value.length; - } - locationData = {}; - _ref2 = this.getLineAndColumnFromChunk(offsetInChunk), locationData.first_line = _ref2[0], locationData.first_column = _ref2[1]; - lastCharacter = Math.max(0, length - 1); - _ref3 = this.getLineAndColumnFromChunk(offsetInChunk + lastCharacter), locationData.last_line = _ref3[0], locationData.last_column = _ref3[1]; - token = [tag, value, locationData]; - return token; - }; - - Lexer.prototype.token = function(tag, value, offsetInChunk, length) { - var token; - token = this.makeToken(tag, value, offsetInChunk, length); - this.tokens.push(token); - return token; - }; - - Lexer.prototype.tag = function(index, tag) { - var tok; - return (tok = last(this.tokens, index)) && (tag ? tok[0] = tag : tok[0]); - }; - - Lexer.prototype.value = function(index, val) { - var tok; - return (tok = last(this.tokens, index)) && (val ? tok[1] = val : tok[1]); - }; - - Lexer.prototype.unfinished = function() { - var _ref2; - return LINE_CONTINUER.test(this.chunk) || ((_ref2 = this.tag()) === '\\' || _ref2 === '.' || _ref2 === '?.' || _ref2 === '?::' || _ref2 === 'UNARY' || _ref2 === 'MATH' || _ref2 === '+' || _ref2 === '-' || _ref2 === 'SHIFT' || _ref2 === 'RELATION' || _ref2 === 'COMPARE' || _ref2 === 'LOGIC' || _ref2 === 'THROW' || _ref2 === 'EXTENDS'); - }; - - Lexer.prototype.escapeLines = function(str, heredoc) { - return str.replace(MULTILINER, heredoc ? '\\n' : ''); - }; - - Lexer.prototype.makeString = function(body, quote, heredoc) { - if (!body) { - return quote + quote; - } - body = body.replace(/\\([\s\S])/g, function(match, contents) { - if (contents === '\n' || contents === quote) { - return contents; - } else { - return match; - } - }); - body = body.replace(RegExp("" + quote, "g"), '\\$&'); - return quote + this.escapeLines(body, heredoc) + quote; - }; - - Lexer.prototype.error = function(message, offset) { - var first_column, first_line, _ref2; - if (offset == null) { - offset = 0; - } - _ref2 = this.getLineAndColumnFromChunk(offset), first_line = _ref2[0], first_column = _ref2[1]; - return throwSyntaxError(message, { - first_line: first_line, - first_column: first_column - }); - }; - - return Lexer; - - })(); - - JS_KEYWORDS = ['true', 'false', 'null', 'this', 'new', 'delete', 'typeof', 'in', 'instanceof', 'return', 'throw', 'break', 'continue', 'debugger', 'if', 'else', 'switch', 'for', 'while', 'do', 'try', 'catch', 'finally', 'class', 'extends', 'super']; - - COFFEE_KEYWORDS = ['undefined', 'then', 'unless', 'until', 'loop', 'of', 'by', 'when']; - - COFFEE_KEYWORDS = COFFEE_KEYWORDS.concat(['await', 'defer']); - - COFFEE_ALIAS_MAP = { - and: '&&', - or: '||', - is: '==', - isnt: '!=', - not: '!', - yes: 'true', - no: 'false', - on: 'true', - off: 'false' - }; - - COFFEE_ALIASES = (function() { - var _results; - _results = []; - for (key in COFFEE_ALIAS_MAP) { - _results.push(key); - } - return _results; - })(); - - COFFEE_KEYWORDS = COFFEE_KEYWORDS.concat(COFFEE_ALIASES); - - RESERVED = ['case', 'default', 'function', 'var', 'void', 'with', 'const', 'let', 'enum', 'export', 'import', 'native', '__hasProp', '__extends', '__slice', '__bind', '__indexOf', 'implements', 'interface', 'package', 'private', 'protected', 'public', 'static', 'yield']; - - STRICT_PROSCRIBED = ['arguments', 'eval']; - - JS_FORBIDDEN = JS_KEYWORDS.concat(RESERVED).concat(STRICT_PROSCRIBED); - - exports.RESERVED = RESERVED.concat(JS_KEYWORDS).concat(COFFEE_KEYWORDS).concat(STRICT_PROSCRIBED); - - exports.STRICT_PROSCRIBED = STRICT_PROSCRIBED; - - BOM = 65279; - - IDENTIFIER = /^([$A-Za-z_\x7f-\uffff][$\w\x7f-\uffff]*)([^\n\S]*:(?!:))?/; - - NUMBER = /^0b[01]+|^0o[0-7]+|^0x[\da-f]+|^\d*\.?\d+(?:e[+-]?\d+)?/i; - - HEREDOC = /^("""|''')([\s\S]*?)(?:\n[^\n\S]*)?\1/; - - OPERATOR = /^(?:[-=]>|[-+*\/%<>&|^!?=]=|>>>=?|([-+:])\1|([&|<>])\2=?|\?(\.|::)|\.{2,3})/; - - WHITESPACE = /^[^\n\S]+/; - - COMMENT = /^###([^#][\s\S]*?)(?:###[^\n\S]*|(?:###)$)|^(?:\s*#(?!##[^#]).*)+/; - - CODE = /^[-=]>/; - - MULTI_DENT = /^(?:\n[^\n\S]*)+/; - - SIMPLESTR = /^'[^\\']*(?:\\.[^\\']*)*'/; - - JSTOKEN = /^`[^\\`]*(?:\\.[^\\`]*)*`/; - - REGEX = /^(\/(?![\s=])[^[\/\n\\]*(?:(?:\\[\s\S]|\[[^\]\n\\]*(?:\\[\s\S][^\]\n\\]*)*])[^[\/\n\\]*)*\/)([imgy]{0,4})(?!\w)/; - - HEREGEX = /^\/{3}([\s\S]+?)\/{3}([imgy]{0,4})(?!\w)/; - - HEREGEX_OMIT = /\s+(?:#.*)?/g; - - MULTILINER = /\n/g; - - HEREDOC_INDENT = /\n+([^\n\S]*)/g; - - HEREDOC_ILLEGAL = /\*\//; - - LINE_CONTINUER = /^\s*(?:,|\??\.(?![.\d])|::)/; - - TRAILING_SPACES = /\s+$/; - - COMPOUND_ASSIGN = ['-=', '+=', '/=', '*=', '%=', '||=', '&&=', '?=', '<<=', '>>=', '>>>=', '&=', '^=', '|=']; - - UNARY = ['!', '~', 'NEW', 'TYPEOF', 'DELETE', 'DO']; - - LOGIC = ['&&', '||', '&', '|', '^']; - - SHIFT = ['<<', '>>', '>>>']; - - COMPARE = ['==', '!=', '<', '>', '<=', '>=']; - - MATH = ['*', '/', '%']; - - RELATION = ['IN', 'OF', 'INSTANCEOF']; - - BOOL = ['TRUE', 'FALSE']; - - NOT_REGEX = ['NUMBER', 'REGEX', 'BOOL', 'NULL', 'UNDEFINED', '++', '--']; - - NOT_SPACED_REGEX = NOT_REGEX.concat(')', '}', 'THIS', 'IDENTIFIER', 'STRING', ']'); - - CALLABLE = ['IDENTIFIER', 'STRING', 'REGEX', ')', ']', '}', '?', '::', '@', 'THIS', 'SUPER']; - - INDEXABLE = CALLABLE.concat('NUMBER', 'BOOL', 'NULL', 'UNDEFINED'); - - LINE_BREAK = ['INDENT', 'OUTDENT', 'TERMINATOR']; - - CALLABLE.push('DEFER'); - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/nodes.js b/node_modules/iced-coffee-script/lib/coffee-script/nodes.js deleted file mode 100644 index 9460036..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/nodes.js +++ /dev/null @@ -1,4280 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var Access, Arr, Assign, Await, Base, Block, Call, Class, Closure, Code, CodeFragment, Comment, CpsCascade, Defer, Existence, Extends, For, IDENTIFIER, IDENTIFIER_STR, IS_STRING, IcedReturnValue, IcedRuntime, IcedTailCall, If, In, Index, InlineRuntime, LEVEL_ACCESS, LEVEL_COND, LEVEL_LIST, LEVEL_OP, LEVEL_PAREN, LEVEL_TOP, Literal, METHOD_DEF, NEGATE, NO, NULL, Obj, Op, Param, Parens, RESERVED, Range, Return, SIMPLENUM, STRICT_PROSCRIBED, Scope, Slice, Slot, Splat, Switch, TAB, THIS, Throw, Try, UTILITIES, Value, While, YES, addLocationDataFn, compact, del, ends, extend, flatten, fragmentsToText, iced, last, locationDataToString, merge, multident, some, starts, throwSyntaxError, unfoldSoak, utility, _ref, _ref1, _ref2, _ref3, - __hasProp = {}.hasOwnProperty, - __extends = function(child, parent) { for (var key in parent) { if (__hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; }, - __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; }, - __slice = [].slice; - - - - Error.stackTraceLimit = Infinity; - - Scope = require('./scope').Scope; - - _ref = require('./lexer'), RESERVED = _ref.RESERVED, STRICT_PROSCRIBED = _ref.STRICT_PROSCRIBED; - - iced = require('./iced'); - - _ref1 = require('./helpers'), compact = _ref1.compact, flatten = _ref1.flatten, extend = _ref1.extend, merge = _ref1.merge, del = _ref1.del, starts = _ref1.starts, ends = _ref1.ends, last = _ref1.last, some = _ref1.some, addLocationDataFn = _ref1.addLocationDataFn, locationDataToString = _ref1.locationDataToString, throwSyntaxError = _ref1.throwSyntaxError; - - exports.extend = extend; - - exports.addLocationDataFn = addLocationDataFn; - - YES = function() { - return true; - }; - - NO = function() { - return false; - }; - - THIS = function() { - return this; - }; - - NEGATE = function() { - this.negated = !this.negated; - return this; - }; - - NULL = function() { - return new Value(new Literal('null')); - }; - - exports.CodeFragment = CodeFragment = (function() { - function CodeFragment(parent, code) { - var _ref2; - this.code = "" + code; - this.locationData = parent != null ? parent.locationData : void 0; - this.type = (parent != null ? (_ref2 = parent.constructor) != null ? _ref2.name : void 0 : void 0) || 'unknown'; - } - - CodeFragment.prototype.toString = function() { - return "" + this.code + (this.locationData ? ": " + locationDataToString(this.locationData) : ''); - }; - - return CodeFragment; - - })(); - - fragmentsToText = function(fragments) { - var fragment; - return ((function() { - var _i, _len, _results; - _results = []; - for (_i = 0, _len = fragments.length; _i < _len; _i++) { - fragment = fragments[_i]; - _results.push(fragment.code); - } - return _results; - })()).join(''); - }; - - exports.Base = Base = (function() { - function Base() { - this.icedContinuationBlock = null; - this.icedLoopFlag = false; - this.icedNodeFlag = false; - this.icedGotCpsSplitFlag = false; - this.icedCpsPivotFlag = false; - this.icedHasAutocbFlag = false; - this.icedFoundArguments = false; - this.icedParentAwait = null; - this.icedCallContinuationFlag = false; - } - - Base.prototype.compile = function(o, lvl) { - return fragmentsToText(this.compileToFragments(o, lvl)); - }; - - Base.prototype.compileToFragments = function(o, lvl) { - var node; - o = extend({}, o); - if (lvl) { - o.level = lvl; - } - node = this.unfoldSoak(o) || this; - node.tab = o.indent; - if (node.icedHasContinuation() && !node.icedGotCpsSplitFlag) { - return node.icedCompileCps(o); - } else if (o.level === LEVEL_TOP || !node.isStatement(o)) { - return node.compileNode(o); - } else { - return node.compileClosure(o); - } - }; - - Base.prototype.compileClosure = function(o) { - var jumpNode; - if (jumpNode = this.jumps()) { - jumpNode.error('cannot use a pure statement in an expression'); - } - o.sharedScope = true; - this.icedClearAutocbFlags(); - return Closure.wrap(this).compileNode(o); - }; - - Base.prototype.cache = function(o, level, reused) { - var ref, sub; - if (!this.isComplex()) { - ref = level ? this.compileToFragments(o, level) : this; - return [ref, ref]; - } else { - ref = new Literal(reused || o.scope.freeVariable('ref')); - sub = new Assign(ref, this); - if (level) { - return [sub.compileToFragments(o, level), [this.makeCode(ref.value)]]; - } else { - return [sub, ref]; - } - } - }; - - Base.prototype.cacheToCodeFragments = function(cacheValues) { - return [fragmentsToText(cacheValues[0]), fragmentsToText(cacheValues[1])]; - }; - - Base.prototype.makeReturn = function(res) { - var me; - me = this.unwrapAll(); - if (res) { - return new Call(new Literal("" + res + ".push"), [me]); - } else { - return new Return(me, this.icedHasAutocbFlag); - } - }; - - Base.prototype.contains = function(pred) { - var node; - node = void 0; - this.traverseChildren(false, function(n) { - if (pred(n)) { - node = n; - return false; - } - }); - return node; - }; - - Base.prototype.lastNonComment = function(list) { - var i; - i = list.length; - while (i--) { - if (!(list[i] instanceof Comment)) { - return list[i]; - } - } - return null; - }; - - Base.prototype.toString = function(idt, name) { - var extras, tree; - if (idt == null) { - idt = ''; - } - if (name == null) { - name = this.constructor.name; - } - extras = []; - if (this.icedNodeFlag) { - extras.push("A"); - } - if (this.icedLoopFlag) { - extras.push("L"); - } - if (this.icedCpsPivotFlag) { - extras.push("P"); - } - if (this.icedHasAutocbFlag) { - extras.push("C"); - } - if (this.icedParentAwait) { - extras.push("D"); - } - if (this.icedFoundArguments) { - extras.push("G"); - } - if (extras.length) { - extras = " (" + extras.join('') + ")"; - } - tree = '\n' + idt + name; - tree = '\n' + idt + name; - if (this.soak) { - tree += '?'; - } - tree += extras; - this.eachChild(function(node) { - return tree += node.toString(idt + TAB); - }); - if (this.icedContinuationBlock) { - idt += TAB; - tree += '\n' + idt + "Continuation"; - tree += this.icedContinuationBlock.toString(idt + TAB); - } - return tree; - }; - - Base.prototype.eachChild = function(func) { - var attr, child, _i, _j, _len, _len1, _ref2, _ref3; - if (!this.children) { - return this; - } - _ref2 = this.children; - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - attr = _ref2[_i]; - if (this[attr]) { - _ref3 = flatten([this[attr]]); - for (_j = 0, _len1 = _ref3.length; _j < _len1; _j++) { - child = _ref3[_j]; - if (func(child) === false) { - return this; - } - } - } - } - return this; - }; - - Base.prototype.traverseChildren = function(crossScope, func) { - return this.eachChild(function(child) { - var recur; - recur = func(child); - if (recur !== false) { - return child.traverseChildren(crossScope, func); - } - }); - }; - - Base.prototype.invert = function() { - return new Op('!', this); - }; - - Base.prototype.unwrapAll = function() { - var node; - node = this; - while (node !== (node = node.unwrap())) { - continue; - } - return node; - }; - - Base.prototype.flattenChildren = function() { - var attr, child, out, _i, _j, _len, _len1, _ref2, _ref3; - out = []; - _ref2 = this.children; - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - attr = _ref2[_i]; - if (this[attr]) { - _ref3 = flatten([this[attr]]); - for (_j = 0, _len1 = _ref3.length; _j < _len1; _j++) { - child = _ref3[_j]; - out.push(child); - } - } - } - return out; - }; - - Base.prototype.icedCompileCps = function(o) { - var code; - this.icedGotCpsSplitFlag = true; - code = CpsCascade.wrap(this, this.icedContinuationBlock, null, o); - return code.compileNode(o); - }; - - Base.prototype.icedWalkAst = function(p, o) { - var child, _i, _len, _ref2; - this.icedParentAwait = p; - this.icedHasAutocbFlag = o.foundAutocb; - _ref2 = this.flattenChildren(); - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - child = _ref2[_i]; - if (child.icedWalkAst(p, o)) { - this.icedNodeFlag = true; - } - } - return this.icedNodeFlag; - }; - - Base.prototype.icedWalkAstLoops = function(flood) { - var child, _i, _len, _ref2; - if (this.isLoop() && this.icedNodeFlag) { - flood = true; - } - if (this.isLoop() && !this.icedNodeFlag) { - flood = false; - } - this.icedLoopFlag = flood; - _ref2 = this.flattenChildren(); - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - child = _ref2[_i]; - if (child.icedWalkAstLoops(flood)) { - this.icedLoopFlag = true; - } - } - return this.icedLoopFlag; - }; - - Base.prototype.icedWalkCpsPivots = function() { - var child, _i, _len, _ref2; - if (this.icedNodeFlag || (this.icedLoopFlag && this.icedIsJump())) { - this.icedCpsPivotFlag = true; - } - _ref2 = this.flattenChildren(); - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - child = _ref2[_i]; - if (child.icedWalkCpsPivots()) { - this.icedCpsPivotFlag = true; - } - } - return this.icedCpsPivotFlag; - }; - - Base.prototype.icedClearAutocbFlags = function() { - this.icedHasAutocbFlag = false; - return this.traverseChildren(false, function(node) { - node.icedHasAutocbFlag = false; - return true; - }); - }; - - Base.prototype.icedCpsRotate = function() { - var child, _i, _len, _ref2; - _ref2 = this.flattenChildren(); - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - child = _ref2[_i]; - child.icedCpsRotate(); - } - return this; - }; - - Base.prototype.icedIsCpsPivot = function() { - return this.icedCpsPivotFlag; - }; - - Base.prototype.icedNestContinuationBlock = function(b) { - return this.icedContinuationBlock = b; - }; - - Base.prototype.icedHasContinuation = function() { - return !!this.icedContinuationBlock; - }; - - Base.prototype.icedCallContinuation = function() { - return this.icedCallContinuationFlag = true; - }; - - Base.prototype.icedWrapContinuation = NO; - - Base.prototype.icedIsJump = NO; - - Base.prototype.icedUnwrap = function(e) { - if (e.icedHasContinuation() && this.icedHasContinuation()) { - return this; - } else { - if (this.icedHasContinuation()) { - e.icedContinuationBlock = this.icedContinuationBlock; - } - return e; - } - }; - - Base.prototype.icedStatementAssertion = function() { - if (this.icedIsCpsPivot()) { - return this.error("await'ed statements can't act as expressions"); - } - }; - - Base.prototype.children = []; - - Base.prototype.isStatement = NO; - - Base.prototype.jumps = NO; - - Base.prototype.isComplex = YES; - - Base.prototype.isChainable = NO; - - Base.prototype.isAssignable = NO; - - Base.prototype.isLoop = NO; - - Base.prototype.unwrap = THIS; - - Base.prototype.unfoldSoak = NO; - - Base.prototype.assigns = NO; - - Base.prototype.updateLocationDataIfMissing = function(locationData) { - if (this.locationData) { - return this; - } - this.locationData = locationData; - return this.eachChild(function(child) { - return child.updateLocationDataIfMissing(locationData); - }); - }; - - Base.prototype.error = function(message) { - return throwSyntaxError(message, this.locationData); - }; - - Base.prototype.makeCode = function(code) { - return new CodeFragment(this, code); - }; - - Base.prototype.wrapInBraces = function(fragments) { - return [].concat(this.makeCode('('), fragments, this.makeCode(')')); - }; - - Base.prototype.joinFragmentArrays = function(fragmentsList, joinStr) { - var answer, fragments, i, _i, _len; - answer = []; - for (i = _i = 0, _len = fragmentsList.length; _i < _len; i = ++_i) { - fragments = fragmentsList[i]; - if (i) { - answer.push(this.makeCode(joinStr)); - } - answer = answer.concat(fragments); - } - return answer; - }; - - return Base; - - })(); - - exports.Block = Block = (function(_super) { - __extends(Block, _super); - - function Block(nodes) { - Block.__super__.constructor.call(this); - this.expressions = compact(flatten(nodes || [])); - } - - Block.prototype.children = ['expressions']; - - Block.prototype.push = function(node) { - this.expressions.push(node); - return this; - }; - - Block.prototype.pop = function() { - return this.expressions.pop(); - }; - - Block.prototype.unshift = function(node) { - this.expressions.unshift(node); - return this; - }; - - Block.prototype.unwrap = function() { - if (this.expressions.length === 1) { - return this.icedUnwrap(this.expressions[0]); - } else { - return this; - } - }; - - Block.prototype.isEmpty = function() { - return !this.expressions.length; - }; - - Block.prototype.isStatement = function(o) { - var exp, _i, _len, _ref2; - _ref2 = this.expressions; - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - exp = _ref2[_i]; - if (exp.isStatement(o)) { - return true; - } - } - return false; - }; - - Block.prototype.jumps = function(o) { - var exp, _i, _len, _ref2; - _ref2 = this.expressions; - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - exp = _ref2[_i]; - if (exp.jumps(o)) { - return exp; - } - } - }; - - Block.prototype.makeReturn = function(res) { - var expr, foundReturn, len; - len = this.expressions.length; - foundReturn = false; - while (len--) { - expr = this.expressions[len]; - if (!(expr instanceof Comment)) { - this.expressions[len] = expr.makeReturn(res); - if (expr instanceof Return && !expr.expression && !expr.icedHasAutocbFlag) { - this.expressions.splice(len, 1); - foundReturn = true; - } else if (!(expr instanceof If) || expr.elseBody) { - foundReturn = true; - } - break; - } - } - if (this.icedHasAutocbFlag && !this.icedNodeFlag && !foundReturn) { - this.expressions.push(new Return(null, true)); - } - return this; - }; - - Block.prototype.compileToFragments = function(o, level) { - if (o == null) { - o = {}; - } - if (o.scope) { - return Block.__super__.compileToFragments.call(this, o, level); - } else { - return this.compileRoot(o); - } - }; - - Block.prototype.compileNode = function(o) { - var answer, compiledNodes, fragments, index, node, top, _i, _len, _ref2; - this.tab = o.indent; - top = o.level === LEVEL_TOP; - compiledNodes = []; - _ref2 = this.expressions; - for (index = _i = 0, _len = _ref2.length; _i < _len; index = ++_i) { - node = _ref2[index]; - node = node.unwrapAll(); - node = node.unfoldSoak(o) || node; - if (node instanceof Block) { - compiledNodes.push(node.compileNode(o)); - } else if (top) { - node.front = true; - fragments = node.compileToFragments(o); - if (!node.isStatement(o)) { - fragments.unshift(this.makeCode("" + this.tab)); - fragments.push(this.makeCode(";")); - } - compiledNodes.push(fragments); - } else { - compiledNodes.push(node.compileToFragments(o, LEVEL_LIST)); - } - } - if (top) { - if (this.spaced) { - return [].concat(this.joinFragmentArrays(compiledNodes, '\n\n'), this.makeCode("\n")); - } else { - return this.joinFragmentArrays(compiledNodes, '\n'); - } - } - if (compiledNodes.length) { - answer = this.joinFragmentArrays(compiledNodes, ', '); - } else { - answer = [this.makeCode("void 0")]; - } - if (compiledNodes.length > 1 && o.level >= LEVEL_LIST) { - return this.wrapInBraces(answer); - } else { - return answer; - } - }; - - Block.prototype.compileRoot = function(o) { - var exp, fragments, i, name, prelude, preludeExps, rest, _i, _len, _ref2; - o.indent = o.bare ? '' : TAB; - o.level = LEVEL_TOP; - this.spaced = true; - o.scope = new Scope(null, this, null); - _ref2 = o.locals || []; - for (_i = 0, _len = _ref2.length; _i < _len; _i++) { - name = _ref2[_i]; - o.scope.parameter(name); - } - prelude = []; - if (!o.bare) { - preludeExps = (function() { - var _j, _len1, _ref3, _results; - _ref3 = this.expressions; - _results = []; - for (i = _j = 0, _len1 = _ref3.length; _j < _len1; i = ++_j) { - exp = _ref3[i]; - if (!(exp.unwrap() instanceof Comment)) { - break; - } - _results.push(exp); - } - return _results; - }).call(this); - rest = this.expressions.slice(preludeExps.length); - this.expressions = preludeExps; - if (preludeExps.length) { - prelude = this.compileNode(merge(o, { - indent: '' - })); - prelude.push(this.makeCode("\n")); - } - this.expressions = rest; - } - fragments = this.compileWithDeclarations(o); - if (o.bare) { - return fragments; - } - return [].concat(prelude, this.makeCode("(function() {\n"), fragments, this.makeCode("\n}).call(this);\n")); - }; - - Block.prototype.compileWithDeclarations = function(o) { - var assigns, declars, exp, fragments, i, post, rest, scope, spaced, _i, _len, _ref2, _ref3, _ref4; - fragments = []; - post = []; - _ref2 = this.expressions; - for (i = _i = 0, _len = _ref2.length; _i < _len; i = ++_i) { - exp = _ref2[i]; - exp = exp.unwrap(); - if (!(exp instanceof Comment || exp instanceof Literal)) { - break; - } - } - o = merge(o, { - level: LEVEL_TOP - }); - if (i) { - rest = this.expressions.splice(i, 9e9); - _ref3 = [this.spaced, false], spaced = _ref3[0], this.spaced = _ref3[1]; - _ref4 = [this.compileNode(o), spaced], fragments = _ref4[0], this.spaced = _ref4[1]; - this.expressions = rest; - } - post = this.compileNode(o); - scope = o.scope; - if (scope.expressions === this) { - declars = o.scope.hasDeclarations(); - assigns = scope.hasAssignments; - if (declars || assigns) { - if (i) { - fragments.push(this.makeCode('\n')); - } - fragments.push(this.makeCode("" + this.tab + "var ")); - if (declars) { - fragments.push(this.makeCode(scope.declaredVariables().join(', '))); - } - if (assigns) { - if (declars) { - fragments.push(this.makeCode(",\n" + (this.tab + TAB))); - } - fragments.push(this.makeCode(scope.assignedVariables().join(",\n" + (this.tab + TAB)))); - } - fragments.push(this.makeCode(";\n" + (this.spaced ? '\n' : ''))); - } else if (fragments.length && post.length) { - fragments.push(this.makeCode("\n")); - } - } - return fragments.concat(post); - }; - - Block.wrap = function(nodes) { - if (nodes.length === 1 && nodes[0] instanceof Block) { - return nodes[0]; - } - return new Block(nodes); - }; - - Block.prototype.icedThreadReturn = function(call) { - var expr, len; - call = call || new IcedTailCall; - len = this.expressions.length; - while (len--) { - expr = this.expressions[len]; - if (expr.isStatement()) { - break; - } - if (!(expr instanceof Comment) && !(expr instanceof Return)) { - call.assignValue(expr); - this.expressions[len] = call; - return; - } - } - return this.expressions.push(call); - }; - - Block.prototype.icedCompileCps = function(o) { - this.icedGotCpsSplitFlag = true; - if (this.expressions.length > 1) { - return Block.__super__.icedCompileCps.call(this, o); - } else { - return this.compileNode(o); - } - }; - - Block.prototype.icedCpsRotate = function() { - var child, e, i, pivot, rest, _i, _j, _len, _len1, _ref2; - pivot = null; - _ref2 = this.expressions; - for (i = _i = 0, _len = _ref2.length; _i < _len; i = ++_i) { - e = _ref2[i]; - if (e.icedIsCpsPivot()) { - pivot = e; - pivot.icedCallContinuation(); - } - e.icedCpsRotate(); - if (pivot) { - break; - } - } - if (!pivot) { - return this; - } - if (pivot.icedContinuationBlock) { - throw SyntaxError("unexpected continuation block in node"); - } - rest = this.expressions.slice(i + 1); - this.expressions = this.expressions.slice(0, i + 1); - if (rest.length) { - child = new Block(rest); - pivot.icedNestContinuationBlock(child); - for (_j = 0, _len1 = rest.length; _j < _len1; _j++) { - e = rest[_j]; - if (e.icedNodeFlag) { - child.icedNodeFlag = true; - } - if (e.icedLoopFlag) { - child.icedLoopFlag = true; - } - if (e.icedCpsPivotFlag) { - child.icedCpsPivotFlag = true; - } - if (e.icedHasAutocbFlag) { - child.icedHasAutocbFlag = true; - } - } - child.icedCpsRotate(); - } - return this; - }; - - Block.prototype.icedAddRuntime = function(foundDefer, foundAwait) { - var index, node; - index = 0; - while ((node = this.expressions[index]) && node instanceof Comment || node instanceof Value && node.isString()) { - index++; - } - return this.expressions.splice(index, 0, new IcedRuntime(foundDefer, foundAwait)); - }; - - Block.prototype.icedTransform = function(opts) { - var obj; - obj = {}; - this.icedWalkAst(null, obj); - if (!(opts != null ? opts.repl : void 0)) { - this.icedAddRuntime(obj.foundDefer, obj.foundAwait); - } - if (obj.foundAwait) { - this.icedWalkAstLoops(false); - this.icedWalkCpsPivots(); - this.icedCpsRotate(); - } - return this; - }; - - Block.prototype.icedGetSingle = function() { - if (this.expressions.length === 1) { - return this.expressions[0]; - } else { - return null; - } - }; - - return Block; - - })(Base); - - exports.Literal = Literal = (function(_super) { - __extends(Literal, _super); - - function Literal(value) { - this.value = value; - Literal.__super__.constructor.call(this); - } - - Literal.prototype.makeReturn = function() { - if (this.isStatement()) { - return this; - } else { - return Literal.__super__.makeReturn.apply(this, arguments); - } - }; - - Literal.prototype.isAssignable = function() { - return IDENTIFIER.test(this.value); - }; - - Literal.prototype.isStatement = function() { - var _ref2; - return (_ref2 = this.value) === 'break' || _ref2 === 'continue' || _ref2 === 'debugger'; - }; - - Literal.prototype.isComplex = NO; - - Literal.prototype.assigns = function(name) { - return name === this.value; - }; - - Literal.prototype.jumps = function(o) { - if (this.value === 'break' && !((o != null ? o.loop : void 0) || (o != null ? o.block : void 0))) { - return this; - } - if (this.value === 'continue' && !(o != null ? o.loop : void 0)) { - return this; - } - }; - - Literal.prototype.compileNode = function(o) { - var answer, code, _ref2; - if (this.icedLoopFlag && this.icedIsJump()) { - return this.icedCompileIced(o); - } - code = this.value === 'this' ? ((_ref2 = o.scope.method) != null ? _ref2.bound : void 0) ? o.scope.method.context : this.value : this.value.reserved ? "\"" + this.value + "\"" : this.value; - answer = this.isStatement() ? "" + this.tab + code + ";" : code; - return [this.makeCode(answer)]; - }; - - Literal.prototype.toString = function() { - return ' "' + this.value + '"'; - }; - - Literal.prototype.icedWalkAst = function(parent, o) { - if (this.value === 'arguments' && o.foundAwaitFunc) { - o.foundArguments = true; - this.value = "_arguments"; - } - return false; - }; - - Literal.prototype.icedIsJump = function() { - return this.isStatement(); - }; - - Literal.prototype.icedCompileIced = function(o) { - var call, d, func, l; - d = { - 'continue': iced["const"].c_while, - 'break': iced["const"].b_while - }; - l = d[this.value]; - func = new Value(new Literal(l)); - call = new Call(func, []); - return call.compileNode(o); - }; - - return Literal; - - })(Base); - - exports.Undefined = (function(_super) { - __extends(Undefined, _super); - - function Undefined() { - _ref2 = Undefined.__super__.constructor.apply(this, arguments); - return _ref2; - } - - Undefined.prototype.isAssignable = NO; - - Undefined.prototype.isComplex = NO; - - Undefined.prototype.compileNode = function(o) { - return [this.makeCode(o.level >= LEVEL_ACCESS ? '(void 0)' : 'void 0')]; - }; - - return Undefined; - - })(Base); - - exports.Null = (function(_super) { - __extends(Null, _super); - - function Null() { - _ref3 = Null.__super__.constructor.apply(this, arguments); - return _ref3; - } - - Null.prototype.isAssignable = NO; - - Null.prototype.isComplex = NO; - - Null.prototype.compileNode = function() { - return [this.makeCode("null")]; - }; - - return Null; - - })(Base); - - exports.Bool = (function(_super) { - __extends(Bool, _super); - - Bool.prototype.isAssignable = NO; - - Bool.prototype.isComplex = NO; - - Bool.prototype.compileNode = function() { - return [this.makeCode(this.val)]; - }; - - function Bool(val) { - this.val = val; - } - - return Bool; - - })(Base); - - exports.Return = Return = (function(_super) { - __extends(Return, _super); - - function Return(expr, auto) { - Return.__super__.constructor.call(this); - this.icedHasAutocbFlag = auto; - if (expr && !expr.unwrap().isUndefined) { - this.expression = expr; - } - } - - Return.prototype.children = ['expression']; - - Return.prototype.isStatement = YES; - - Return.prototype.makeReturn = THIS; - - Return.prototype.jumps = THIS; - - Return.prototype.compileToFragments = function(o, level) { - var expr, _ref4; - expr = (_ref4 = this.expression) != null ? _ref4.makeReturn() : void 0; - if (expr && !(expr instanceof Return)) { - return expr.compileToFragments(o, level); - } else { - return Return.__super__.compileToFragments.call(this, o, level); - } - }; - - Return.prototype.compileNode = function(o) { - var answer; - if (this.icedHasAutocbFlag) { - return this.icedCompileIced(o); - } - answer = []; - answer.push(this.makeCode(this.tab + ("return" + (this.expression ? " " : "")))); - if (this.expression) { - answer = answer.concat(this.expression.compileToFragments(o, LEVEL_PAREN)); - } - answer.push(this.makeCode(";")); - return answer; - }; - - Return.prototype.icedCompileIced = function(o) { - var args, block, call, cb, ret; - cb = new Value(new Literal(iced["const"].autocb)); - args = this.expression ? [this.expression] : []; - call = new Call(cb, args); - ret = new Literal("return"); - block = new Block([call, ret]); - return block.compileNode(o); - }; - - return Return; - - })(Base); - - exports.Value = Value = (function(_super) { - __extends(Value, _super); - - function Value(base, props, tag) { - Value.__super__.constructor.call(this); - if (!props && base instanceof Value) { - return base; - } - this.base = base; - this.properties = props || []; - if (tag) { - this[tag] = true; - } - return this; - } - - Value.prototype.children = ['base', 'properties']; - - Value.prototype.copy = function() { - return new Value(this.base, this.properties); - }; - - Value.prototype.add = function(props) { - this.properties = this.properties.concat(props); - return this; - }; - - Value.prototype.hasProperties = function() { - return !!this.properties.length; - }; - - Value.prototype.isArray = function() { - return !this.properties.length && this.base instanceof Arr; - }; - - Value.prototype.isComplex = function() { - return this.hasProperties() || this.base.isComplex(); - }; - - Value.prototype.isAssignable = function() { - return this.hasProperties() || this.base.isAssignable(); - }; - - Value.prototype.isSimpleNumber = function() { - return this.base instanceof Literal && SIMPLENUM.test(this.base.value); - }; - - Value.prototype.isString = function() { - return this.base instanceof Literal && IS_STRING.test(this.base.value); - }; - - Value.prototype.isAtomic = function() { - var node, _i, _len, _ref4; - _ref4 = this.properties.concat(this.base); - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - node = _ref4[_i]; - if (node.soak || node instanceof Call) { - return false; - } - } - return true; - }; - - Value.prototype.isStatement = function(o) { - return !this.properties.length && this.base.isStatement(o); - }; - - Value.prototype.assigns = function(name) { - return !this.properties.length && this.base.assigns(name); - }; - - Value.prototype.jumps = function(o) { - return !this.properties.length && this.base.jumps(o); - }; - - Value.prototype.isObject = function(onlyGenerated) { - if (this.properties.length) { - return false; - } - return (this.base instanceof Obj) && (!onlyGenerated || this.base.generated); - }; - - Value.prototype.isSplice = function() { - return last(this.properties) instanceof Slice; - }; - - Value.prototype.unwrap = function() { - if (this.properties.length) { - return this; - } else { - return this.base; - } - }; - - Value.prototype.cacheReference = function(o) { - var base, bref, name, nref; - name = last(this.properties); - if (this.properties.length < 2 && !this.base.isComplex() && !(name != null ? name.isComplex() : void 0)) { - return [this, this]; - } - base = new Value(this.base, this.properties.slice(0, -1)); - if (base.isComplex()) { - bref = new Literal(o.scope.freeVariable('base')); - base = new Value(new Parens(new Assign(bref, base))); - } - if (!name) { - return [base, bref]; - } - if (name.isComplex()) { - nref = new Literal(o.scope.freeVariable('name')); - name = new Index(new Assign(nref, name.index)); - nref = new Index(nref); - } - return [base.add(name), new Value(bref || base.base, [nref || name])]; - }; - - Value.prototype.compileNode = function(o) { - var fragments, prop, props, _i, _len; - this.base.front = this.front; - props = this.properties; - fragments = this.base.compileToFragments(o, (props.length ? LEVEL_ACCESS : null)); - if ((this.base instanceof Parens || props.length) && SIMPLENUM.test(fragmentsToText(fragments))) { - fragments.push(this.makeCode('.')); - } - for (_i = 0, _len = props.length; _i < _len; _i++) { - prop = props[_i]; - fragments.push.apply(fragments, prop.compileToFragments(o)); - } - return fragments; - }; - - Value.prototype.unfoldSoak = function(o) { - var _this = this; - return this.unfoldedSoak != null ? this.unfoldedSoak : this.unfoldedSoak = (function() { - var fst, i, ifn, prop, ref, snd, _i, _len, _ref4, _ref5; - if (ifn = _this.base.unfoldSoak(o)) { - (_ref4 = ifn.body.properties).push.apply(_ref4, _this.properties); - return ifn; - } - _ref5 = _this.properties; - for (i = _i = 0, _len = _ref5.length; _i < _len; i = ++_i) { - prop = _ref5[i]; - if (!prop.soak) { - continue; - } - prop.soak = false; - fst = new Value(_this.base, _this.properties.slice(0, i)); - snd = new Value(_this.base, _this.properties.slice(i)); - if (fst.isComplex()) { - ref = new Literal(o.scope.freeVariable('ref')); - fst = new Parens(new Assign(ref, fst)); - snd.base = ref; - } - return new If(new Existence(fst), snd, { - soak: true - }); - } - return false; - })(); - }; - - Value.prototype.icedToSlot = function(i) { - var sufffix, suffix; - if (this.base instanceof Obj) { - return this.base.icedToSlot(i); - } - sufffix = null; - if (this.properties && this.properties.length) { - suffix = this.properties.pop(); - } - return new Slot(i, this, suffix); - }; - - Value.prototype.icedToSlotAccess = function() { - if (this["this"]) { - return this.properties[0]; - } else { - return new Access(this); - } - }; - - return Value; - - })(Base); - - exports.Comment = Comment = (function(_super) { - __extends(Comment, _super); - - function Comment(comment) { - this.comment = comment; - Comment.__super__.constructor.call(this); - } - - Comment.prototype.isStatement = YES; - - Comment.prototype.makeReturn = THIS; - - Comment.prototype.compileNode = function(o, level) { - var code; - code = "/*" + (multident(this.comment, this.tab)) + (__indexOf.call(this.comment, '\n') >= 0 ? "\n" + this.tab : '') + "*/"; - if ((level || o.level) === LEVEL_TOP) { - code = o.indent + code; - } - return [this.makeCode("\n"), this.makeCode(code)]; - }; - - return Comment; - - })(Base); - - exports.Call = Call = (function(_super) { - __extends(Call, _super); - - function Call(variable, args, soak) { - this.args = args != null ? args : []; - this.soak = soak; - Call.__super__.constructor.call(this); - this.isNew = false; - this.isSuper = variable === 'super'; - this.variable = this.isSuper ? null : variable; - } - - Call.prototype.children = ['variable', 'args']; - - Call.prototype.newInstance = function() { - var base, _ref4; - base = ((_ref4 = this.variable) != null ? _ref4.base : void 0) || this.variable; - if (base instanceof Call && !base.isNew) { - base.newInstance(); - } else { - this.isNew = true; - } - return this; - }; - - Call.prototype.superReference = function(o) { - var accesses, method; - method = o.scope.namedMethod(); - if (method != null ? method.klass : void 0) { - accesses = [new Access(new Literal('__super__'))]; - if (method["static"]) { - accesses.push(new Access(new Literal('constructor'))); - } - accesses.push(new Access(new Literal(method.name))); - return (new Value(new Literal(method.klass), accesses)).compile(o); - } else if (method != null ? method.ctor : void 0) { - return "" + method.name + ".__super__.constructor"; - } else { - return this.error('cannot call super outside of an instance method.'); - } - }; - - Call.prototype.superThis = function(o) { - var method; - if (o.scope.icedgen) { - return "_this"; - } else { - method = o.scope.method; - return (method && !method.klass && method.context) || "this"; - } - }; - - Call.prototype.unfoldSoak = function(o) { - var call, ifn, left, list, rite, _i, _len, _ref4, _ref5; - if (this.soak) { - if (this.variable) { - if (ifn = unfoldSoak(o, this, 'variable')) { - return ifn; - } - _ref4 = new Value(this.variable).cacheReference(o), left = _ref4[0], rite = _ref4[1]; - } else { - left = new Literal(this.superReference(o)); - rite = new Value(left); - } - rite = new Call(rite, this.args); - rite.isNew = this.isNew; - left = new Literal("typeof " + (left.compile(o)) + " === \"function\""); - return new If(left, new Value(rite), { - soak: true - }); - } - call = this; - list = []; - while (true) { - if (call.variable instanceof Call) { - list.push(call); - call = call.variable; - continue; - } - if (!(call.variable instanceof Value)) { - break; - } - list.push(call); - if (!((call = call.variable.base) instanceof Call)) { - break; - } - } - _ref5 = list.reverse(); - for (_i = 0, _len = _ref5.length; _i < _len; _i++) { - call = _ref5[_i]; - if (ifn) { - if (call.variable instanceof Call) { - call.variable = ifn; - } else { - call.variable.base = ifn; - } - } - ifn = unfoldSoak(o, call, 'variable'); - } - return ifn; - }; - - Call.prototype.compileNode = function(o) { - var arg, argIndex, compiledArgs, compiledArray, fragments, preface, _i, _len, _ref4, _ref5; - if ((_ref4 = this.variable) != null) { - _ref4.front = this.front; - } - compiledArray = Splat.compileSplattedArray(o, this.args, true); - if (compiledArray.length) { - return this.compileSplat(o, compiledArray); - } - compiledArgs = []; - _ref5 = this.args; - for (argIndex = _i = 0, _len = _ref5.length; _i < _len; argIndex = ++_i) { - arg = _ref5[argIndex]; - arg.icedStatementAssertion(); - if (argIndex) { - compiledArgs.push(this.makeCode(", ")); - } - compiledArgs.push.apply(compiledArgs, arg.compileToFragments(o, LEVEL_LIST)); - } - fragments = []; - if (this.isSuper) { - preface = this.superReference(o) + (".call(" + (this.superThis(o))); - if (compiledArgs.length) { - preface += ", "; - } - fragments.push(this.makeCode(preface)); - } else { - if (this.isNew) { - fragments.push(this.makeCode('new ')); - } - fragments.push.apply(fragments, this.variable.compileToFragments(o, LEVEL_ACCESS)); - fragments.push(this.makeCode("(")); - } - fragments.push.apply(fragments, compiledArgs); - fragments.push(this.makeCode(")")); - return fragments; - }; - - Call.prototype.compileSplat = function(o, splatArgs) { - var answer, base, fun, idt, name, ref; - if (this.isSuper) { - return [].concat(this.makeCode("" + (this.superReference(o)) + ".apply(" + (this.superThis(o)) + ", "), splatArgs, this.makeCode(")")); - } - if (this.isNew) { - idt = this.tab + TAB; - return [].concat(this.makeCode("(function(func, args, ctor) {\n" + idt + "ctor.prototype = func.prototype;\n" + idt + "var child = new ctor, result = func.apply(child, args);\n" + idt + "return Object(result) === result ? result : child;\n" + this.tab + "})("), this.variable.compileToFragments(o, LEVEL_LIST), this.makeCode(", "), splatArgs, this.makeCode(", function(){})")); - } - answer = []; - base = new Value(this.variable); - if ((name = base.properties.pop()) && base.isComplex()) { - ref = o.scope.freeVariable('ref'); - answer = answer.concat(this.makeCode("(" + ref + " = "), base.compileToFragments(o, LEVEL_LIST), this.makeCode(")"), name.compileToFragments(o)); - } else { - fun = base.compileToFragments(o, LEVEL_ACCESS); - if (SIMPLENUM.test(fragmentsToText(fun))) { - fun = this.wrapInBraces(fun); - } - if (name) { - ref = fragmentsToText(fun); - fun.push.apply(fun, name.compileToFragments(o)); - } else { - ref = 'null'; - } - answer = answer.concat(fun); - } - return answer = answer.concat(this.makeCode(".apply(" + ref + ", "), splatArgs, this.makeCode(")")); - }; - - return Call; - - })(Base); - - exports.Extends = Extends = (function(_super) { - __extends(Extends, _super); - - function Extends(child, parent) { - this.child = child; - this.parent = parent; - Extends.__super__.constructor.call(this); - } - - Extends.prototype.children = ['child', 'parent']; - - Extends.prototype.compileToFragments = function(o) { - return new Call(new Value(new Literal(utility('extends'))), [this.child, this.parent]).compileToFragments(o); - }; - - return Extends; - - })(Base); - - exports.Access = Access = (function(_super) { - __extends(Access, _super); - - function Access(name, tag) { - this.name = name; - Access.__super__.constructor.call(this); - this.name.asKey = true; - this.soak = tag === 'soak'; - } - - Access.prototype.children = ['name']; - - Access.prototype.compileToFragments = function(o) { - var name; - name = this.name.compileToFragments(o); - if ((IDENTIFIER.test(fragmentsToText(name))) || this.name instanceof Defer) { - name.unshift(this.makeCode(".")); - } else { - name.unshift(this.makeCode("[")); - name.push(this.makeCode("]")); - } - return name; - }; - - Access.prototype.isComplex = NO; - - return Access; - - })(Base); - - exports.Index = Index = (function(_super) { - __extends(Index, _super); - - function Index(index) { - this.index = index; - Index.__super__.constructor.call(this); - } - - Index.prototype.children = ['index']; - - Index.prototype.compileToFragments = function(o) { - return [].concat(this.makeCode("["), this.index.compileToFragments(o, LEVEL_PAREN), this.makeCode("]")); - }; - - Index.prototype.isComplex = function() { - return this.index.isComplex(); - }; - - return Index; - - })(Base); - - exports.Range = Range = (function(_super) { - __extends(Range, _super); - - Range.prototype.children = ['from', 'to']; - - function Range(from, to, tag) { - this.from = from; - this.to = to; - Range.__super__.constructor.call(this); - this.exclusive = tag === 'exclusive'; - this.equals = this.exclusive ? '' : '='; - } - - Range.prototype.compileVariables = function(o) { - var step, _ref4, _ref5, _ref6, _ref7; - o = merge(o, { - top: true - }); - _ref4 = this.cacheToCodeFragments(this.from.cache(o, LEVEL_LIST)), this.fromC = _ref4[0], this.fromVar = _ref4[1]; - _ref5 = this.cacheToCodeFragments(this.to.cache(o, LEVEL_LIST)), this.toC = _ref5[0], this.toVar = _ref5[1]; - if (step = del(o, 'step')) { - _ref6 = this.cacheToCodeFragments(step.cache(o, LEVEL_LIST)), this.step = _ref6[0], this.stepVar = _ref6[1]; - } - _ref7 = [this.fromVar.match(SIMPLENUM), this.toVar.match(SIMPLENUM)], this.fromNum = _ref7[0], this.toNum = _ref7[1]; - if (this.stepVar) { - return this.stepNum = this.stepVar.match(SIMPLENUM); - } - }; - - Range.prototype.compileNode = function(o) { - var cond, condPart, from, gt, idx, idxName, known, lt, namedIndex, stepPart, to, varPart, _ref4, _ref5; - if (!this.fromVar) { - this.compileVariables(o); - } - if (!o.index) { - return this.compileArray(o); - } - known = this.fromNum && this.toNum; - idx = del(o, 'index'); - idxName = del(o, 'name'); - namedIndex = idxName && idxName !== idx; - varPart = "" + idx + " = " + this.fromC; - if (this.toC !== this.toVar) { - varPart += ", " + this.toC; - } - if (this.step !== this.stepVar) { - varPart += ", " + this.step; - } - _ref4 = ["" + idx + " <" + this.equals, "" + idx + " >" + this.equals], lt = _ref4[0], gt = _ref4[1]; - condPart = this.stepNum ? +this.stepNum > 0 ? "" + lt + " " + this.toVar : "" + gt + " " + this.toVar : known ? ((_ref5 = [+this.fromNum, +this.toNum], from = _ref5[0], to = _ref5[1], _ref5), from <= to ? "" + lt + " " + to : "" + gt + " " + to) : (cond = this.stepVar ? "" + this.stepVar + " > 0" : "" + this.fromVar + " <= " + this.toVar, "" + cond + " ? " + lt + " " + this.toVar + " : " + gt + " " + this.toVar); - stepPart = this.stepVar ? "" + idx + " += " + this.stepVar : known ? namedIndex ? from <= to ? "++" + idx : "--" + idx : from <= to ? "" + idx + "++" : "" + idx + "--" : namedIndex ? "" + cond + " ? ++" + idx + " : --" + idx : "" + cond + " ? " + idx + "++ : " + idx + "--"; - if (namedIndex) { - varPart = "" + idxName + " = " + varPart; - } - if (namedIndex) { - stepPart = "" + idxName + " = " + stepPart; - } - return [this.makeCode("" + varPart + "; " + condPart + "; " + stepPart)]; - }; - - Range.prototype.compileArray = function(o) { - var args, body, cond, hasArgs, i, idt, post, pre, range, result, vars, _i, _ref4, _ref5, _results; - if (this.fromNum && this.toNum && Math.abs(this.fromNum - this.toNum) <= 20) { - range = (function() { - _results = []; - for (var _i = _ref4 = +this.fromNum, _ref5 = +this.toNum; _ref4 <= _ref5 ? _i <= _ref5 : _i >= _ref5; _ref4 <= _ref5 ? _i++ : _i--){ _results.push(_i); } - return _results; - }).apply(this); - if (this.exclusive) { - range.pop(); - } - return [this.makeCode("[" + (range.join(', ')) + "]")]; - } - idt = this.tab + TAB; - i = o.scope.freeVariable('i'); - result = o.scope.freeVariable('results'); - pre = "\n" + idt + result + " = [];"; - if (this.fromNum && this.toNum) { - o.index = i; - body = fragmentsToText(this.compileNode(o)); - } else { - vars = ("" + i + " = " + this.fromC) + (this.toC !== this.toVar ? ", " + this.toC : ''); - cond = "" + this.fromVar + " <= " + this.toVar; - body = "var " + vars + "; " + cond + " ? " + i + " <" + this.equals + " " + this.toVar + " : " + i + " >" + this.equals + " " + this.toVar + "; " + cond + " ? " + i + "++ : " + i + "--"; - } - post = "{ " + result + ".push(" + i + "); }\n" + idt + "return " + result + ";\n" + o.indent; - hasArgs = function(node) { - return node != null ? node.contains(function(n) { - return n instanceof Literal && n.value === 'arguments' && !n.asKey; - }) : void 0; - }; - if (hasArgs(this.from) || hasArgs(this.to)) { - args = ', arguments'; - } - return [this.makeCode("(function() {" + pre + "\n" + idt + "for (" + body + ")" + post + "}).apply(this" + (args != null ? args : '') + ")")]; - }; - - return Range; - - })(Base); - - exports.Slice = Slice = (function(_super) { - __extends(Slice, _super); - - Slice.prototype.children = ['range']; - - function Slice(range) { - this.range = range; - Slice.__super__.constructor.call(this); - } - - Slice.prototype.compileNode = function(o) { - var compiled, compiledText, from, fromCompiled, to, toStr, _ref4; - _ref4 = this.range, to = _ref4.to, from = _ref4.from; - fromCompiled = from && from.compileToFragments(o, LEVEL_PAREN) || [this.makeCode('0')]; - if (to) { - compiled = to.compileToFragments(o, LEVEL_PAREN); - compiledText = fragmentsToText(compiled); - if (!(!this.range.exclusive && +compiledText === -1)) { - toStr = ', ' + (this.range.exclusive ? compiledText : SIMPLENUM.test(compiledText) ? "" + (+compiledText + 1) : (compiled = to.compileToFragments(o, LEVEL_ACCESS), "+" + (fragmentsToText(compiled)) + " + 1 || 9e9")); - } - } - return [this.makeCode(".slice(" + (fragmentsToText(fromCompiled)) + (toStr || '') + ")")]; - }; - - return Slice; - - })(Base); - - exports.Obj = Obj = (function(_super) { - __extends(Obj, _super); - - function Obj(props, generated) { - this.generated = generated != null ? generated : false; - this.objects = this.properties = props || []; - Obj.__super__.constructor.call(this); - } - - Obj.prototype.children = ['properties']; - - Obj.prototype.compileNode = function(o) { - var answer, i, idt, indent, join, lastNoncom, node, prop, props, _i, _j, _len, _len1; - props = this.properties; - if (!props.length) { - return [this.makeCode(this.front ? '({})' : '{}')]; - } - if (this.generated) { - for (_i = 0, _len = props.length; _i < _len; _i++) { - node = props[_i]; - if (node instanceof Value) { - node.error('cannot have an implicit value in an implicit object'); - } - } - } - idt = o.indent += TAB; - lastNoncom = this.lastNonComment(this.properties); - answer = []; - for (i = _j = 0, _len1 = props.length; _j < _len1; i = ++_j) { - prop = props[i]; - join = i === props.length - 1 ? '' : prop === lastNoncom || prop instanceof Comment ? '\n' : ',\n'; - indent = prop instanceof Comment ? '' : idt; - if (prop instanceof Assign && prop.variable instanceof Value && prop.variable.hasProperties()) { - prop.variable.error('Invalid object key'); - } - if (prop instanceof Value && prop["this"]) { - prop = new Assign(prop.properties[0].name, prop, 'object'); - } - if (!(prop instanceof Comment)) { - if (!(prop instanceof Assign)) { - prop = new Assign(prop, prop, 'object'); - } - (prop.variable.base || prop.variable).asKey = true; - } - if (indent) { - answer.push(this.makeCode(indent)); - } - answer.push.apply(answer, prop.compileToFragments(o, LEVEL_TOP)); - if (join) { - answer.push(this.makeCode(join)); - } - } - answer.unshift(this.makeCode("{" + (props.length && '\n'))); - answer.push(this.makeCode("" + (props.length && '\n' + this.tab) + "}")); - if (this.front) { - return this.wrapInBraces(answer); - } else { - return answer; - } - }; - - Obj.prototype.assigns = function(name) { - var prop, _i, _len, _ref4; - _ref4 = this.properties; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - prop = _ref4[_i]; - if (prop.assigns(name)) { - return true; - } - } - return false; - }; - - Obj.prototype.icedToSlot = function(i) { - var access, prop, _i, _len, _ref4, _results; - _ref4 = this.properties; - _results = []; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - prop = _ref4[_i]; - if (prop instanceof Assign) { - _results.push((prop.value.icedToSlot(i)).addAccess(prop.variable.icedToSlotAccess())); - } else if (prop instanceof Value) { - access = prop.icedToSlotAccess(); - _results.push((prop.icedToSlot(i)).addAccess(access)); - } else { - _results.push(void 0); - } - } - return _results; - }; - - return Obj; - - })(Base); - - exports.Arr = Arr = (function(_super) { - __extends(Arr, _super); - - function Arr(objs) { - this.objects = objs || []; - Arr.__super__.constructor.call(this); - } - - Arr.prototype.children = ['objects']; - - Arr.prototype.compileNode = function(o) { - var answer, compiledObjs, fragments, index, obj, _i, _len; - if (!this.objects.length) { - return [this.makeCode('[]')]; - } - o.indent += TAB; - answer = Splat.compileSplattedArray(o, this.objects); - if (answer.length) { - return answer; - } - answer = []; - compiledObjs = (function() { - var _i, _len, _ref4, _results; - _ref4 = this.objects; - _results = []; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - obj = _ref4[_i]; - _results.push(obj.compileToFragments(o, LEVEL_LIST)); - } - return _results; - }).call(this); - for (index = _i = 0, _len = compiledObjs.length; _i < _len; index = ++_i) { - fragments = compiledObjs[index]; - if (index) { - answer.push(this.makeCode(", ")); - } - answer.push.apply(answer, fragments); - } - if (fragmentsToText(answer).indexOf('\n') >= 0) { - answer.unshift(this.makeCode("[\n" + o.indent)); - answer.push(this.makeCode("\n" + this.tab + "]")); - } else { - answer.unshift(this.makeCode("[")); - answer.push(this.makeCode("]")); - } - return answer; - }; - - Arr.prototype.assigns = function(name) { - var obj, _i, _len, _ref4; - _ref4 = this.objects; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - obj = _ref4[_i]; - if (obj.assigns(name)) { - return true; - } - } - return false; - }; - - return Arr; - - })(Base); - - exports.Class = Class = (function(_super) { - __extends(Class, _super); - - function Class(variable, parent, body) { - this.variable = variable; - this.parent = parent; - this.body = body != null ? body : new Block; - Class.__super__.constructor.call(this); - this.boundFuncs = []; - this.body.classBody = true; - } - - Class.prototype.children = ['variable', 'parent', 'body']; - - Class.prototype.determineName = function() { - var decl, tail; - if (!this.variable) { - return null; - } - decl = (tail = last(this.variable.properties)) ? tail instanceof Access && tail.name.value : this.variable.base.value; - if (__indexOf.call(STRICT_PROSCRIBED, decl) >= 0) { - this.variable.error("class variable name may not be " + decl); - } - return decl && (decl = IDENTIFIER.test(decl) && decl); - }; - - Class.prototype.setContext = function(name) { - return this.body.traverseChildren(false, function(node) { - if (node.classBody) { - return false; - } - if (node instanceof Literal && node.value === 'this') { - return node.value = name; - } else if (node instanceof Code) { - node.klass = name; - if (node.bound) { - return node.context = name; - } - } - }); - }; - - Class.prototype.addBoundFunctions = function(o) { - var bvar, lhs, _i, _len, _ref4; - _ref4 = this.boundFuncs; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - bvar = _ref4[_i]; - lhs = (new Value(new Literal("this"), [new Access(bvar)])).compile(o); - this.ctor.body.unshift(new Literal("" + lhs + " = " + (utility('bind')) + "(" + lhs + ", this)")); - } - }; - - Class.prototype.addProperties = function(node, name, o) { - var assign, base, exprs, func, props; - props = node.base.properties.slice(0); - exprs = (function() { - var _results; - _results = []; - while (assign = props.shift()) { - if (assign instanceof Assign) { - base = assign.variable.base; - delete assign.context; - func = assign.value; - if (base.value === 'constructor') { - if (this.ctor) { - assign.error('cannot define more than one constructor in a class'); - } - if (func.bound) { - assign.error('cannot define a constructor as a bound function'); - } - if (func instanceof Code) { - assign = this.ctor = func; - } else { - this.externalCtor = o.scope.freeVariable('class'); - assign = new Assign(new Literal(this.externalCtor), func); - } - } else { - if (assign.variable["this"]) { - func["static"] = true; - if (func.bound) { - func.context = name; - } - } else { - assign.variable = new Value(new Literal(name), [new Access(new Literal('prototype')), new Access(base)]); - if (func instanceof Code && func.bound) { - this.boundFuncs.push(base); - func.bound = false; - } - } - } - } - _results.push(assign); - } - return _results; - }).call(this); - return compact(exprs); - }; - - Class.prototype.walkBody = function(name, o) { - var _this = this; - return this.traverseChildren(false, function(child) { - var cont, exps, i, node, _i, _len, _ref4; - cont = true; - if (child instanceof Class) { - return false; - } - if (child instanceof Block) { - _ref4 = exps = child.expressions; - for (i = _i = 0, _len = _ref4.length; _i < _len; i = ++_i) { - node = _ref4[i]; - if (node instanceof Value && node.isObject(true)) { - cont = false; - exps[i] = _this.addProperties(node, name, o); - } - } - child.expressions = exps = flatten(exps); - } - return cont && !(child instanceof Class); - }); - }; - - Class.prototype.hoistDirectivePrologue = function() { - var expressions, index, node; - index = 0; - expressions = this.body.expressions; - while ((node = expressions[index]) && node instanceof Comment || node instanceof Value && node.isString()) { - ++index; - } - return this.directives = expressions.splice(0, index); - }; - - Class.prototype.ensureConstructor = function(name, o) { - var missing, ref, superCall; - missing = !this.ctor; - this.ctor || (this.ctor = new Code); - this.ctor.ctor = this.ctor.name = name; - this.ctor.klass = null; - this.ctor.noReturn = true; - if (missing) { - if (this.parent) { - superCall = new Literal("" + name + ".__super__.constructor.apply(this, arguments)"); - } - if (this.externalCtor) { - superCall = new Literal("" + this.externalCtor + ".apply(this, arguments)"); - } - if (superCall) { - ref = new Literal(o.scope.freeVariable('ref')); - this.ctor.body.unshift(new Assign(ref, superCall)); - } - this.addBoundFunctions(o); - if (superCall) { - this.ctor.body.push(ref); - this.ctor.body.makeReturn(); - } - return this.body.expressions.unshift(this.ctor); - } else { - return this.addBoundFunctions(o); - } - }; - - Class.prototype.compileNode = function(o) { - var call, decl, klass, lname, name, params, _ref4; - decl = this.determineName(); - name = decl || '_Class'; - if (name.reserved) { - name = "_" + name; - } - lname = new Literal(name); - this.hoistDirectivePrologue(); - this.setContext(name); - this.walkBody(name, o); - this.ensureConstructor(name, o); - this.body.spaced = true; - if (!(this.ctor instanceof Code)) { - this.body.expressions.unshift(this.ctor); - } - this.body.expressions.push(lname); - (_ref4 = this.body.expressions).unshift.apply(_ref4, this.directives); - call = Closure.wrap(this.body); - if (this.parent) { - this.superClass = new Literal(o.scope.freeVariable('super', false)); - this.body.expressions.unshift(new Extends(lname, this.superClass)); - call.args.push(this.parent); - params = call.variable.params || call.variable.base.params; - params.push(new Param(this.superClass)); - } - klass = new Parens(call, true); - if (this.variable) { - klass = new Assign(this.variable, klass); - } - return klass.compileToFragments(o); - }; - - return Class; - - })(Base); - - exports.Assign = Assign = (function(_super) { - __extends(Assign, _super); - - function Assign(variable, value, context, options) { - var forbidden, name, _ref4; - this.variable = variable; - this.value = value; - this.context = context; - Assign.__super__.constructor.call(this); - this.param = options && options.param; - this.subpattern = options && options.subpattern; - forbidden = (_ref4 = (name = this.variable.unwrapAll().value), __indexOf.call(STRICT_PROSCRIBED, _ref4) >= 0); - if (forbidden && this.context !== 'object') { - this.variable.error("variable name may not be \"" + name + "\""); - } - this.icedlocal = options && options.icedlocal; - } - - Assign.prototype.children = ['variable', 'value']; - - Assign.prototype.isStatement = function(o) { - return (o != null ? o.level : void 0) === LEVEL_TOP && (this.context != null) && __indexOf.call(this.context, "?") >= 0; - }; - - Assign.prototype.assigns = function(name) { - return this[this.context === 'object' ? 'value' : 'variable'].assigns(name); - }; - - Assign.prototype.unfoldSoak = function(o) { - return unfoldSoak(o, this, 'variable'); - }; - - Assign.prototype.compileNode = function(o) { - var answer, compiledName, isValue, match, name, val, varBase, _ref4, _ref5, _ref6, _ref7; - this.value.icedStatementAssertion(); - if (isValue = this.variable instanceof Value) { - if (this.variable.isArray() || this.variable.isObject()) { - return this.compilePatternMatch(o); - } - if (this.variable.isSplice()) { - return this.compileSplice(o); - } - if ((_ref4 = this.context) === '||=' || _ref4 === '&&=' || _ref4 === '?=') { - return this.compileConditional(o); - } - } - compiledName = this.variable.compileToFragments(o, LEVEL_LIST); - name = fragmentsToText(compiledName); - if (!this.context) { - varBase = this.variable.unwrapAll(); - if (!varBase.isAssignable()) { - this.variable.error("\"" + (this.variable.compile(o)) + "\" cannot be assigned"); - } - if (!(typeof varBase.hasProperties === "function" ? varBase.hasProperties() : void 0)) { - if (this.param || this.icedlocal) { - o.scope.add(name, 'var', this.icedlocal); - } else { - o.scope.find(name); - } - } - } - if (this.value instanceof Code && (match = METHOD_DEF.exec(name))) { - if (match[1]) { - this.value.klass = match[1]; - } - this.value.name = (_ref5 = (_ref6 = (_ref7 = match[2]) != null ? _ref7 : match[3]) != null ? _ref6 : match[4]) != null ? _ref5 : match[5]; - } - val = this.value.compileToFragments(o, LEVEL_LIST); - if (this.context === 'object') { - return compiledName.concat(this.makeCode(": "), val); - } - answer = compiledName.concat(this.makeCode(" " + (this.context || '=') + " "), val); - if (o.level <= LEVEL_LIST) { - return answer; - } else { - return this.wrapInBraces(answer); - } - }; - - Assign.prototype.compilePatternMatch = function(o) { - var acc, assigns, code, fragments, i, idx, isObject, ivar, name, obj, objects, olen, ref, rest, splat, top, val, value, vvar, vvarText, _i, _len, _ref4, _ref5, _ref6, _ref7, _ref8, _ref9; - top = o.level === LEVEL_TOP; - value = this.value; - objects = this.variable.base.objects; - if (!(olen = objects.length)) { - code = value.compileToFragments(o); - if (o.level >= LEVEL_OP) { - return this.wrapInBraces(code); - } else { - return code; - } - } - isObject = this.variable.isObject(); - if (top && olen === 1 && !((obj = objects[0]) instanceof Splat)) { - if (obj instanceof Assign) { - _ref4 = obj, (_ref5 = _ref4.variable, idx = _ref5.base), obj = _ref4.value; - } else { - idx = isObject ? obj["this"] ? obj.properties[0].name : obj : new Literal(0); - } - acc = IDENTIFIER.test(idx.unwrap().value || 0); - value = new Value(value); - value.properties.push(new (acc ? Access : Index)(idx)); - if (_ref6 = obj.unwrap().value, __indexOf.call(RESERVED, _ref6) >= 0) { - obj.error("assignment to a reserved word: " + (obj.compile(o))); - } - return new Assign(obj, value, null, { - param: this.param - }).compileToFragments(o, LEVEL_TOP); - } - vvar = value.compileToFragments(o, LEVEL_LIST); - vvarText = fragmentsToText(vvar); - assigns = []; - splat = false; - if (!IDENTIFIER.test(vvarText) || this.variable.assigns(vvarText)) { - assigns.push([this.makeCode("" + (ref = o.scope.freeVariable('ref')) + " = ")].concat(__slice.call(vvar))); - vvar = [this.makeCode(ref)]; - vvarText = ref; - } - for (i = _i = 0, _len = objects.length; _i < _len; i = ++_i) { - obj = objects[i]; - idx = i; - if (isObject) { - if (obj instanceof Assign) { - _ref7 = obj, (_ref8 = _ref7.variable, idx = _ref8.base), obj = _ref7.value; - } else { - if (obj.base instanceof Parens) { - _ref9 = new Value(obj.unwrapAll()).cacheReference(o), obj = _ref9[0], idx = _ref9[1]; - } else { - idx = obj["this"] ? obj.properties[0].name : obj; - } - } - } - if (!splat && obj instanceof Splat) { - name = obj.name.unwrap().value; - obj = obj.unwrap(); - val = "" + olen + " <= " + vvarText + ".length ? " + (utility('slice')) + ".call(" + vvarText + ", " + i; - if (rest = olen - i - 1) { - ivar = o.scope.freeVariable('i'); - val += ", " + ivar + " = " + vvarText + ".length - " + rest + ") : (" + ivar + " = " + i + ", [])"; - } else { - val += ") : []"; - } - val = new Literal(val); - splat = "" + ivar + "++"; - } else { - name = obj.unwrap().value; - if (obj instanceof Splat) { - obj.error("multiple splats are disallowed in an assignment"); - } - if (typeof idx === 'number') { - idx = new Literal(splat || idx); - acc = false; - } else { - acc = isObject && IDENTIFIER.test(idx.unwrap().value || 0); - } - val = new Value(new Literal(vvarText), [new (acc ? Access : Index)(idx)]); - } - if ((name != null) && __indexOf.call(RESERVED, name) >= 0) { - obj.error("assignment to a reserved word: " + (obj.compile(o))); - } - assigns.push(new Assign(obj, val, null, { - param: this.param, - subpattern: true - }).compileToFragments(o, LEVEL_LIST)); - } - if (!(top || this.subpattern)) { - assigns.push(vvar); - } - fragments = this.joinFragmentArrays(assigns, ', '); - if (o.level < LEVEL_LIST) { - return fragments; - } else { - return this.wrapInBraces(fragments); - } - }; - - Assign.prototype.compileConditional = function(o) { - var left, right, _ref4; - _ref4 = this.variable.cacheReference(o), left = _ref4[0], right = _ref4[1]; - if (!left.properties.length && left.base instanceof Literal && left.base.value !== "this" && !o.scope.check(left.base.value)) { - this.variable.error("the variable \"" + left.base.value + "\" can't be assigned with " + this.context + " because it has not been declared before"); - } - if (__indexOf.call(this.context, "?") >= 0) { - o.isExistentialEquals = true; - } - return new Op(this.context.slice(0, -1), left, new Assign(right, this.value, '=')).compileToFragments(o); - }; - - Assign.prototype.compileSplice = function(o) { - var answer, exclusive, from, fromDecl, fromRef, name, to, valDef, valRef, _ref4, _ref5, _ref6; - _ref4 = this.variable.properties.pop().range, from = _ref4.from, to = _ref4.to, exclusive = _ref4.exclusive; - name = this.variable.compile(o); - if (from) { - _ref5 = this.cacheToCodeFragments(from.cache(o, LEVEL_OP)), fromDecl = _ref5[0], fromRef = _ref5[1]; - } else { - fromDecl = fromRef = '0'; - } - if (to) { - if ((from != null ? from.isSimpleNumber() : void 0) && to.isSimpleNumber()) { - to = +to.compile(o) - +fromRef; - if (!exclusive) { - to += 1; - } - } else { - to = to.compile(o, LEVEL_ACCESS) + ' - ' + fromRef; - if (!exclusive) { - to += ' + 1'; - } - } - } else { - to = "9e9"; - } - _ref6 = this.value.cache(o, LEVEL_LIST), valDef = _ref6[0], valRef = _ref6[1]; - answer = [].concat(this.makeCode("[].splice.apply(" + name + ", [" + fromDecl + ", " + to + "].concat("), valDef, this.makeCode(")), "), valRef); - if (o.level > LEVEL_TOP) { - return this.wrapInBraces(answer); - } else { - return answer; - } - }; - - return Assign; - - })(Base); - - exports.Code = Code = (function(_super) { - __extends(Code, _super); - - function Code(params, body, tag) { - Code.__super__.constructor.call(this); - this.params = params || []; - this.body = body || new Block; - this.icedgen = tag === 'icedgen'; - this.bound = tag === 'boundfunc' || this.icedgen; - if (this.bound || this.icedgen) { - this.context = '_this'; - } - this.icedPassedDeferral = null; - } - - Code.prototype.children = ['params', 'body']; - - Code.prototype.isStatement = function() { - return !!this.ctor; - }; - - Code.prototype.jumps = NO; - - Code.prototype.compileNode = function(o) { - var answer, code, exprs, i, idt, lit, p, param, params, ref, splats, uniqs, val, wasEmpty, _i, _j, _k, _l, _len, _len1, _len2, _len3, _len4, _m, _ref4, _ref5, _ref6, _ref7, _ref8; - o.scope = new Scope(o.scope, this.body, this); - o.scope.shared = del(o, 'sharedScope') || this.icedgen; - o.scope.icedgen = this.icedgen; - o.indent += TAB; - delete o.bare; - delete o.isExistentialEquals; - params = []; - exprs = []; - this.eachParamName(function(name) { - if (!o.scope.check(name)) { - return o.scope.parameter(name); - } - }); - _ref4 = this.params; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - param = _ref4[_i]; - if (!param.splat) { - continue; - } - _ref5 = this.params; - for (_j = 0, _len1 = _ref5.length; _j < _len1; _j++) { - p = _ref5[_j].name; - if (p["this"]) { - p = p.properties[0].name; - } - if (p.value) { - o.scope.add(p.value, 'var', true); - } - } - splats = new Assign(new Value(new Arr((function() { - var _k, _len2, _ref6, _results; - _ref6 = this.params; - _results = []; - for (_k = 0, _len2 = _ref6.length; _k < _len2; _k++) { - p = _ref6[_k]; - _results.push(p.asReference(o)); - } - return _results; - }).call(this))), new Value(new Literal('arguments'))); - break; - } - _ref6 = this.params; - for (_k = 0, _len2 = _ref6.length; _k < _len2; _k++) { - param = _ref6[_k]; - if (param.isComplex()) { - val = ref = param.asReference(o); - if (param.value) { - val = new Op('?', ref, param.value); - } - exprs.push(new Assign(new Value(param.name), val, '=', { - param: true - })); - } else { - ref = param; - if (param.value) { - lit = new Literal(ref.name.value + ' == null'); - val = new Assign(new Value(param.name), param.value, '='); - exprs.push(new If(lit, val)); - } - } - if (!splats) { - params.push(ref); - } - } - wasEmpty = this.body.isEmpty(); - if (splats) { - exprs.unshift(splats); - } - if (exprs.length) { - (_ref7 = this.body.expressions).unshift.apply(_ref7, exprs); - } - for (i = _l = 0, _len3 = params.length; _l < _len3; i = ++_l) { - p = params[i]; - params[i] = p.compileToFragments(o); - o.scope.parameter(fragmentsToText(params[i])); - } - uniqs = []; - this.eachParamName(function(name, node) { - if (__indexOf.call(uniqs, name) >= 0) { - node.error("multiple parameters named '" + name + "'"); - } - return uniqs.push(name); - }); - if (this.icedHasAutocbFlag) { - wasEmpty = false; - } - if (!(wasEmpty || this.noReturn)) { - this.body.makeReturn(); - } - if (this.bound) { - if ((_ref8 = o.scope.parent.method) != null ? _ref8.bound : void 0) { - this.bound = this.context = o.scope.parent.method.context; - } else if (!this["static"]) { - o.scope.parent.assign('_this', 'this'); - } - } - idt = o.indent; - code = 'function'; - if (this.ctor) { - code += ' ' + this.name; - } - code += '('; - answer = [this.makeCode(code)]; - for (i = _m = 0, _len4 = params.length; _m < _len4; i = ++_m) { - p = params[i]; - if (i) { - answer.push(this.makeCode(", ")); - } - answer.push.apply(answer, p); - } - answer.push(this.makeCode(') {')); - this.icedPatchBody(o); - if (!this.body.isEmpty()) { - answer = answer.concat(this.makeCode("\n"), this.body.compileWithDeclarations(o), this.makeCode("\n" + this.tab)); - } - answer.push(this.makeCode('}')); - if (this.ctor) { - return [this.makeCode(this.tab)].concat(__slice.call(answer)); - } - if (this.front || (o.level >= LEVEL_ACCESS)) { - return this.wrapInBraces(answer); - } else { - return answer; - } - }; - - Code.prototype.eachParamName = function(iterator) { - var param, _i, _len, _ref4, _results; - _ref4 = this.params; - _results = []; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - param = _ref4[_i]; - _results.push(param.eachName(iterator)); - } - return _results; - }; - - Code.prototype.traverseChildren = function(crossScope, func) { - if (crossScope) { - return Code.__super__.traverseChildren.call(this, crossScope, func); - } - }; - - Code.prototype.icedPatchBody = function(o) { - var f, lhs, r, rhs; - if (this.icedFoundArguments && this.icedNodeFlag) { - o.scope.assign('_arguments', 'arguments'); - } - if (this.icedNodeFlag && !this.icedgen) { - this.icedPassedDeferral = o.scope.freeVariable(iced["const"].passed_deferral); - lhs = new Value(new Literal(this.icedPassedDeferral)); - f = new Value(new Literal(iced["const"].ns)); - f.add(new Access(new Value(new Literal(iced["const"].findDeferral)))); - rhs = new Call(f, [new Value(new Literal('arguments'))]); - this.body.unshift(new Assign(lhs, rhs)); - } - if (this.icedNodeFlag && !this.icedgen) { - r = this.icedHasAutocbFlag ? iced["const"].autocb : iced["const"].k_noop; - rhs = new Value(new Literal(r)); - lhs = new Value(new Literal(iced["const"].k)); - return this.body.unshift(new Assign(lhs, rhs, null, { - icedlocal: true - })); - } - }; - - Code.prototype.icedWalkAst = function(parent, o) { - var cf_prev, fa_prev, faf_prev, fg_prev, param, _i, _len, _ref4; - this.icedParentAwait = parent; - fa_prev = o.foundAutocb; - cf_prev = o.currFunc; - fg_prev = o.foundArguments; - faf_prev = o.foundAwaitFunc; - o.foundAutocb = false; - o.foundArguments = false; - o.foundAwaitFunc = false; - o.currFunc = this; - _ref4 = this.params; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - param = _ref4[_i]; - if (param.name instanceof Literal && param.name.value === iced["const"].autocb) { - o.foundAutocb = true; - break; - } - } - this.icedHasAutocbFlag = o.foundAutocb; - Code.__super__.icedWalkAst.call(this, parent, o); - this.icedFoundArguments = o.foundArguments; - o.foundAwaitFunc = faf_prev; - o.foundArguments = fg_prev; - o.foundAutocb = fa_prev; - o.currFunc = cf_prev; - return false; - }; - - Code.prototype.icedWalkAstLoops = function(flood) { - if (Code.__super__.icedWalkAstLoops.call(this, false)) { - this.icedLoopFlag = true; - } - return false; - }; - - Code.prototype.icedWalkCpsPivots = function() { - Code.__super__.icedWalkCpsPivots.call(this); - return this.icedCpsPivotFlag = false; - }; - - Code.prototype.icedTraceName = function() { - var parts; - parts = []; - if (this.klass) { - parts.push(this.klass); - } - if (this.name) { - parts.push(this.name); - } - return parts.join('.'); - }; - - return Code; - - })(Base); - - exports.Param = Param = (function(_super) { - __extends(Param, _super); - - function Param(name, value, splat) { - var _ref4; - this.name = name; - this.value = value; - this.splat = splat; - Param.__super__.constructor.call(this); - if (_ref4 = (name = this.name.unwrapAll().value), __indexOf.call(STRICT_PROSCRIBED, _ref4) >= 0) { - this.name.error("parameter name \"" + name + "\" is not allowed"); - } - } - - Param.prototype.children = ['name', 'value']; - - Param.prototype.compileToFragments = function(o) { - return this.name.compileToFragments(o, LEVEL_LIST); - }; - - Param.prototype.asReference = function(o) { - var node; - if (this.reference) { - return this.reference; - } - node = this.name; - if (node["this"]) { - node = node.properties[0].name; - if (node.value.reserved) { - node = new Literal(o.scope.freeVariable(node.value)); - } - } else if (node.isComplex()) { - node = new Literal(o.scope.freeVariable('arg')); - } - node = new Value(node); - if (this.splat) { - node = new Splat(node); - } - return this.reference = node; - }; - - Param.prototype.isComplex = function() { - return this.name.isComplex(); - }; - - Param.prototype.eachName = function(iterator, name) { - var atParam, node, obj, _i, _len, _ref4; - if (name == null) { - name = this.name; - } - atParam = function(obj) { - var node; - node = obj.properties[0].name; - if (!node.value.reserved) { - return iterator(node.value, node); - } - }; - if (name instanceof Literal) { - return iterator(name.value, name); - } - if (name instanceof Value) { - return atParam(name); - } - _ref4 = name.objects; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - obj = _ref4[_i]; - if (obj instanceof Assign) { - this.eachName(iterator, obj.value.unwrap()); - } else if (obj instanceof Splat) { - node = obj.name.unwrap(); - iterator(node.value, node); - } else if (obj instanceof Value) { - if (obj.isArray() || obj.isObject()) { - this.eachName(iterator, obj.base); - } else if (obj["this"]) { - atParam(obj); - } else { - iterator(obj.base.value, obj.base); - } - } else { - obj.error("illegal parameter " + (obj.compile())); - } - } - }; - - return Param; - - })(Base); - - exports.Splat = Splat = (function(_super) { - __extends(Splat, _super); - - Splat.prototype.children = ['name']; - - Splat.prototype.isAssignable = YES; - - function Splat(name) { - Splat.__super__.constructor.call(this); - this.name = name.compile ? name : new Literal(name); - } - - Splat.prototype.assigns = function(name) { - return this.name.assigns(name); - }; - - Splat.prototype.compileToFragments = function(o) { - return this.name.compileToFragments(o); - }; - - Splat.prototype.unwrap = function() { - return this.name; - }; - - Splat.compileSplattedArray = function(o, list, apply) { - var args, base, compiledNode, concatPart, fragments, i, index, node, _i, _len; - index = -1; - while ((node = list[++index]) && !(node instanceof Splat)) { - continue; - } - if (index >= list.length) { - return []; - } - if (list.length === 1) { - node = list[0]; - fragments = node.compileToFragments(o, LEVEL_LIST); - if (apply) { - return fragments; - } - return [].concat(node.makeCode("" + (utility('slice')) + ".call("), fragments, node.makeCode(")")); - } - args = list.slice(index); - for (i = _i = 0, _len = args.length; _i < _len; i = ++_i) { - node = args[i]; - compiledNode = node.compileToFragments(o, LEVEL_LIST); - args[i] = node instanceof Splat ? [].concat(node.makeCode("" + (utility('slice')) + ".call("), compiledNode, node.makeCode(")")) : [].concat(node.makeCode("["), compiledNode, node.makeCode("]")); - } - if (index === 0) { - node = list[0]; - concatPart = node.joinFragmentArrays(args.slice(1), ', '); - return args[0].concat(node.makeCode(".concat("), concatPart, node.makeCode(")")); - } - base = (function() { - var _j, _len1, _ref4, _results; - _ref4 = list.slice(0, index); - _results = []; - for (_j = 0, _len1 = _ref4.length; _j < _len1; _j++) { - node = _ref4[_j]; - _results.push(node.compileToFragments(o, LEVEL_LIST)); - } - return _results; - })(); - base = list[0].joinFragmentArrays(base, ', '); - concatPart = list[index].joinFragmentArrays(args, ', '); - return [].concat(list[0].makeCode("["), base, list[index].makeCode("].concat("), concatPart, (last(list)).makeCode(")")); - }; - - Splat.prototype.icedToSlot = function(i) { - return new Slot(i, new Value(this.name), null, true); - }; - - return Splat; - - })(Base); - - exports.While = While = (function(_super) { - __extends(While, _super); - - function While(condition, options) { - this.condition = (options != null ? options.invert : void 0) ? condition.invert() : condition; - this.guard = options != null ? options.guard : void 0; - } - - While.prototype.children = ['condition', 'guard', 'body']; - - While.prototype.isStatement = YES; - - While.prototype.isLoop = YES; - - While.prototype.makeReturn = function(res) { - if (res) { - return While.__super__.makeReturn.apply(this, arguments); - } else { - this.returns = !this.jumps({ - loop: true - }); - return this; - } - }; - - While.prototype.addBody = function(body) { - this.body = body; - return this; - }; - - While.prototype.jumps = function() { - var expressions, node, _i, _len; - expressions = this.body.expressions; - if (!expressions.length) { - return false; - } - for (_i = 0, _len = expressions.length; _i < _len; _i++) { - node = expressions[_i]; - if (node.jumps({ - loop: true - })) { - return node; - } - } - return false; - }; - - While.prototype.compileNode = function(o) { - var answer, body, rvar, set; - this.condition.icedStatementAssertion(); - if (this.icedNodeFlag) { - return this.icedCompileIced(o); - } - o.indent += TAB; - set = ''; - body = this.body; - if (body.isEmpty()) { - body = this.makeCode(''); - } else { - if (this.returns) { - body.makeReturn(rvar = o.scope.freeVariable('results')); - set = "" + this.tab + rvar + " = [];\n"; - } - if (this.guard) { - if (body.expressions.length > 1) { - body.expressions.unshift(new If((new Parens(this.guard)).invert(), new Literal("continue"))); - } else { - if (this.guard) { - body = Block.wrap([new If(this.guard, body)]); - } - } - } - body = [].concat(this.makeCode("\n"), body.compileToFragments(o, LEVEL_TOP), this.makeCode("\n" + this.tab)); - } - answer = [].concat(this.makeCode(set + this.tab + "while ("), this.condition.compileToFragments(o, LEVEL_PAREN), this.makeCode(") {"), body, this.makeCode("}")); - if (this.returns) { - if (this.icedHasAutocbFlag) { - answer.push(this.makeCode("\n" + this.tab + iced["const"].autocb + "(" + rvar + ");")); - answer.push(this.makeCode("\n" + this.tab + "return;")); - } else { - answer.push(this.makeCode("\n" + this.tab + "return " + rvar + ";")); - } - } - return answer; - }; - - While.prototype.icedWrap = function(d) { - var body, break_assign, break_block, break_body, break_expr, break_id, call1, call2, cond, condition, continue_assign, continue_block, continue_block_inner, continue_body, continue_fn, continue_id, f, guard_if, k_id, k_param, next_arg, next_assign, next_block, next_body, next_id, outStatements, rvar, rvar_init, rvar_value, top_assign, top_block, top_body, top_call, top_func, top_id, top_statements, tramp; - condition = d.condition; - body = d.body; - rvar = d.rvar; - outStatements = []; - if (rvar) { - rvar_value = new Value(new Literal(rvar)); - } - top_id = new Value(new Literal(iced["const"].t_while)); - k_id = new Value(new Literal(iced["const"].k)); - k_param = new Param(new Literal(iced["const"].k)); - break_id = new Value(new Literal(iced["const"].b_while)); - if (rvar) { - break_expr = new Call(k_id, [rvar_value]); - break_block = new Block([break_expr]); - break_body = new Code([], break_block, 'icedgen'); - break_assign = new Assign(break_id, break_body, null, { - icedlocal: true - }); - } else { - break_assign = new Assign(break_id, k_id, null, { - icedlocal: true - }); - } - continue_id = new Value(new Literal(iced["const"].c_while)); - continue_block_inner = new Block([new Call(top_id, [k_id])]); - if (d.step) { - continue_block_inner.unshift(d.step); - } - continue_fn = new Code([], continue_block_inner); - tramp = new Value(new Literal(iced["const"].ns)); - tramp.add(new Access(new Value(new Literal(iced["const"].trampoline)))); - continue_block = new Block([new Call(tramp, [continue_fn])]); - continue_body = new Code([], continue_block, 'icedgen'); - continue_assign = new Assign(continue_id, continue_body, null, { - icedlocal: true - }); - next_id = new Value(new Literal(iced["const"].n_while)); - if (rvar) { - next_arg = new Param(new Literal(iced["const"].n_arg)); - f = rvar_value.copy(); - f.add(new Access(new Value(new Literal('push')))); - call1 = new Call(f, [next_arg]); - call2 = new Call(continue_id, []); - next_block = new Block([call1, call2]); - next_body = new Code([next_arg], next_block, 'icedgen'); - next_assign = new Assign(next_id, next_body, null, { - icedlocal: true - }); - } else { - next_assign = new Assign(next_id, continue_id); - } - cond = new If(condition.invert(), new Block([new Call(break_id, [])])); - if (d.guard) { - continue_block = new Block([new Call(continue_id, [])]); - guard_if = new If(d.guard, body); - guard_if.addElse(continue_block); - cond.addElse(new Block([d.pre_body, guard_if])); - } else { - cond.addElse(new Block([d.pre_body, body])); - } - top_body = new Block([break_assign, continue_assign, next_assign, cond]); - top_func = new Code([k_param], top_body, 'icedgen'); - top_assign = new Assign(top_id, top_func, null, { - icedlocal: true - }); - top_call = new Call(top_id, [k_id]); - top_statements = []; - if (d.init) { - top_statements = top_statements.concat(d.init); - } - if (rvar) { - rvar_init = new Assign(rvar_value, new Arr); - top_statements.push(rvar_init); - } - top_statements = top_statements.concat([top_assign, top_call]); - return top_block = new Block(top_statements); - }; - - While.prototype.icedCallContinuation = function() { - return this.body.icedThreadReturn(new IcedTailCall(iced["const"].n_while)); - }; - - While.prototype.icedCompileIced = function(o) { - var b, opts; - opts = { - condition: this.condition, - body: this.body, - guard: this.guard - }; - if (this.returns) { - opts.rvar = o.scope.freeVariable('results'); - } - b = this.icedWrap(opts); - return b.compileNode(o); - }; - - return While; - - })(Base); - - exports.Op = Op = (function(_super) { - var CONVERSIONS, INVERSIONS; - - __extends(Op, _super); - - function Op(op, first, second, flip) { - Op.__super__.constructor.call(this); - if (op === 'in') { - return new In(first, second); - } - if (op === 'do') { - return this.generateDo(first); - } - if (op === 'new') { - if (first instanceof Call && !first["do"] && !first.isNew) { - return first.newInstance(); - } - if (first instanceof Code && first.bound || first["do"]) { - first = new Parens(first); - } - } - this.operator = CONVERSIONS[op] || op; - this.first = first; - this.second = second; - this.flip = !!flip; - return this; - } - - CONVERSIONS = { - '==': '===', - '!=': '!==', - 'of': 'in' - }; - - INVERSIONS = { - '!==': '===', - '===': '!==' - }; - - Op.prototype.children = ['first', 'second']; - - Op.prototype.isSimpleNumber = NO; - - Op.prototype.isUnary = function() { - return !this.second; - }; - - Op.prototype.isComplex = function() { - var _ref4; - return !(this.isUnary() && ((_ref4 = this.operator) === '+' || _ref4 === '-')) || this.first.isComplex(); - }; - - Op.prototype.isChainable = function() { - var _ref4; - return (_ref4 = this.operator) === '<' || _ref4 === '>' || _ref4 === '>=' || _ref4 === '<=' || _ref4 === '===' || _ref4 === '!=='; - }; - - Op.prototype.invert = function() { - var allInvertable, curr, fst, op, _ref4; - if (this.isChainable() && this.first.isChainable()) { - allInvertable = true; - curr = this; - while (curr && curr.operator) { - allInvertable && (allInvertable = curr.operator in INVERSIONS); - curr = curr.first; - } - if (!allInvertable) { - return new Parens(this).invert(); - } - curr = this; - while (curr && curr.operator) { - curr.invert = !curr.invert; - curr.operator = INVERSIONS[curr.operator]; - curr = curr.first; - } - return this; - } else if (op = INVERSIONS[this.operator]) { - this.operator = op; - if (this.first.unwrap() instanceof Op) { - this.first.invert(); - } - return this; - } else if (this.second) { - return new Parens(this).invert(); - } else if (this.operator === '!' && (fst = this.first.unwrap()) instanceof Op && ((_ref4 = fst.operator) === '!' || _ref4 === 'in' || _ref4 === 'instanceof')) { - return fst; - } else { - return new Op('!', this); - } - }; - - Op.prototype.unfoldSoak = function(o) { - var _ref4; - return ((_ref4 = this.operator) === '++' || _ref4 === '--' || _ref4 === 'delete') && unfoldSoak(o, this, 'first'); - }; - - Op.prototype.generateDo = function(exp) { - var call, func, param, passedParams, ref, _i, _len, _ref4; - passedParams = []; - func = exp instanceof Assign && (ref = exp.value.unwrap()) instanceof Code ? ref : exp; - _ref4 = func.params || []; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - param = _ref4[_i]; - if (param.value) { - passedParams.push(param.value); - delete param.value; - } else { - passedParams.push(param); - } - } - call = new Call(exp, passedParams); - call["do"] = true; - return call; - }; - - Op.prototype.compileNode = function(o) { - var answer, isChain, _ref4, _ref5; - isChain = this.isChainable() && this.first.isChainable(); - if (!isChain) { - this.first.front = this.front; - } - if (this.operator === 'delete' && o.scope.check(this.first.unwrapAll().value)) { - this.error('delete operand may not be argument or var'); - } - if (((_ref4 = this.operator) === '--' || _ref4 === '++') && (_ref5 = this.first.unwrapAll().value, __indexOf.call(STRICT_PROSCRIBED, _ref5) >= 0)) { - this.error("cannot increment/decrement \"" + (this.first.unwrapAll().value) + "\""); - } - if (this.isUnary()) { - return this.compileUnary(o); - } - if (isChain) { - return this.compileChain(o); - } - if (this.operator === '?') { - return this.compileExistence(o); - } - answer = [].concat(this.first.compileToFragments(o, LEVEL_OP), this.makeCode(' ' + this.operator + ' '), this.second.compileToFragments(o, LEVEL_OP)); - if (o.level <= LEVEL_OP) { - return answer; - } else { - return this.wrapInBraces(answer); - } - }; - - Op.prototype.compileChain = function(o) { - var fragments, fst, shared, _ref4; - _ref4 = this.first.second.cache(o), this.first.second = _ref4[0], shared = _ref4[1]; - fst = this.first.compileToFragments(o, LEVEL_OP); - fragments = fst.concat(this.makeCode(" " + (this.invert ? '&&' : '||') + " "), shared.compileToFragments(o), this.makeCode(" " + this.operator + " "), this.second.compileToFragments(o, LEVEL_OP)); - return this.wrapInBraces(fragments); - }; - - Op.prototype.compileExistence = function(o) { - var fst, ref; - if (!o.isExistentialEquals && this.first.isComplex()) { - ref = new Literal(o.scope.freeVariable('ref')); - fst = new Parens(new Assign(ref, this.first)); - } else { - fst = this.first; - ref = fst; - } - return new If(new Existence(fst), ref, { - type: 'if' - }).addElse(this.second).compileToFragments(o); - }; - - Op.prototype.compileUnary = function(o) { - var op, parts, plusMinus; - parts = []; - op = this.operator; - parts.push([this.makeCode(op)]); - if (op === '!' && this.first instanceof Existence) { - this.first.negated = !this.first.negated; - return this.first.compileToFragments(o); - } - if (o.level >= LEVEL_ACCESS) { - return (new Parens(this)).compileToFragments(o); - } - plusMinus = op === '+' || op === '-'; - if ((op === 'new' || op === 'typeof' || op === 'delete') || plusMinus && this.first instanceof Op && this.first.operator === op) { - parts.push([this.makeCode(' ')]); - } - if ((plusMinus && this.first instanceof Op) || (op === 'new' && this.first.isStatement(o))) { - this.first = new Parens(this.first); - } - parts.push(this.first.compileToFragments(o, LEVEL_OP)); - if (this.flip) { - parts.reverse(); - } - return this.joinFragmentArrays(parts, ''); - }; - - Op.prototype.toString = function(idt) { - return Op.__super__.toString.call(this, idt, this.constructor.name + ' ' + this.operator); - }; - - Op.prototype.icedWrapContinuation = function() { - return this.icedCallContinuationFlag; - }; - - return Op; - - })(Base); - - exports.In = In = (function(_super) { - __extends(In, _super); - - function In(object, array) { - this.object = object; - this.array = array; - In.__super__.constructor.call(this); - } - - In.prototype.children = ['object', 'array']; - - In.prototype.invert = NEGATE; - - In.prototype.compileNode = function(o) { - var hasSplat, obj, _i, _len, _ref4; - if (this.array instanceof Value && this.array.isArray()) { - _ref4 = this.array.base.objects; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - obj = _ref4[_i]; - if (!(obj instanceof Splat)) { - continue; - } - hasSplat = true; - break; - } - if (!hasSplat) { - return this.compileOrTest(o); - } - } - return this.compileLoopTest(o); - }; - - In.prototype.compileOrTest = function(o) { - var cmp, cnj, i, item, ref, sub, tests, _i, _len, _ref4, _ref5, _ref6; - if (this.array.base.objects.length === 0) { - return [this.makeCode("" + (!!this.negated))]; - } - _ref4 = this.object.cache(o, LEVEL_OP), sub = _ref4[0], ref = _ref4[1]; - _ref5 = this.negated ? [' !== ', ' && '] : [' === ', ' || '], cmp = _ref5[0], cnj = _ref5[1]; - tests = []; - _ref6 = this.array.base.objects; - for (i = _i = 0, _len = _ref6.length; _i < _len; i = ++_i) { - item = _ref6[i]; - if (i) { - tests.push(this.makeCode(cnj)); - } - tests = tests.concat((i ? ref : sub), this.makeCode(cmp), item.compileToFragments(o, LEVEL_ACCESS)); - } - if (o.level < LEVEL_OP) { - return tests; - } else { - return this.wrapInBraces(tests); - } - }; - - In.prototype.compileLoopTest = function(o) { - var fragments, ref, sub, _ref4; - _ref4 = this.object.cache(o, LEVEL_LIST), sub = _ref4[0], ref = _ref4[1]; - fragments = [].concat(this.makeCode(utility('indexOf') + ".call("), this.array.compileToFragments(o, LEVEL_LIST), this.makeCode(", "), ref, this.makeCode(") " + (this.negated ? '< 0' : '>= 0'))); - if (fragmentsToText(sub) === fragmentsToText(ref)) { - return fragments; - } - fragments = sub.concat(this.makeCode(', '), fragments); - if (o.level < LEVEL_LIST) { - return fragments; - } else { - return this.wrapInBraces(fragments); - } - }; - - In.prototype.toString = function(idt) { - return In.__super__.toString.call(this, idt, this.constructor.name + (this.negated ? '!' : '')); - }; - - return In; - - })(Base); - - exports.Slot = Slot = (function(_super) { - __extends(Slot, _super); - - function Slot(index, value, suffix, splat) { - Slot.__super__.constructor.call(this); - this.index = index; - this.value = value; - this.suffix = suffix; - this.splat = splat; - this.access = null; - } - - Slot.prototype.addAccess = function(a) { - this.access = a; - return this; - }; - - Slot.prototype.children = ['value', 'suffix']; - - return Slot; - - })(Base); - - exports.Defer = Defer = (function(_super) { - __extends(Defer, _super); - - function Defer(args, lineno) { - var a, i; - this.lineno = lineno; - Defer.__super__.constructor.call(this); - this.slots = flatten((function() { - var _i, _len, _results; - _results = []; - for (i = _i = 0, _len = args.length; _i < _len; i = ++_i) { - a = args[i]; - _results.push(a.icedToSlot(i)); - } - return _results; - })()); - this.params = []; - this.vars = []; - this.custom = false; - } - - Defer.prototype.children = ['slots']; - - Defer.prototype.setCustom = function() { - this.custom = true; - return this; - }; - - Defer.prototype.newParam = function() { - var l; - l = "" + iced["const"].slot + "_" + (this.params.length + 1); - this.params.push(new Param(new Literal(l))); - return new Value(new Literal(l)); - }; - - Defer.prototype.makeAssignFn = function(o) { - var a, args, assign, assignments, block, call, func, i, i_lit, inner_fn, lit, outer_block, outer_fn, prop, s, slot, _i, _len, _ref4; - if (this.slots.length === 0) { - return null; - } - assignments = []; - args = []; - i = 0; - _ref4 = this.slots; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - s = _ref4[_i]; - i = s.index; - a = new Value(new Literal("arguments")); - i_lit = new Value(new Literal(i)); - if (s.splat) { - func = new Value(new Literal(utility('slice'))); - func.add(new Access(new Value(new Literal('call')))); - call = new Call(func, [a, i_lit]); - slot = s.value; - this.vars.push(slot); - assign = new Assign(slot, call); - } else { - a.add(new Index(i_lit)); - if (s.access) { - a.add(s.access); - } - if (!s.suffix) { - lit = s.value.compile(o, LEVEL_TOP); - if (lit === "_") { - slot = new Value(new Literal(iced["const"].deferrals)); - slot.add(new Access(new Value(new Literal(iced["const"].retslot)))); - } else { - slot = s.value; - this.vars.push(slot); - } - } else { - args.push(s.value); - slot = this.newParam(); - if (s.suffix instanceof Index) { - prop = new Index(this.newParam()); - args.push(s.suffix.index); - } else { - prop = s.suffix; - } - slot.add(prop); - } - assign = new Assign(slot, a); - } - assignments.push(assign); - } - block = new Block(assignments); - inner_fn = new Code([], block, 'icedgen'); - outer_block = new Block([new Return(inner_fn)]); - outer_fn = new Code(this.params, outer_block, 'icedgen'); - return call = new Call(outer_fn, args); - }; - - Defer.prototype.transform = function(o) { - var assign_fn, assignments, context_assign, context_lhs, context_rhs, fn, ln_assign, ln_lhs, ln_rhs, meth; - meth = new Value(new Literal(iced["const"].defer_method)); - if (this.custom) { - fn = meth; - } else { - fn = new Value(new Literal(iced["const"].deferrals)); - fn.add(new Access(meth)); - } - assignments = []; - if ((assign_fn = this.makeAssignFn(o))) { - assignments.push(new Assign(new Value(new Literal(iced["const"].assign_fn)), assign_fn, "object")); - } - ln_lhs = new Value(new Literal(iced["const"].lineno)); - ln_rhs = new Value(new Literal(this.lineno)); - ln_assign = new Assign(ln_lhs, ln_rhs, "object"); - assignments.push(ln_assign); - if (this.custom) { - context_lhs = new Value(new Literal(iced["const"].context)); - context_rhs = new Value(new Literal(iced["const"].deferrals)); - context_assign = new Assign(context_lhs, context_rhs, "object"); - assignments.push(context_assign); - } - o = new Obj(assignments); - return new Call(fn, [new Value(o)]); - }; - - Defer.prototype.compileNode = function(o) { - var call, name, scope, v, _i, _len, _ref4; - call = this.transform(o); - _ref4 = this.vars; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - v = _ref4[_i]; - name = v.compile(o, LEVEL_LIST); - scope = o.scope; - scope.add(name, 'var'); - } - return call.compileNode(o); - }; - - Defer.prototype.icedWalkAst = function(p, o) { - this.icedHasAutocbFlag = o.foundAutocb; - o.foundDefer = true; - this.parentFunc = o.currFunc; - return Defer.__super__.icedWalkAst.call(this, p, o); - }; - - return Defer; - - })(Base); - - exports.Await = Await = (function(_super) { - __extends(Await, _super); - - function Await(body) { - this.body = body; - Await.__super__.constructor.call(this); - } - - Await.prototype.transform = function(o) { - var assign, assignments, body, call, cb_assignment, cb_lhs, cb_rhs, cls, fn_assignment, fn_lhs, fn_rhs, func_assignment, func_lhs, func_rhs, lhs, meth, n, name, rhs, trace, _ref4, _ref5; - body = this.body; - name = iced["const"].deferrals; - o.scope.add(name, 'var'); - lhs = new Value(new Literal(name)); - cls = new Value(new Literal(iced["const"].ns)); - cls.add(new Access(new Value(new Literal(iced["const"].Deferrals)))); - assignments = []; - if (n = (_ref4 = this.parentFunc) != null ? _ref4.icedPassedDeferral : void 0) { - cb_lhs = new Value(new Literal(iced["const"].parent)); - cb_rhs = new Value(new Literal(n)); - cb_assignment = new Assign(cb_lhs, cb_rhs, "object"); - assignments.push(cb_assignment); - } - if (o.filename != null) { - fn_lhs = new Value(new Literal(iced["const"].filename)); - fn_rhs = new Value(new Literal('"' + o.filename.replace('\\', '\\\\') + '"')); - fn_assignment = new Assign(fn_lhs, fn_rhs, "object"); - assignments.push(fn_assignment); - } - if (n = (_ref5 = this.parentFunc) != null ? _ref5.icedTraceName() : void 0) { - func_lhs = new Value(new Literal(iced["const"].funcname)); - func_rhs = new Value(new Literal('"' + n + '"')); - func_assignment = new Assign(func_lhs, func_rhs, "object"); - assignments.push(func_assignment); - } - trace = new Obj(assignments, true); - call = new Call(cls, [new Value(new Literal(iced["const"].k)), trace]); - rhs = new Op("new", call); - assign = new Assign(lhs, rhs); - body.unshift(assign); - meth = lhs.copy().add(new Access(new Value(new Literal(iced["const"].fulfill)))); - call = new Call(meth, []); - body.push(call); - return this.body = body; - }; - - Await.prototype.children = ['body']; - - Await.prototype.isStatement = function() { - return YES; - }; - - Await.prototype.makeReturn = THIS; - - Await.prototype.compileNode = function(o) { - this.transform(o); - return this.body.compileNode(o); - }; - - Await.prototype.icedWalkAst = function(p, o) { - this.icedHasAutocbFlag = o.foundAutocb; - this.parentFunc = o.currFunc; - p = p || this; - this.icedParentAwait = p; - Await.__super__.icedWalkAst.call(this, p, o); - return this.icedNodeFlag = o.foundAwaitFunc = o.foundAwait = true; - }; - - return Await; - - })(Base); - - IcedRuntime = (function(_super) { - __extends(IcedRuntime, _super); - - function IcedRuntime(foundDefer, foundAwait) { - this.foundDefer = foundDefer; - this.foundAwait = foundAwait; - IcedRuntime.__super__.constructor.call(this); - } - - IcedRuntime.prototype.compileNode = function(o, level) { - var access, accessname, assign, call, callv, file, inc, k, klass, lhs_vec, modname, ns, req, rhs, v, val, window_mode, window_val, _i, _j, _len, _len1, _ref4; - this.expressions = []; - v = o.runtime ? o.runtime : o.bare ? "none" : this.foundDefer ? "node" : "none"; - if (o.runtime && !this.foundDefer && !o.runforce) { - v = "none"; - } - window_mode = false; - window_val = null; - inc = null; - inc = (function() { - switch (v) { - case "inline": - case "window": - if (v === "window") { - window_mode = true; - } - if (window_mode) { - window_val = new Value(new Literal(v)); - } - return InlineRuntime.generate(window_val ? window_val.copy() : null); - case "node": - case "browserify": - if (v === "browserify") { - modname = "iced-coffee-script/lib/coffee-script/iced"; - accessname = iced["const"].runtime; - } else { - modname = "iced-coffee-script"; - accessname = iced["const"].ns; - } - file = new Literal("'" + modname + "'"); - access = new Access(new Literal(accessname)); - req = new Value(new Literal("require")); - call = new Call(req, [file]); - callv = new Value(call); - callv.add(access); - ns = new Value(new Literal(iced["const"].ns)); - return new Assign(ns, callv); - case "none": - return null; - default: - throw SyntaxError("unexpected flag IcedRuntime " + v); - } - })(); - if (inc) { - this.push(inc); - } - if (this.foundAwait) { - rhs = new Code([], new Block([])); - lhs_vec = []; - _ref4 = [iced["const"].k_noop, iced["const"].k]; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - k = _ref4[_i]; - val = new Value(new Literal(k)); - if (window_val) { - klass = window_val.copy(); - klass.add(new Access(val)); - val = klass; - } - lhs_vec.push(val); - } - assign = rhs; - for (_j = 0, _len1 = lhs_vec.length; _j < _len1; _j++) { - v = lhs_vec[_j]; - assign = new Assign(v, assign); - } - this.push(assign); - } - if (this.isEmpty()) { - return []; - } else { - return IcedRuntime.__super__.compileNode.call(this, o); - } - }; - - IcedRuntime.prototype.icedWalkAst = function(p, o) { - this.icedHasAutocbFlag = o.foundAutocb; - return IcedRuntime.__super__.icedWalkAst.call(this, p, o); - }; - - return IcedRuntime; - - })(Block); - - exports.Try = Try = (function(_super) { - __extends(Try, _super); - - function Try(attempt, errorVariable, recovery, ensure) { - this.attempt = attempt; - this.errorVariable = errorVariable; - this.recovery = recovery; - this.ensure = ensure; - } - - Try.prototype.children = ['attempt', 'recovery', 'ensure']; - - Try.prototype.isStatement = YES; - - Try.prototype.jumps = function(o) { - var _ref4; - return this.attempt.jumps(o) || ((_ref4 = this.recovery) != null ? _ref4.jumps(o) : void 0); - }; - - Try.prototype.makeReturn = function(res) { - if (this.attempt) { - this.attempt = this.attempt.makeReturn(res); - } - if (this.recovery) { - this.recovery = this.recovery.makeReturn(res); - } - return this; - }; - - Try.prototype.compileNode = function(o) { - var catchPart, ensurePart, placeholder, tryPart; - o.indent += TAB; - tryPart = this.attempt.compileToFragments(o, LEVEL_TOP); - catchPart = this.recovery ? (placeholder = new Literal('_error'), this.errorVariable ? this.recovery.unshift(new Assign(this.errorVariable, placeholder)) : void 0, [].concat(this.makeCode(" catch ("), placeholder.compileToFragments(o), this.makeCode(") {\n"), this.recovery.compileToFragments(o, LEVEL_TOP), this.makeCode("\n" + this.tab + "}"))) : !(this.ensure || this.recovery) ? [this.makeCode(' catch (_error) {}')] : []; - ensurePart = this.ensure ? [].concat(this.makeCode(" finally {\n"), this.ensure.compileToFragments(o, LEVEL_TOP), this.makeCode("\n" + this.tab + "}")) : []; - return [].concat(this.makeCode("" + this.tab + "try {\n"), tryPart, this.makeCode("\n" + this.tab + "}"), catchPart, ensurePart); - }; - - return Try; - - })(Base); - - exports.Throw = Throw = (function(_super) { - __extends(Throw, _super); - - function Throw(expression) { - this.expression = expression; - Throw.__super__.constructor.call(this); - } - - Throw.prototype.children = ['expression']; - - Throw.prototype.isStatement = YES; - - Throw.prototype.jumps = NO; - - Throw.prototype.makeReturn = THIS; - - Throw.prototype.compileNode = function(o) { - return [].concat(this.makeCode(this.tab + "throw "), this.expression.compileToFragments(o), this.makeCode(";")); - }; - - return Throw; - - })(Base); - - exports.Existence = Existence = (function(_super) { - __extends(Existence, _super); - - function Existence(expression) { - this.expression = expression; - Existence.__super__.constructor.call(this); - } - - Existence.prototype.children = ['expression']; - - Existence.prototype.invert = NEGATE; - - Existence.prototype.compileNode = function(o) { - var cmp, cnj, code, _ref4; - this.expression.front = this.front; - code = this.expression.compile(o, LEVEL_OP); - if (IDENTIFIER.test(code) && !o.scope.check(code)) { - _ref4 = this.negated ? ['===', '||'] : ['!==', '&&'], cmp = _ref4[0], cnj = _ref4[1]; - code = "typeof " + code + " " + cmp + " \"undefined\" " + cnj + " " + code + " " + cmp + " null"; - } else { - code = "" + code + " " + (this.negated ? '==' : '!=') + " null"; - } - return [this.makeCode(o.level <= LEVEL_COND ? code : "(" + code + ")")]; - }; - - return Existence; - - })(Base); - - exports.Parens = Parens = (function(_super) { - __extends(Parens, _super); - - function Parens(body) { - this.body = body; - Parens.__super__.constructor.call(this); - } - - Parens.prototype.children = ['body']; - - Parens.prototype.unwrap = function() { - return this.body; - }; - - Parens.prototype.isComplex = function() { - return this.body.isComplex(); - }; - - Parens.prototype.compileNode = function(o) { - var bare, expr, fragments; - expr = this.body.unwrap(); - if (expr instanceof Value && expr.isAtomic()) { - expr.front = this.front; - return expr.compileToFragments(o); - } - fragments = expr.compileToFragments(o, LEVEL_PAREN); - bare = o.level < LEVEL_OP && (expr instanceof Op || expr instanceof Call || (expr instanceof For && expr.returns)); - if (bare) { - return fragments; - } else { - return this.wrapInBraces(fragments); - } - }; - - return Parens; - - })(Base); - - exports.For = For = (function(_super) { - __extends(For, _super); - - function For(body, source) { - var _ref4; - For.__super__.constructor.call(this); - this.source = source.source, this.guard = source.guard, this.step = source.step, this.name = source.name, this.index = source.index; - this.body = Block.wrap([body]); - this.own = !!source.own; - this.object = !!source.object; - if (this.object) { - _ref4 = [this.index, this.name], this.name = _ref4[0], this.index = _ref4[1]; - } - if (this.index instanceof Value) { - this.index.error('index cannot be a pattern matching expression'); - } - this.range = this.source instanceof Value && this.source.base instanceof Range && !this.source.properties.length; - this.pattern = this.name instanceof Value; - if (this.range && this.index) { - this.index.error('indexes do not apply to range loops'); - } - if (this.range && this.pattern) { - this.name.error('cannot pattern match over range loops'); - } - if (this.own && !this.object) { - this.index.error('cannot use own with for-in'); - } - this.returns = false; - } - - For.prototype.children = ['body', 'source', 'guard', 'step']; - - For.prototype.compileNode = function(o) { - var body, bodyFragments, compare, compareDown, declare, declareDown, defPart, defPartFragments, down, forPartFragments, guardPart, idt1, increment, index, ivar, kvar, kvarAssign, lastJumps, lvar, name, namePart, ref, resultPart, returnResult, rvar, scope, source, step, stepNum, stepVar, svar, varPart, _ref4, _ref5; - body = Block.wrap([this.body]); - lastJumps = (_ref4 = last(body.expressions)) != null ? _ref4.jumps() : void 0; - if (lastJumps && lastJumps instanceof Return) { - this.returns = false; - } - source = this.range ? this.source.base : this.source; - scope = o.scope; - name = this.name && (this.name.compile(o, LEVEL_LIST)); - index = this.index && (this.index.compile(o, LEVEL_LIST)); - if (name && !this.pattern) { - scope.find(name); - } - if (index) { - scope.find(index); - } - if (this.returns) { - rvar = scope.freeVariable('results'); - } - ivar = (this.object && index) || scope.freeVariable('i'); - kvar = (this.range && name) || index || ivar; - kvarAssign = kvar !== ivar ? "" + kvar + " = " : ""; - if (this.step && !this.range) { - _ref5 = this.cacheToCodeFragments(this.step.cache(o, LEVEL_LIST)), step = _ref5[0], stepVar = _ref5[1]; - stepNum = stepVar.match(SIMPLENUM); - } - if (this.pattern) { - name = ivar; - } - varPart = ''; - guardPart = ''; - defPart = ''; - idt1 = this.tab + TAB; - source.icedStatementAssertion(); - if (this.icedNodeFlag) { - return this.icedCompileIced(o, { - stepVar: stepVar, - body: body, - rvar: rvar, - kvar: kvar, - guard: this.guard - }); - } - if (this.range) { - forPartFragments = source.compileToFragments(merge(o, { - index: ivar, - name: name, - step: this.step - })); - } else { - svar = this.source.compile(o, LEVEL_LIST); - if ((name || this.own) && !IDENTIFIER.test(svar)) { - defPart += "" + this.tab + (ref = scope.freeVariable('ref')) + " = " + svar + ";\n"; - svar = ref; - } - if (name && !this.pattern) { - namePart = "" + name + " = " + svar + "[" + kvar + "]"; - } - if (!this.object) { - if (step !== stepVar) { - defPart += "" + this.tab + step + ";\n"; - } - if (!(this.step && stepNum && (down = +stepNum < 0))) { - lvar = scope.freeVariable('len'); - } - declare = "" + kvarAssign + ivar + " = 0, " + lvar + " = " + svar + ".length"; - declareDown = "" + kvarAssign + ivar + " = " + svar + ".length - 1"; - compare = "" + ivar + " < " + lvar; - compareDown = "" + ivar + " >= 0"; - if (this.step) { - if (stepNum) { - if (down) { - compare = compareDown; - declare = declareDown; - } - } else { - compare = "" + stepVar + " > 0 ? " + compare + " : " + compareDown; - declare = "(" + stepVar + " > 0 ? (" + declare + ") : " + declareDown + ")"; - } - increment = "" + ivar + " += " + stepVar; - } else { - increment = "" + (kvar !== ivar ? "++" + ivar : "" + ivar + "++"); - } - forPartFragments = [this.makeCode("" + declare + "; " + compare + "; " + kvarAssign + increment)]; - } - } - if (this.returns) { - resultPart = "" + this.tab + rvar + " = [];\n"; - returnResult = this.icedHasAutocbFlag ? "\n" + this.tab + iced["const"].autocb + "(" + rvar + "); return;" : "\n" + this.tab + "return " + rvar + ";"; - body.makeReturn(rvar); - } - if (this.guard) { - if (body.expressions.length > 1) { - body.expressions.unshift(new If((new Parens(this.guard)).invert(), new Literal("continue"))); - } else { - if (this.guard) { - body = Block.wrap([new If(this.guard, body)]); - } - } - } - if (this.pattern) { - body.expressions.unshift(new Assign(this.name, new Literal("" + svar + "[" + kvar + "]"))); - } - defPartFragments = [].concat(this.makeCode(defPart), this.pluckDirectCall(o, body)); - if (namePart) { - varPart = "\n" + idt1 + namePart + ";"; - } - if (this.object) { - forPartFragments = [this.makeCode("" + kvar + " in " + svar)]; - if (this.own) { - guardPart = "\n" + idt1 + "if (!" + (utility('hasProp')) + ".call(" + svar + ", " + kvar + ")) continue;"; - } - } - bodyFragments = body.compileToFragments(merge(o, { - indent: idt1 - }), LEVEL_TOP); - if (bodyFragments && (bodyFragments.length > 0)) { - bodyFragments = [].concat(this.makeCode("\n"), bodyFragments, this.makeCode("\n")); - } - return [].concat(defPartFragments, this.makeCode("" + (resultPart || '') + this.tab + "for ("), forPartFragments, this.makeCode(") {" + guardPart + varPart), bodyFragments, this.makeCode("" + this.tab + "}" + (returnResult || ''))); - }; - - For.prototype.pluckDirectCall = function(o, body) { - var base, defs, expr, fn, idx, ref, val, _i, _len, _ref4, _ref5, _ref6, _ref7, _ref8, _ref9; - defs = []; - _ref4 = body.expressions; - for (idx = _i = 0, _len = _ref4.length; _i < _len; idx = ++_i) { - expr = _ref4[idx]; - expr = expr.unwrapAll(); - if (!(expr instanceof Call)) { - continue; - } - val = expr.variable.unwrapAll(); - if (!((val instanceof Code) || (val instanceof Value && ((_ref5 = val.base) != null ? _ref5.unwrapAll() : void 0) instanceof Code && val.properties.length === 1 && ((_ref6 = (_ref7 = val.properties[0].name) != null ? _ref7.value : void 0) === 'call' || _ref6 === 'apply')))) { - continue; - } - fn = ((_ref8 = val.base) != null ? _ref8.unwrapAll() : void 0) || val; - ref = new Literal(o.scope.freeVariable('fn')); - base = new Value(ref); - if (val.base) { - _ref9 = [base, val], val.base = _ref9[0], base = _ref9[1]; - } - body.expressions[idx] = new Call(base, expr.args); - defs = defs.concat(this.makeCode(this.tab), new Assign(ref, fn).compileToFragments(o, LEVEL_TOP), this.makeCode(';\n')); - } - return defs; - }; - - For.prototype.icedCompileIced = function(o, d) { - var a1, a2, a3, a4, a5, b, body, condition, empty_arr, guard, iname, init, ival, key, key_lit, key_val, keys, keys_access, keys_len, keys_val, kval, len, len_rhs, len_val, loop_body, loop_keys, loop_source, pos, pre_body, ref, ref_val, ref_val_copy, rop, rvar, scope, source_access, step; - body = d.body; - condition = null; - init = []; - step = null; - scope = o.scope; - pre_body = new Block([]); - if (this.object) { - ref = scope.freeVariable('ref'); - ref_val = new Value(new Literal(ref)); - a1 = new Assign(ref_val, this.source); - keys = scope.freeVariable('keys'); - keys_val = new Value(new Literal(keys)); - key = scope.freeVariable('k'); - key_lit = new Literal(key); - key_val = new Value(key_lit); - empty_arr = new Value(new Arr); - loop_body = new Block([key_val]); - loop_source = { - object: true, - name: key_lit, - source: ref_val - }; - loop_keys = new For(loop_body, loop_source); - a2 = new Assign(keys_val, loop_keys); - iname = scope.freeVariable('i'); - ival = new Value(new Literal(iname)); - a3 = new Assign(ival, new Value(new Literal(0))); - init = [a1, a2, a3]; - keys_len = keys_val.copy(); - keys_len.add(new Access(new Value(new Literal("length")))); - condition = new Op('<', ival, keys_len); - step = new Op('++', ival); - if (this.name) { - source_access = ref_val.copy(); - source_access.add(new Index(this.index)); - a5 = new Assign(this.name, source_access); - pre_body.unshift(a5); - } - keys_access = keys_val.copy(); - keys_access.add(new Index(ival)); - a4 = new Assign(this.index, keys_access); - pre_body.unshift(a4); - } else if (this.range && this.name) { - pos = this.source.base.from <= this.source.base.to; - rop = this.source.base.exclusive ? (pos ? '<' : '>') : (pos ? '<=' : '>='); - condition = new Op(rop, this.name, this.source.base.to); - init = [new Assign(this.name, this.source.base.from)]; - if (this.step != null) { - step = new Op((pos ? "+=" : "-="), this.name, this.step); - } else { - step = new Op((pos ? '++' : "--"), this.name); - } - } else if (!this.range && this.name) { - kval = new Value(new Literal(d.kvar)); - len = scope.freeVariable('len'); - ref = scope.freeVariable('ref'); - ref_val = new Value(new Literal(ref)); - len_val = new Value(new Literal(len)); - a1 = new Assign(ref_val, this.source); - len_rhs = ref_val.copy().add(new Access(new Value(new Literal("length")))); - a2 = new Assign(len_val, len_rhs); - a3 = new Assign(kval, new Value(new Literal(0))); - init = [a1, a2, a3]; - condition = new Op('<', kval, len_val); - step = new Op('++', kval); - ref_val_copy = ref_val.copy(); - ref_val_copy.add(new Index(kval)); - a4 = new Assign(this.name, ref_val_copy); - pre_body.unshift(a4); - } - rvar = d.rvar; - guard = d.guard; - b = this.icedWrap({ - condition: condition, - body: body, - init: init, - step: step, - rvar: rvar, - guard: guard, - pre_body: pre_body - }); - return b.compileNode(o); - }; - - return For; - - })(While); - - exports.Switch = Switch = (function(_super) { - __extends(Switch, _super); - - function Switch(subject, cases, otherwise) { - this.subject = subject; - this.cases = cases; - this.otherwise = otherwise; - Switch.__super__.constructor.call(this); - } - - Switch.prototype.children = ['subject', 'cases', 'otherwise']; - - Switch.prototype.isStatement = YES; - - Switch.prototype.jumps = function(o) { - var block, conds, _i, _len, _ref4, _ref5, _ref6; - if (o == null) { - o = { - block: true - }; - } - _ref4 = this.cases; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - _ref5 = _ref4[_i], conds = _ref5[0], block = _ref5[1]; - if (block.jumps(o)) { - return block; - } - } - return (_ref6 = this.otherwise) != null ? _ref6.jumps(o) : void 0; - }; - - Switch.prototype.makeReturn = function(res) { - var pair, _i, _len, _ref4, _ref5; - _ref4 = this.cases; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - pair = _ref4[_i]; - pair[1].makeReturn(res); - } - if (res) { - this.otherwise || (this.otherwise = new Block([new Literal('void 0')])); - } - if ((_ref5 = this.otherwise) != null) { - _ref5.makeReturn(res); - } - return this; - }; - - Switch.prototype.compileNode = function(o) { - var block, body, cond, conditions, expr, fragments, i, idt1, idt2, _i, _j, _len, _len1, _ref4, _ref5, _ref6; - if (this.subject) { - this.subject.icedStatementAssertion(); - } - idt1 = o.indent + TAB; - idt2 = o.indent = idt1 + TAB; - fragments = [].concat(this.makeCode(this.tab + "switch ("), (this.subject ? this.subject.compileToFragments(o, LEVEL_PAREN) : this.makeCode("false")), this.makeCode(") {\n")); - _ref4 = this.cases; - for (i = _i = 0, _len = _ref4.length; _i < _len; i = ++_i) { - _ref5 = _ref4[i], conditions = _ref5[0], block = _ref5[1]; - _ref6 = flatten([conditions]); - for (_j = 0, _len1 = _ref6.length; _j < _len1; _j++) { - cond = _ref6[_j]; - if (!this.subject) { - cond = cond.invert(); - } - fragments = fragments.concat(this.makeCode(idt1 + "case "), cond.compileToFragments(o, LEVEL_PAREN), this.makeCode(":\n")); - } - if ((body = block.compileToFragments(o, LEVEL_TOP)).length > 0) { - fragments = fragments.concat(body, this.makeCode('\n')); - } - if (i === this.cases.length - 1 && !this.otherwise) { - break; - } - expr = this.lastNonComment(block.expressions); - if (expr instanceof Return || (expr instanceof Literal && expr.jumps() && expr.value !== 'debugger')) { - continue; - } - fragments.push(cond.makeCode(idt2 + 'break;\n')); - } - if (this.otherwise && this.otherwise.expressions.length) { - fragments.push.apply(fragments, [this.makeCode(idt1 + "default:\n")].concat(__slice.call(this.otherwise.compileToFragments(o, LEVEL_TOP)), [this.makeCode("\n")])); - } - fragments.push(this.makeCode(this.tab + '}')); - return fragments; - }; - - Switch.prototype.icedCallContinuation = function() { - var block, condition, _i, _len, _ref4, _ref5; - _ref4 = this.cases; - for (_i = 0, _len = _ref4.length; _i < _len; _i++) { - _ref5 = _ref4[_i], condition = _ref5[0], block = _ref5[1]; - block.icedThreadReturn(); - } - if (this.otherwise != null) { - return this.otherwise.icedThreadReturn(); - } else { - return this.otherwise = new Block([new IcedTailCall]); - } - }; - - return Switch; - - })(Base); - - exports.If = If = (function(_super) { - __extends(If, _super); - - function If(condition, body, options) { - this.body = body; - if (options == null) { - options = {}; - } - If.__super__.constructor.call(this); - this.condition = options.type === 'unless' ? condition.invert() : condition; - this.elseBody = null; - this.isChain = false; - this.soak = options.soak; - } - - If.prototype.children = ['condition', 'body', 'elseBody']; - - If.prototype.bodyNode = function() { - var _ref4; - return (_ref4 = this.body) != null ? _ref4.unwrap() : void 0; - }; - - If.prototype.elseBodyNode = function() { - var _ref4; - return (_ref4 = this.elseBody) != null ? _ref4.unwrap() : void 0; - }; - - If.prototype.addElse = function(elseBody) { - if (this.isChain) { - this.elseBodyNode().addElse(elseBody); - } else { - this.isChain = elseBody instanceof If; - this.elseBody = this.ensureBlock(elseBody); - this.elseBody.updateLocationDataIfMissing(elseBody.locationData); - } - return this; - }; - - If.prototype.isStatement = function(o) { - var _ref4; - return (o != null ? o.level : void 0) === LEVEL_TOP || this.bodyNode().isStatement(o) || ((_ref4 = this.elseBodyNode()) != null ? _ref4.isStatement(o) : void 0); - }; - - If.prototype.jumps = function(o) { - var _ref4; - return this.body.jumps(o) || ((_ref4 = this.elseBody) != null ? _ref4.jumps(o) : void 0); - }; - - If.prototype.compileNode = function(o) { - this.condition.icedStatementAssertion(); - if (this.isStatement(o || this.icedIsCpsPivot())) { - return this.compileStatement(o); - } else { - return this.compileExpression(o); - } - }; - - If.prototype.makeReturn = function(res) { - if (res) { - this.elseBody || (this.elseBody = new Block([new Literal('void 0')])); - } - this.body && (this.body = new Block([this.body.makeReturn(res)])); - this.elseBody && (this.elseBody = new Block([this.elseBody.makeReturn(res)])); - return this; - }; - - If.prototype.ensureBlock = function(node) { - if (node instanceof Block) { - return node; - } else { - return new Block([node]); - } - }; - - If.prototype.compileStatement = function(o) { - var answer, body, child, cond, exeq, ifPart, indent; - child = del(o, 'chainChild'); - exeq = del(o, 'isExistentialEquals'); - if (exeq) { - return new If(this.condition.invert(), this.elseBodyNode(), { - type: 'if' - }).compileToFragments(o); - } - indent = o.indent + TAB; - cond = this.condition.compileToFragments(o, LEVEL_PAREN); - body = this.ensureBlock(this.body).compileToFragments(merge(o, { - indent: indent - })); - ifPart = [].concat(this.makeCode("if ("), cond, this.makeCode(") {\n"), body, this.makeCode("\n" + this.tab + "}")); - if (!child) { - ifPart.unshift(this.makeCode(this.tab)); - } - if (!this.elseBody) { - return ifPart; - } - answer = ifPart.concat(this.makeCode(' else ')); - if (this.isChain) { - o.chainChild = true; - answer = answer.concat(this.elseBody.unwrap().compileToFragments(o, LEVEL_TOP)); - } else { - answer = answer.concat(this.makeCode("{\n"), this.elseBody.compileToFragments(merge(o, { - indent: indent - }), LEVEL_TOP), this.makeCode("\n" + this.tab + "}")); - } - return answer; - }; - - If.prototype.compileExpression = function(o) { - var alt, body, cond, fragments; - cond = this.condition.compileToFragments(o, LEVEL_COND); - body = this.bodyNode().compileToFragments(o, LEVEL_LIST); - alt = this.elseBodyNode() ? this.elseBodyNode().compileToFragments(o, LEVEL_LIST) : [this.makeCode('void 0')]; - fragments = cond.concat(this.makeCode(" ? "), body, this.makeCode(" : "), alt); - if (o.level >= LEVEL_COND) { - return this.wrapInBraces(fragments); - } else { - return fragments; - } - }; - - If.prototype.unfoldSoak = function() { - return this.soak && this; - }; - - If.prototype.icedCallContinuation = function() { - if (this.elseBody) { - this.elseBody.icedThreadReturn(); - this.isChain = false; - } else { - this.addElse(new IcedTailCall); - } - return this.body.icedThreadReturn(); - }; - - return If; - - })(Base); - - Closure = { - wrap: function(expressions, statement, noReturn) { - var args, argumentsNode, call, func, meth; - if (expressions.jumps()) { - return expressions; - } - func = new Code([], Block.wrap([expressions])); - args = []; - argumentsNode = expressions.contains(this.isLiteralArguments); - if (argumentsNode && expressions.classBody) { - argumentsNode.error("Class bodies shouldn't reference arguments"); - } - if (argumentsNode || expressions.contains(this.isLiteralThis)) { - meth = new Literal(argumentsNode ? 'apply' : 'call'); - args = [new Literal('this')]; - if (argumentsNode) { - args.push(new Literal('arguments')); - } - func = new Value(func, [new Access(meth)]); - } - func.noReturn = noReturn; - call = new Call(func, args); - if (statement) { - return Block.wrap([call]); - } else { - return call; - } - }, - isLiteralArguments: function(node) { - return node instanceof Literal && node.value === 'arguments' && !node.asKey; - }, - isLiteralThis: function(node) { - return (node instanceof Literal && node.value === 'this' && !node.asKey) || (node instanceof Code && node.bound) || (node instanceof Call && node.isSuper); - } - }; - - unfoldSoak = function(o, parent, name) { - var ifn; - if (!(ifn = parent[name].unfoldSoak(o))) { - return; - } - parent[name] = ifn.body; - ifn.body = new Value(parent); - return ifn; - }; - - CpsCascade = { - wrap: function(statement, rest, returnValue, o) { - var args, block, call, cont, e, func; - func = new Code([new Param(new Literal(iced["const"].k))], Block.wrap([statement]), 'icedgen'); - args = []; - if (returnValue) { - returnValue.bindName(o); - args.push(returnValue); - } - block = Block.wrap([rest]); - if ((e = block.icedGetSingle()) && e instanceof IcedTailCall && e.canInline()) { - cont = e.extractFunc(); - } else { - cont = new Code(args, block, 'icedgen'); - } - call = new Call(func, [cont]); - return new Block([call]); - } - }; - - IcedTailCall = (function(_super) { - __extends(IcedTailCall, _super); - - function IcedTailCall(func, val) { - this.func = func; - if (val == null) { - val = null; - } - IcedTailCall.__super__.constructor.call(this); - if (!this.func) { - this.func = iced["const"].k; - } - this.value = val; - } - - IcedTailCall.prototype.children = ['value']; - - IcedTailCall.prototype.assignValue = function(v) { - return this.value = v; - }; - - IcedTailCall.prototype.canInline = function() { - return !this.value || this.value instanceof IcedReturnValue; - }; - - IcedTailCall.prototype.literalFunc = function() { - return new Literal(this.func); - }; - - IcedTailCall.prototype.extractFunc = function() { - return new Value(this.literalFunc()); - }; - - IcedTailCall.prototype.compileNode = function(o) { - var args, f, out; - f = this.literalFunc(); - out = o.level === LEVEL_TOP ? this.value ? new Block([this.value, new Call(f)]) : new Call(f) : (args = this.value ? [this.value] : [], new Call(f, args)); - return out.compileNode(o); - }; - - return IcedTailCall; - - })(Base); - - IcedReturnValue = (function(_super) { - __extends(IcedReturnValue, _super); - - IcedReturnValue.counter = 0; - - function IcedReturnValue() { - IcedReturnValue.__super__.constructor.call(this, null, null, false); - } - - IcedReturnValue.prototype.bindName = function(o) { - var l; - l = "" + (o.scope.freeVariable(iced["const"].param, false)) + "_" + (IcedReturnValue.counter++); - return this.name = new Literal(l); - }; - - IcedReturnValue.prototype.compile = function(o) { - if (!this.name) { - this.bindName(o); - } - return IcedReturnValue.__super__.compile.call(this, o); - }; - - return IcedReturnValue; - - })(Param); - - InlineRuntime = { - generate: function(ns_window) { - var a1, a2, af, apply_call, assignments, body, call_meth, cn, cnt, cnt_member, constructor_assign, constructor_body, constructor_code, constructor_name, constructor_params, decr, defer_assign, defer_body, defer_code, defer_name, defer_params, dp, dp_value, fn, fn_assign, fn_code, fn_name, if_body, if_cond, if_expr, inc, inner_body, inner_code, inner_params, ip, k, k_member, klass, klass_assign, my_apply, my_if, my_null, ns, ns_obj, ns_val, obj, outer_block, p1, ret_member, tr_assign, tr_block, tr_code, tr_name, tr_params, _fulfill_assign, _fulfill_body, _fulfill_call, _fulfill_code, _fulfill_method, _fulfill_name; - k = new Literal("continuation"); - cnt = new Literal("count"); - cn = new Value(new Literal(iced["const"].Deferrals)); - ns = new Value(new Literal(iced["const"].ns)); - if (ns_window) { - ns_window.add(new Access(ns)); - ns = ns_window; - } - k_member = new Value(new Literal("this")); - k_member.add(new Access(k)); - p1 = new Param(k_member); - cnt_member = new Value(new Literal("this")); - cnt_member.add(new Access(cnt)); - ret_member = new Value(new Literal("this")); - ret_member.add(new Access(new Value(new Literal(iced["const"].retslot)))); - a1 = new Assign(cnt_member, new Value(new Literal(1))); - a2 = new Assign(ret_member, NULL()); - constructor_params = [p1]; - constructor_body = new Block([a1, a2]); - constructor_code = new Code(constructor_params, constructor_body); - constructor_name = new Value(new Literal("constructor")); - constructor_assign = new Assign(constructor_name, constructor_code); - if_expr = new Call(k_member, [ret_member]); - if_body = new Block([if_expr]); - decr = new Op('--', cnt_member); - if_cond = new Op('!', decr); - my_if = new If(if_cond, if_body); - _fulfill_body = new Block([my_if]); - _fulfill_code = new Code([], _fulfill_body); - _fulfill_name = new Value(new Literal(iced["const"].fulfill)); - _fulfill_assign = new Assign(_fulfill_name, _fulfill_code); - inc = new Op("++", cnt_member); - ip = new Literal("inner_params"); - dp = new Literal("defer_params"); - dp_value = new Value(dp); - call_meth = new Value(dp); - af = new Literal(iced["const"].assign_fn); - call_meth.add(new Access(af, "soak")); - my_apply = new Literal("apply"); - call_meth.add(new Access(my_apply, "soak")); - my_null = NULL(); - apply_call = new Call(call_meth, [my_null, new Value(ip)]); - _fulfill_method = new Value(new Literal("this")); - _fulfill_method.add(new Access(new Literal(iced["const"].fulfill))); - _fulfill_call = new Call(_fulfill_method, []); - inner_body = new Block([apply_call, _fulfill_call]); - inner_params = [new Param(ip, null, true)]; - inner_code = new Code(inner_params, inner_body, "boundfunc"); - defer_body = new Block([inc, inner_code]); - defer_params = [new Param(dp)]; - defer_code = new Code(defer_params, defer_body); - defer_name = new Value(new Literal(iced["const"].defer_method)); - defer_assign = new Assign(defer_name, defer_code); - assignments = [constructor_assign, _fulfill_assign, defer_assign]; - obj = new Obj(assignments, true); - body = new Block([new Value(obj)]); - klass = new Class(null, null, body); - klass_assign = new Assign(cn, klass, "object"); - outer_block = new Block([NULL()]); - fn_code = new Code([], outer_block); - fn_name = new Value(new Literal(iced["const"].findDeferral)); - fn_assign = new Assign(fn_name, fn_code, "object"); - fn = new Literal("_fn"); - tr_block = new Block([new Call(new Value(fn), [])]); - tr_params = [new Param(fn)]; - tr_code = new Code(tr_params, tr_block); - tr_name = new Value(new Literal(iced["const"].trampoline)); - tr_assign = new Assign(tr_name, tr_code, "object"); - ns_obj = new Obj([klass_assign, fn_assign, tr_assign], true); - ns_val = new Value(ns_obj); - return new Assign(ns, ns_val); - } - }; - - UTILITIES = { - "extends": function() { - return "function(child, parent) { for (var key in parent) { if (" + (utility('hasProp')) + ".call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; }"; - }, - bind: function() { - return 'function(fn, me){ return function(){ return fn.apply(me, arguments); }; }'; - }, - indexOf: function() { - return "[].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; }"; - }, - hasProp: function() { - return '{}.hasOwnProperty'; - }, - slice: function() { - return '[].slice'; - } - }; - - LEVEL_TOP = 1; - - LEVEL_PAREN = 2; - - LEVEL_LIST = 3; - - LEVEL_COND = 4; - - LEVEL_OP = 5; - - LEVEL_ACCESS = 6; - - TAB = ' '; - - IDENTIFIER_STR = "[$A-Za-z_\\x7f-\\uffff][$\\w\\x7f-\\uffff]*"; - - IDENTIFIER = RegExp("^" + IDENTIFIER_STR + "$"); - - SIMPLENUM = /^[+-]?\d+$/; - - METHOD_DEF = RegExp("^(?:(" + IDENTIFIER_STR + ")\\.prototype(?:\\.(" + IDENTIFIER_STR + ")|\\[(\"(?:[^\\\\\"\\r\\n]|\\\\.)*\"|'(?:[^\\\\'\\r\\n]|\\\\.)*')\\]|\\[(0x[\\da-fA-F]+|\\d*\\.?\\d+(?:[eE][+-]?\\d+)?)\\]))|(" + IDENTIFIER_STR + ")$"); - - IS_STRING = /^['"]/; - - utility = function(name) { - var ref; - ref = "__" + name; - Scope.root.assign(ref, UTILITIES[name]()); - return ref; - }; - - multident = function(code, tab) { - code = code.replace(/\n/g, '$&' + tab); - return code.replace(/\s+$/, ''); - }; - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/optparse.js b/node_modules/iced-coffee-script/lib/coffee-script/optparse.js deleted file mode 100644 index f7f8b79..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/optparse.js +++ /dev/null @@ -1,141 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var LONG_FLAG, MULTI_FLAG, OPTIONAL, OptionParser, SHORT_FLAG, buildRule, buildRules, normalizeArguments, repeat; - - - - repeat = require('./helpers').repeat; - - exports.OptionParser = OptionParser = (function() { - function OptionParser(rules, banner) { - this.banner = banner; - this.rules = buildRules(rules); - } - - OptionParser.prototype.parse = function(args) { - var arg, i, isOption, matchedRule, options, originalArgs, pos, rule, seenNonOptionArg, skippingArgument, value, _i, _j, _len, _len1, _ref; - options = { - "arguments": [] - }; - skippingArgument = false; - originalArgs = args; - args = normalizeArguments(args); - for (i = _i = 0, _len = args.length; _i < _len; i = ++_i) { - arg = args[i]; - if (skippingArgument) { - skippingArgument = false; - continue; - } - if (arg === '--') { - pos = originalArgs.indexOf('--'); - options["arguments"] = options["arguments"].concat(originalArgs.slice(pos + 1)); - break; - } - isOption = !!(arg.match(LONG_FLAG) || arg.match(SHORT_FLAG)); - seenNonOptionArg = options["arguments"].length > 0; - if (!seenNonOptionArg) { - matchedRule = false; - _ref = this.rules; - for (_j = 0, _len1 = _ref.length; _j < _len1; _j++) { - rule = _ref[_j]; - if (rule.shortFlag === arg || rule.longFlag === arg) { - value = true; - if (rule.hasArgument) { - skippingArgument = true; - value = args[i + 1]; - } - options[rule.name] = rule.isList ? (options[rule.name] || []).concat(value) : value; - matchedRule = true; - break; - } - } - if (isOption && !matchedRule) { - throw new Error("unrecognized option: " + arg); - } - } - if (seenNonOptionArg || !isOption) { - options["arguments"].push(arg); - } - } - return options; - }; - - OptionParser.prototype.help = function() { - var letPart, lines, rule, spaces, _i, _len, _ref; - lines = []; - if (this.banner) { - lines.unshift("" + this.banner + "\n"); - } - _ref = this.rules; - for (_i = 0, _len = _ref.length; _i < _len; _i++) { - rule = _ref[_i]; - spaces = 15 - rule.longFlag.length; - spaces = spaces > 0 ? repeat(' ', spaces) : ''; - letPart = rule.shortFlag ? rule.shortFlag + ', ' : ' '; - lines.push(' ' + letPart + rule.longFlag + spaces + rule.description); - } - return "\n" + (lines.join('\n')) + "\n"; - }; - - return OptionParser; - - })(); - - LONG_FLAG = /^(--\w[\w\-]*)/; - - SHORT_FLAG = /^(-\w)$/; - - MULTI_FLAG = /^-(\w{2,})/; - - OPTIONAL = /\[(\w+(\*?))\]/; - - buildRules = function(rules) { - var tuple, _i, _len, _results; - _results = []; - for (_i = 0, _len = rules.length; _i < _len; _i++) { - tuple = rules[_i]; - if (tuple.length < 3) { - tuple.unshift(null); - } - _results.push(buildRule.apply(null, tuple)); - } - return _results; - }; - - buildRule = function(shortFlag, longFlag, description, options) { - var match; - if (options == null) { - options = {}; - } - match = longFlag.match(OPTIONAL); - longFlag = longFlag.match(LONG_FLAG)[1]; - return { - name: longFlag.substr(2), - shortFlag: shortFlag, - longFlag: longFlag, - description: description, - hasArgument: !!(match && match[1]), - isList: !!(match && match[2]) - }; - }; - - normalizeArguments = function(args) { - var arg, l, match, result, _i, _j, _len, _len1, _ref; - args = args.slice(0); - result = []; - for (_i = 0, _len = args.length; _i < _len; _i++) { - arg = args[_i]; - if (match = arg.match(MULTI_FLAG)) { - _ref = match[1].split(''); - for (_j = 0, _len1 = _ref.length; _j < _len1; _j++) { - l = _ref[_j]; - result.push('-' + l); - } - } else { - result.push(arg); - } - } - return result; - }; - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/parser.js b/node_modules/iced-coffee-script/lib/coffee-script/parser.js deleted file mode 100755 index 87e5557..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/parser.js +++ /dev/null @@ -1,620 +0,0 @@ -/* parser generated by jison 0.4.2 */ -var parser = (function(){ -var parser = {trace: function trace() { }, -yy: {}, -symbols_: {"error":2,"Root":3,"Body":4,"Line":5,"TERMINATOR":6,"Expression":7,"Statement":8,"Return":9,"Comment":10,"STATEMENT":11,"Await":12,"AWAIT":13,"Block":14,"Value":15,"Invocation":16,"Code":17,"Operation":18,"Assign":19,"If":20,"Try":21,"While":22,"For":23,"Switch":24,"Class":25,"Throw":26,"Defer":27,"INDENT":28,"OUTDENT":29,"Identifier":30,"IDENTIFIER":31,"AlphaNumeric":32,"NUMBER":33,"STRING":34,"Literal":35,"JS":36,"REGEX":37,"DEBUGGER":38,"UNDEFINED":39,"NULL":40,"BOOL":41,"Assignable":42,"=":43,"AssignObj":44,"ObjAssignable":45,":":46,"ThisProperty":47,"RETURN":48,"HERECOMMENT":49,"PARAM_START":50,"ParamList":51,"PARAM_END":52,"FuncGlyph":53,"->":54,"=>":55,"OptComma":56,",":57,"Param":58,"ParamVar":59,"...":60,"Array":61,"Object":62,"Splat":63,"SimpleAssignable":64,"Accessor":65,"Parenthetical":66,"Range":67,"This":68,".":69,"?.":70,"::":71,"?::":72,"Index":73,"INDEX_START":74,"IndexValue":75,"INDEX_END":76,"INDEX_SOAK":77,"Slice":78,"{":79,"AssignList":80,"}":81,"CLASS":82,"EXTENDS":83,"OptFuncExist":84,"Arguments":85,"SUPER":86,"DEFER":87,"FUNC_EXIST":88,"CALL_START":89,"CALL_END":90,"ArgList":91,"THIS":92,"@":93,"[":94,"]":95,"RangeDots":96,"..":97,"Arg":98,"SimpleArgs":99,"TRY":100,"Catch":101,"FINALLY":102,"CATCH":103,"THROW":104,"(":105,")":106,"WhileSource":107,"WHILE":108,"WHEN":109,"UNTIL":110,"Loop":111,"LOOP":112,"ForBody":113,"FOR":114,"ForStart":115,"ForSource":116,"ForVariables":117,"OWN":118,"ForValue":119,"FORIN":120,"FOROF":121,"BY":122,"SWITCH":123,"Whens":124,"ELSE":125,"When":126,"LEADING_WHEN":127,"IfBlock":128,"IF":129,"POST_IF":130,"UNARY":131,"-":132,"+":133,"--":134,"++":135,"?":136,"MATH":137,"SHIFT":138,"COMPARE":139,"LOGIC":140,"RELATION":141,"COMPOUND_ASSIGN":142,"$accept":0,"$end":1}, -terminals_: {2:"error",6:"TERMINATOR",11:"STATEMENT",13:"AWAIT",28:"INDENT",29:"OUTDENT",31:"IDENTIFIER",33:"NUMBER",34:"STRING",36:"JS",37:"REGEX",38:"DEBUGGER",39:"UNDEFINED",40:"NULL",41:"BOOL",43:"=",46:":",48:"RETURN",49:"HERECOMMENT",50:"PARAM_START",52:"PARAM_END",54:"->",55:"=>",57:",",60:"...",69:".",70:"?.",71:"::",72:"?::",74:"INDEX_START",76:"INDEX_END",77:"INDEX_SOAK",79:"{",81:"}",82:"CLASS",83:"EXTENDS",86:"SUPER",87:"DEFER",88:"FUNC_EXIST",89:"CALL_START",90:"CALL_END",92:"THIS",93:"@",94:"[",95:"]",97:"..",100:"TRY",102:"FINALLY",103:"CATCH",104:"THROW",105:"(",106:")",108:"WHILE",109:"WHEN",110:"UNTIL",112:"LOOP",114:"FOR",118:"OWN",120:"FORIN",121:"FOROF",122:"BY",123:"SWITCH",125:"ELSE",127:"LEADING_WHEN",129:"IF",130:"POST_IF",131:"UNARY",132:"-",133:"+",134:"--",135:"++",136:"?",137:"MATH",138:"SHIFT",139:"COMPARE",140:"LOGIC",141:"RELATION",142:"COMPOUND_ASSIGN"}, -productions_: [0,[3,0],[3,1],[4,1],[4,3],[4,2],[5,1],[5,1],[8,1],[8,1],[8,1],[8,1],[12,2],[12,2],[7,1],[7,1],[7,1],[7,1],[7,1],[7,1],[7,1],[7,1],[7,1],[7,1],[7,1],[7,1],[7,1],[14,2],[14,3],[30,1],[32,1],[32,1],[35,1],[35,1],[35,1],[35,1],[35,1],[35,1],[35,1],[19,3],[19,4],[19,5],[44,1],[44,3],[44,5],[44,1],[45,1],[45,1],[45,1],[9,2],[9,1],[10,1],[17,5],[17,2],[53,1],[53,1],[56,0],[56,1],[51,0],[51,1],[51,3],[51,4],[51,6],[58,1],[58,2],[58,3],[59,1],[59,1],[59,1],[59,1],[63,2],[64,1],[64,2],[64,2],[64,1],[42,1],[42,1],[42,1],[15,1],[15,1],[15,1],[15,1],[15,1],[65,2],[65,2],[65,2],[65,2],[65,2],[65,1],[65,1],[73,3],[73,2],[75,1],[75,1],[62,4],[80,0],[80,1],[80,3],[80,4],[80,6],[25,1],[25,2],[25,3],[25,4],[25,2],[25,3],[25,4],[25,5],[16,3],[16,3],[16,1],[16,2],[27,2],[84,0],[84,1],[85,2],[85,4],[68,1],[68,1],[47,2],[61,2],[61,4],[96,1],[96,1],[67,5],[78,3],[78,2],[78,2],[78,1],[91,1],[91,3],[91,4],[91,4],[91,6],[98,1],[98,1],[99,1],[99,3],[21,2],[21,3],[21,4],[21,5],[101,3],[101,3],[101,2],[26,2],[66,3],[66,5],[107,2],[107,4],[107,2],[107,4],[22,2],[22,2],[22,2],[22,1],[111,2],[111,2],[23,2],[23,2],[23,2],[113,2],[113,2],[115,2],[115,3],[119,1],[119,1],[119,1],[119,1],[117,1],[117,3],[116,2],[116,2],[116,4],[116,4],[116,4],[116,6],[116,6],[24,5],[24,7],[24,4],[24,6],[124,1],[124,2],[126,3],[126,4],[128,3],[128,5],[20,1],[20,3],[20,3],[20,3],[18,2],[18,2],[18,2],[18,2],[18,2],[18,2],[18,2],[18,2],[18,3],[18,3],[18,3],[18,3],[18,3],[18,3],[18,3],[18,3],[18,5],[18,4],[18,3]], -performAction: function anonymous(yytext,yyleng,yylineno,yy,yystate,$$,_$) { - -var $0 = $$.length - 1; -switch (yystate) { -case 1:return this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Block); -break; -case 2:return this.$ = $$[$0]; -break; -case 3:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(yy.Block.wrap([$$[$0]])); -break; -case 4:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])($$[$0-2].push($$[$0])); -break; -case 5:this.$ = $$[$0-1]; -break; -case 6:this.$ = $$[$0]; -break; -case 7:this.$ = $$[$0]; -break; -case 8:this.$ = $$[$0]; -break; -case 9:this.$ = $$[$0]; -break; -case 10:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Literal($$[$0])); -break; -case 11:this.$ = $$[$0]; -break; -case 12:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Await($$[$0])); -break; -case 13:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Await(yy.Block.wrap([$$[$0]]))); -break; -case 14:this.$ = $$[$0]; -break; -case 15:this.$ = $$[$0]; -break; -case 16:this.$ = $$[$0]; -break; -case 17:this.$ = $$[$0]; -break; -case 18:this.$ = $$[$0]; -break; -case 19:this.$ = $$[$0]; -break; -case 20:this.$ = $$[$0]; -break; -case 21:this.$ = $$[$0]; -break; -case 22:this.$ = $$[$0]; -break; -case 23:this.$ = $$[$0]; -break; -case 24:this.$ = $$[$0]; -break; -case 25:this.$ = $$[$0]; -break; -case 26:this.$ = $$[$0]; -break; -case 27:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Block); -break; -case 28:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])($$[$0-1]); -break; -case 29:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Literal($$[$0])); -break; -case 30:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Literal($$[$0])); -break; -case 31:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Literal($$[$0])); -break; -case 32:this.$ = $$[$0]; -break; -case 33:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Literal($$[$0])); -break; -case 34:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Literal($$[$0])); -break; -case 35:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Literal($$[$0])); -break; -case 36:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Undefined); -break; -case 37:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Null); -break; -case 38:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Bool($$[$0])); -break; -case 39:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Assign($$[$0-2], $$[$0])); -break; -case 40:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.Assign($$[$0-3], $$[$0])); -break; -case 41:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Assign($$[$0-4], $$[$0-1])); -break; -case 42:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 43:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Assign(yy.addLocationDataFn(_$[$0-2])(new yy.Value($$[$0-2])), $$[$0], 'object')); -break; -case 44:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Assign(yy.addLocationDataFn(_$[$0-4])(new yy.Value($$[$0-4])), $$[$0-1], 'object')); -break; -case 45:this.$ = $$[$0]; -break; -case 46:this.$ = $$[$0]; -break; -case 47:this.$ = $$[$0]; -break; -case 48:this.$ = $$[$0]; -break; -case 49:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Return($$[$0])); -break; -case 50:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Return); -break; -case 51:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Comment($$[$0])); -break; -case 52:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Code($$[$0-3], $$[$0], $$[$0-1])); -break; -case 53:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Code([], $$[$0], $$[$0-1])); -break; -case 54:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])('func'); -break; -case 55:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])('boundfunc'); -break; -case 56:this.$ = $$[$0]; -break; -case 57:this.$ = $$[$0]; -break; -case 58:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])([]); -break; -case 59:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])([$$[$0]]); -break; -case 60:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])($$[$0-2].concat($$[$0])); -break; -case 61:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])($$[$0-3].concat($$[$0])); -break; -case 62:this.$ = yy.addLocationDataFn(_$[$0-5], _$[$0])($$[$0-5].concat($$[$0-2])); -break; -case 63:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Param($$[$0])); -break; -case 64:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Param($$[$0-1], null, true)); -break; -case 65:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Param($$[$0-2], $$[$0])); -break; -case 66:this.$ = $$[$0]; -break; -case 67:this.$ = $$[$0]; -break; -case 68:this.$ = $$[$0]; -break; -case 69:this.$ = $$[$0]; -break; -case 70:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Splat($$[$0-1])); -break; -case 71:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 72:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])($$[$0-1].add($$[$0])); -break; -case 73:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Value($$[$0-1], [].concat($$[$0]))); -break; -case 74:this.$ = $$[$0]; -break; -case 75:this.$ = $$[$0]; -break; -case 76:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 77:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 78:this.$ = $$[$0]; -break; -case 79:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 80:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 81:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 82:this.$ = $$[$0]; -break; -case 83:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Access($$[$0])); -break; -case 84:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Access($$[$0].setCustom())); -break; -case 85:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Access($$[$0], 'soak')); -break; -case 86:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])([yy.addLocationDataFn(_$[$0-1])(new yy.Access(new yy.Literal('prototype'))), yy.addLocationDataFn(_$[$0])(new yy.Access($$[$0]))]); -break; -case 87:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])([yy.addLocationDataFn(_$[$0-1])(new yy.Access(new yy.Literal('prototype'), 'soak')), yy.addLocationDataFn(_$[$0])(new yy.Access($$[$0]))]); -break; -case 88:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Access(new yy.Literal('prototype'))); -break; -case 89:this.$ = $$[$0]; -break; -case 90:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])($$[$0-1]); -break; -case 91:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(yy.extend($$[$0], { - soak: true - })); -break; -case 92:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Index($$[$0])); -break; -case 93:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Slice($$[$0])); -break; -case 94:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.Obj($$[$0-2], $$[$0-3].generated)); -break; -case 95:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])([]); -break; -case 96:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])([$$[$0]]); -break; -case 97:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])($$[$0-2].concat($$[$0])); -break; -case 98:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])($$[$0-3].concat($$[$0])); -break; -case 99:this.$ = yy.addLocationDataFn(_$[$0-5], _$[$0])($$[$0-5].concat($$[$0-2])); -break; -case 100:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Class); -break; -case 101:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Class(null, null, $$[$0])); -break; -case 102:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Class(null, $$[$0])); -break; -case 103:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.Class(null, $$[$0-1], $$[$0])); -break; -case 104:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Class($$[$0])); -break; -case 105:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Class($$[$0-1], null, $$[$0])); -break; -case 106:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.Class($$[$0-2], $$[$0])); -break; -case 107:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Class($$[$0-3], $$[$0-1], $$[$0])); -break; -case 108:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Call($$[$0-2], $$[$0], $$[$0-1])); -break; -case 109:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Call($$[$0-2], $$[$0], $$[$0-1])); -break; -case 110:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Call('super', [new yy.Splat(new yy.Literal('arguments'))])); -break; -case 111:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Call('super', $$[$0])); -break; -case 112:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Defer($$[$0], yylineno)); -break; -case 113:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(false); -break; -case 114:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(true); -break; -case 115:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])([]); -break; -case 116:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])($$[$0-2]); -break; -case 117:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value(new yy.Literal('this'))); -break; -case 118:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value(new yy.Literal('this'))); -break; -case 119:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Value(yy.addLocationDataFn(_$[$0-1])(new yy.Literal('this')), [yy.addLocationDataFn(_$[$0])(new yy.Access($$[$0]))], 'this')); -break; -case 120:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Arr([])); -break; -case 121:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.Arr($$[$0-2])); -break; -case 122:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])('inclusive'); -break; -case 123:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])('exclusive'); -break; -case 124:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Range($$[$0-3], $$[$0-1], $$[$0-2])); -break; -case 125:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Range($$[$0-2], $$[$0], $$[$0-1])); -break; -case 126:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Range($$[$0-1], null, $$[$0])); -break; -case 127:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Range(null, $$[$0], $$[$0-1])); -break; -case 128:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Range(null, null, $$[$0])); -break; -case 129:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])([$$[$0]]); -break; -case 130:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])($$[$0-2].concat($$[$0])); -break; -case 131:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])($$[$0-3].concat($$[$0])); -break; -case 132:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])($$[$0-2]); -break; -case 133:this.$ = yy.addLocationDataFn(_$[$0-5], _$[$0])($$[$0-5].concat($$[$0-2])); -break; -case 134:this.$ = $$[$0]; -break; -case 135:this.$ = $$[$0]; -break; -case 136:this.$ = $$[$0]; -break; -case 137:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])([].concat($$[$0-2], $$[$0])); -break; -case 138:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Try($$[$0])); -break; -case 139:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Try($$[$0-1], $$[$0][0], $$[$0][1])); -break; -case 140:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.Try($$[$0-2], null, null, $$[$0])); -break; -case 141:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Try($$[$0-3], $$[$0-2][0], $$[$0-2][1], $$[$0])); -break; -case 142:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])([$$[$0-1], $$[$0]]); -break; -case 143:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])([yy.addLocationDataFn(_$[$0-1])(new yy.Value($$[$0-1])), $$[$0]]); -break; -case 144:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])([null, $$[$0]]); -break; -case 145:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Throw($$[$0])); -break; -case 146:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Parens($$[$0-1])); -break; -case 147:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Parens($$[$0-2])); -break; -case 148:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.While($$[$0])); -break; -case 149:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.While($$[$0-2], { - guard: $$[$0] - })); -break; -case 150:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.While($$[$0], { - invert: true - })); -break; -case 151:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.While($$[$0-2], { - invert: true, - guard: $$[$0] - })); -break; -case 152:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])($$[$0-1].addBody($$[$0])); -break; -case 153:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])($$[$0].addBody(yy.addLocationDataFn(_$[$0-1])(yy.Block.wrap([$$[$0-1]])))); -break; -case 154:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])($$[$0].addBody(yy.addLocationDataFn(_$[$0-1])(yy.Block.wrap([$$[$0-1]])))); -break; -case 155:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])($$[$0]); -break; -case 156:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.While(yy.addLocationDataFn(_$[$0-1])(new yy.Literal('true'))).addBody($$[$0])); -break; -case 157:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.While(yy.addLocationDataFn(_$[$0-1])(new yy.Literal('true'))).addBody(yy.addLocationDataFn(_$[$0])(yy.Block.wrap([$$[$0]])))); -break; -case 158:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.For($$[$0-1], $$[$0])); -break; -case 159:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.For($$[$0-1], $$[$0])); -break; -case 160:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.For($$[$0], $$[$0-1])); -break; -case 161:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])({ - source: yy.addLocationDataFn(_$[$0])(new yy.Value($$[$0])) - }); -break; -case 162:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])((function () { - $$[$0].own = $$[$0-1].own; - $$[$0].name = $$[$0-1][0]; - $$[$0].index = $$[$0-1][1]; - return $$[$0]; - }())); -break; -case 163:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])($$[$0]); -break; -case 164:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])((function () { - $$[$0].own = true; - return $$[$0]; - }())); -break; -case 165:this.$ = $$[$0]; -break; -case 166:this.$ = $$[$0]; -break; -case 167:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 168:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])(new yy.Value($$[$0])); -break; -case 169:this.$ = yy.addLocationDataFn(_$[$0], _$[$0])([$$[$0]]); -break; -case 170:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])([$$[$0-2], $$[$0]]); -break; -case 171:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])({ - source: $$[$0] - }); -break; -case 172:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])({ - source: $$[$0], - object: true - }); -break; -case 173:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])({ - source: $$[$0-2], - guard: $$[$0] - }); -break; -case 174:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])({ - source: $$[$0-2], - guard: $$[$0], - object: true - }); -break; -case 175:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])({ - source: $$[$0-2], - step: $$[$0] - }); -break; -case 176:this.$ = yy.addLocationDataFn(_$[$0-5], _$[$0])({ - source: $$[$0-4], - guard: $$[$0-2], - step: $$[$0] - }); -break; -case 177:this.$ = yy.addLocationDataFn(_$[$0-5], _$[$0])({ - source: $$[$0-4], - step: $$[$0-2], - guard: $$[$0] - }); -break; -case 178:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Switch($$[$0-3], $$[$0-1])); -break; -case 179:this.$ = yy.addLocationDataFn(_$[$0-6], _$[$0])(new yy.Switch($$[$0-5], $$[$0-3], $$[$0-1])); -break; -case 180:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.Switch(null, $$[$0-1])); -break; -case 181:this.$ = yy.addLocationDataFn(_$[$0-5], _$[$0])(new yy.Switch(null, $$[$0-3], $$[$0-1])); -break; -case 182:this.$ = $$[$0]; -break; -case 183:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])($$[$0-1].concat($$[$0])); -break; -case 184:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])([[$$[$0-1], $$[$0]]]); -break; -case 185:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])([[$$[$0-2], $$[$0-1]]]); -break; -case 186:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.If($$[$0-1], $$[$0], { - type: $$[$0-2] - })); -break; -case 187:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])($$[$0-4].addElse(yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.If($$[$0-1], $$[$0], { - type: $$[$0-2] - })))); -break; -case 188:this.$ = $$[$0]; -break; -case 189:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])($$[$0-2].addElse($$[$0])); -break; -case 190:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.If($$[$0], yy.addLocationDataFn(_$[$0-2])(yy.Block.wrap([$$[$0-2]])), { - type: $$[$0-1], - statement: true - })); -break; -case 191:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.If($$[$0], yy.addLocationDataFn(_$[$0-2])(yy.Block.wrap([$$[$0-2]])), { - type: $$[$0-1], - statement: true - })); -break; -case 192:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Op($$[$0-1], $$[$0])); -break; -case 193:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Op('-', $$[$0])); -break; -case 194:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Op('+', $$[$0])); -break; -case 195:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Op('--', $$[$0])); -break; -case 196:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Op('++', $$[$0])); -break; -case 197:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Op('--', $$[$0-1], null, true)); -break; -case 198:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Op('++', $$[$0-1], null, true)); -break; -case 199:this.$ = yy.addLocationDataFn(_$[$0-1], _$[$0])(new yy.Existence($$[$0-1])); -break; -case 200:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Op('+', $$[$0-2], $$[$0])); -break; -case 201:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Op('-', $$[$0-2], $$[$0])); -break; -case 202:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Op($$[$0-1], $$[$0-2], $$[$0])); -break; -case 203:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Op($$[$0-1], $$[$0-2], $$[$0])); -break; -case 204:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Op($$[$0-1], $$[$0-2], $$[$0])); -break; -case 205:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Op($$[$0-1], $$[$0-2], $$[$0])); -break; -case 206:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])((function () { - if ($$[$0-1].charAt(0) === '!') { - return new yy.Op($$[$0-1].slice(1), $$[$0-2], $$[$0]).invert(); - } else { - return new yy.Op($$[$0-1], $$[$0-2], $$[$0]); - } - }())); -break; -case 207:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Assign($$[$0-2], $$[$0], $$[$0-1])); -break; -case 208:this.$ = yy.addLocationDataFn(_$[$0-4], _$[$0])(new yy.Assign($$[$0-4], $$[$0-1], $$[$0-3])); -break; -case 209:this.$ = yy.addLocationDataFn(_$[$0-3], _$[$0])(new yy.Assign($$[$0-3], $$[$0], $$[$0-2])); -break; -case 210:this.$ = yy.addLocationDataFn(_$[$0-2], _$[$0])(new yy.Extends($$[$0-2], $$[$0])); -break; -} -}, -table: [{1:[2,1],3:1,4:2,5:3,7:4,8:5,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[3]},{1:[2,2],6:[1,76]},{1:[2,3],6:[2,3],29:[2,3],106:[2,3]},{1:[2,6],6:[2,6],29:[2,6],106:[2,6],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,7],6:[2,7],29:[2,7],106:[2,7],107:89,108:[1,67],110:[1,68],113:90,114:[1,70],115:71,130:[1,88]},{1:[2,14],6:[2,14],28:[2,14],29:[2,14],52:[2,14],57:[2,14],60:[2,14],65:92,69:[1,94],70:[1,95],71:[1,96],72:[1,97],73:98,74:[1,99],76:[2,14],77:[1,100],81:[2,14],84:91,88:[1,93],89:[2,113],90:[2,14],95:[2,14],97:[2,14],106:[2,14],108:[2,14],109:[2,14],110:[2,14],114:[2,14],122:[2,14],130:[2,14],132:[2,14],133:[2,14],136:[2,14],137:[2,14],138:[2,14],139:[2,14],140:[2,14],141:[2,14]},{1:[2,15],6:[2,15],28:[2,15],29:[2,15],52:[2,15],57:[2,15],60:[2,15],65:102,69:[1,94],70:[1,95],71:[1,96],72:[1,97],73:98,74:[1,99],76:[2,15],77:[1,100],81:[2,15],84:101,88:[1,93],89:[2,113],90:[2,15],95:[2,15],97:[2,15],106:[2,15],108:[2,15],109:[2,15],110:[2,15],114:[2,15],122:[2,15],130:[2,15],132:[2,15],133:[2,15],136:[2,15],137:[2,15],138:[2,15],139:[2,15],140:[2,15],141:[2,15]},{1:[2,16],6:[2,16],28:[2,16],29:[2,16],52:[2,16],57:[2,16],60:[2,16],76:[2,16],81:[2,16],90:[2,16],95:[2,16],97:[2,16],106:[2,16],108:[2,16],109:[2,16],110:[2,16],114:[2,16],122:[2,16],130:[2,16],132:[2,16],133:[2,16],136:[2,16],137:[2,16],138:[2,16],139:[2,16],140:[2,16],141:[2,16]},{1:[2,17],6:[2,17],28:[2,17],29:[2,17],52:[2,17],57:[2,17],60:[2,17],76:[2,17],81:[2,17],90:[2,17],95:[2,17],97:[2,17],106:[2,17],108:[2,17],109:[2,17],110:[2,17],114:[2,17],122:[2,17],130:[2,17],132:[2,17],133:[2,17],136:[2,17],137:[2,17],138:[2,17],139:[2,17],140:[2,17],141:[2,17]},{1:[2,18],6:[2,18],28:[2,18],29:[2,18],52:[2,18],57:[2,18],60:[2,18],76:[2,18],81:[2,18],90:[2,18],95:[2,18],97:[2,18],106:[2,18],108:[2,18],109:[2,18],110:[2,18],114:[2,18],122:[2,18],130:[2,18],132:[2,18],133:[2,18],136:[2,18],137:[2,18],138:[2,18],139:[2,18],140:[2,18],141:[2,18]},{1:[2,19],6:[2,19],28:[2,19],29:[2,19],52:[2,19],57:[2,19],60:[2,19],76:[2,19],81:[2,19],90:[2,19],95:[2,19],97:[2,19],106:[2,19],108:[2,19],109:[2,19],110:[2,19],114:[2,19],122:[2,19],130:[2,19],132:[2,19],133:[2,19],136:[2,19],137:[2,19],138:[2,19],139:[2,19],140:[2,19],141:[2,19]},{1:[2,20],6:[2,20],28:[2,20],29:[2,20],52:[2,20],57:[2,20],60:[2,20],76:[2,20],81:[2,20],90:[2,20],95:[2,20],97:[2,20],106:[2,20],108:[2,20],109:[2,20],110:[2,20],114:[2,20],122:[2,20],130:[2,20],132:[2,20],133:[2,20],136:[2,20],137:[2,20],138:[2,20],139:[2,20],140:[2,20],141:[2,20]},{1:[2,21],6:[2,21],28:[2,21],29:[2,21],52:[2,21],57:[2,21],60:[2,21],76:[2,21],81:[2,21],90:[2,21],95:[2,21],97:[2,21],106:[2,21],108:[2,21],109:[2,21],110:[2,21],114:[2,21],122:[2,21],130:[2,21],132:[2,21],133:[2,21],136:[2,21],137:[2,21],138:[2,21],139:[2,21],140:[2,21],141:[2,21]},{1:[2,22],6:[2,22],28:[2,22],29:[2,22],52:[2,22],57:[2,22],60:[2,22],76:[2,22],81:[2,22],90:[2,22],95:[2,22],97:[2,22],106:[2,22],108:[2,22],109:[2,22],110:[2,22],114:[2,22],122:[2,22],130:[2,22],132:[2,22],133:[2,22],136:[2,22],137:[2,22],138:[2,22],139:[2,22],140:[2,22],141:[2,22]},{1:[2,23],6:[2,23],28:[2,23],29:[2,23],52:[2,23],57:[2,23],60:[2,23],76:[2,23],81:[2,23],90:[2,23],95:[2,23],97:[2,23],106:[2,23],108:[2,23],109:[2,23],110:[2,23],114:[2,23],122:[2,23],130:[2,23],132:[2,23],133:[2,23],136:[2,23],137:[2,23],138:[2,23],139:[2,23],140:[2,23],141:[2,23]},{1:[2,24],6:[2,24],28:[2,24],29:[2,24],52:[2,24],57:[2,24],60:[2,24],76:[2,24],81:[2,24],90:[2,24],95:[2,24],97:[2,24],106:[2,24],108:[2,24],109:[2,24],110:[2,24],114:[2,24],122:[2,24],130:[2,24],132:[2,24],133:[2,24],136:[2,24],137:[2,24],138:[2,24],139:[2,24],140:[2,24],141:[2,24]},{1:[2,25],6:[2,25],28:[2,25],29:[2,25],52:[2,25],57:[2,25],60:[2,25],76:[2,25],81:[2,25],90:[2,25],95:[2,25],97:[2,25],106:[2,25],108:[2,25],109:[2,25],110:[2,25],114:[2,25],122:[2,25],130:[2,25],132:[2,25],133:[2,25],136:[2,25],137:[2,25],138:[2,25],139:[2,25],140:[2,25],141:[2,25]},{1:[2,26],6:[2,26],28:[2,26],29:[2,26],52:[2,26],57:[2,26],60:[2,26],76:[2,26],81:[2,26],90:[2,26],95:[2,26],97:[2,26],106:[2,26],108:[2,26],109:[2,26],110:[2,26],114:[2,26],122:[2,26],130:[2,26],132:[2,26],133:[2,26],136:[2,26],137:[2,26],138:[2,26],139:[2,26],140:[2,26],141:[2,26]},{1:[2,8],6:[2,8],29:[2,8],106:[2,8],108:[2,8],110:[2,8],114:[2,8],130:[2,8]},{1:[2,9],6:[2,9],29:[2,9],106:[2,9],108:[2,9],110:[2,9],114:[2,9],130:[2,9]},{1:[2,10],6:[2,10],29:[2,10],106:[2,10],108:[2,10],110:[2,10],114:[2,10],130:[2,10]},{1:[2,11],6:[2,11],29:[2,11],106:[2,11],108:[2,11],110:[2,11],114:[2,11],130:[2,11]},{1:[2,78],6:[2,78],28:[2,78],29:[2,78],43:[1,103],52:[2,78],57:[2,78],60:[2,78],69:[2,78],70:[2,78],71:[2,78],72:[2,78],74:[2,78],76:[2,78],77:[2,78],81:[2,78],88:[2,78],89:[2,78],90:[2,78],95:[2,78],97:[2,78],106:[2,78],108:[2,78],109:[2,78],110:[2,78],114:[2,78],122:[2,78],130:[2,78],132:[2,78],133:[2,78],136:[2,78],137:[2,78],138:[2,78],139:[2,78],140:[2,78],141:[2,78]},{1:[2,79],6:[2,79],28:[2,79],29:[2,79],52:[2,79],57:[2,79],60:[2,79],69:[2,79],70:[2,79],71:[2,79],72:[2,79],74:[2,79],76:[2,79],77:[2,79],81:[2,79],88:[2,79],89:[2,79],90:[2,79],95:[2,79],97:[2,79],106:[2,79],108:[2,79],109:[2,79],110:[2,79],114:[2,79],122:[2,79],130:[2,79],132:[2,79],133:[2,79],136:[2,79],137:[2,79],138:[2,79],139:[2,79],140:[2,79],141:[2,79]},{1:[2,80],6:[2,80],28:[2,80],29:[2,80],52:[2,80],57:[2,80],60:[2,80],69:[2,80],70:[2,80],71:[2,80],72:[2,80],74:[2,80],76:[2,80],77:[2,80],81:[2,80],88:[2,80],89:[2,80],90:[2,80],95:[2,80],97:[2,80],106:[2,80],108:[2,80],109:[2,80],110:[2,80],114:[2,80],122:[2,80],130:[2,80],132:[2,80],133:[2,80],136:[2,80],137:[2,80],138:[2,80],139:[2,80],140:[2,80],141:[2,80]},{1:[2,81],6:[2,81],28:[2,81],29:[2,81],52:[2,81],57:[2,81],60:[2,81],69:[2,81],70:[2,81],71:[2,81],72:[2,81],74:[2,81],76:[2,81],77:[2,81],81:[2,81],88:[2,81],89:[2,81],90:[2,81],95:[2,81],97:[2,81],106:[2,81],108:[2,81],109:[2,81],110:[2,81],114:[2,81],122:[2,81],130:[2,81],132:[2,81],133:[2,81],136:[2,81],137:[2,81],138:[2,81],139:[2,81],140:[2,81],141:[2,81]},{1:[2,82],6:[2,82],28:[2,82],29:[2,82],52:[2,82],57:[2,82],60:[2,82],69:[2,82],70:[2,82],71:[2,82],72:[2,82],74:[2,82],76:[2,82],77:[2,82],81:[2,82],88:[2,82],89:[2,82],90:[2,82],95:[2,82],97:[2,82],106:[2,82],108:[2,82],109:[2,82],110:[2,82],114:[2,82],122:[2,82],130:[2,82],132:[2,82],133:[2,82],136:[2,82],137:[2,82],138:[2,82],139:[2,82],140:[2,82],141:[2,82]},{1:[2,110],6:[2,110],28:[2,110],29:[2,110],52:[2,110],57:[2,110],60:[2,110],69:[2,110],70:[2,110],71:[2,110],72:[2,110],74:[2,110],76:[2,110],77:[2,110],81:[2,110],85:104,88:[2,110],89:[1,105],90:[2,110],95:[2,110],97:[2,110],106:[2,110],108:[2,110],109:[2,110],110:[2,110],114:[2,110],122:[2,110],130:[2,110],132:[2,110],133:[2,110],136:[2,110],137:[2,110],138:[2,110],139:[2,110],140:[2,110],141:[2,110]},{6:[2,58],28:[2,58],30:109,31:[1,75],47:110,51:106,52:[2,58],57:[2,58],58:107,59:108,61:111,62:112,79:[1,72],93:[1,113],94:[1,114]},{14:115,28:[1,116]},{7:117,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:119,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:120,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{15:122,16:123,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:124,47:65,61:49,62:50,64:121,66:25,67:26,68:27,79:[1,72],86:[1,28],92:[1,60],93:[1,61],94:[1,59],105:[1,58]},{15:122,16:123,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:124,47:65,61:49,62:50,64:125,66:25,67:26,68:27,79:[1,72],86:[1,28],92:[1,60],93:[1,61],94:[1,59],105:[1,58]},{1:[2,75],6:[2,75],28:[2,75],29:[2,75],43:[2,75],52:[2,75],57:[2,75],60:[2,75],69:[2,75],70:[2,75],71:[2,75],72:[2,75],74:[2,75],76:[2,75],77:[2,75],81:[2,75],83:[1,129],88:[2,75],89:[2,75],90:[2,75],95:[2,75],97:[2,75],106:[2,75],108:[2,75],109:[2,75],110:[2,75],114:[2,75],122:[2,75],130:[2,75],132:[2,75],133:[2,75],134:[1,126],135:[1,127],136:[2,75],137:[2,75],138:[2,75],139:[2,75],140:[2,75],141:[2,75],142:[1,128]},{1:[2,188],6:[2,188],28:[2,188],29:[2,188],52:[2,188],57:[2,188],60:[2,188],76:[2,188],81:[2,188],90:[2,188],95:[2,188],97:[2,188],106:[2,188],108:[2,188],109:[2,188],110:[2,188],114:[2,188],122:[2,188],125:[1,130],130:[2,188],132:[2,188],133:[2,188],136:[2,188],137:[2,188],138:[2,188],139:[2,188],140:[2,188],141:[2,188]},{14:131,28:[1,116]},{14:132,28:[1,116]},{1:[2,155],6:[2,155],28:[2,155],29:[2,155],52:[2,155],57:[2,155],60:[2,155],76:[2,155],81:[2,155],90:[2,155],95:[2,155],97:[2,155],106:[2,155],108:[2,155],109:[2,155],110:[2,155],114:[2,155],122:[2,155],130:[2,155],132:[2,155],133:[2,155],136:[2,155],137:[2,155],138:[2,155],139:[2,155],140:[2,155],141:[2,155]},{14:133,28:[1,116]},{7:134,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,135],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,100],6:[2,100],14:136,15:122,16:123,28:[1,116],29:[2,100],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:124,47:65,52:[2,100],57:[2,100],60:[2,100],61:49,62:50,64:138,66:25,67:26,68:27,76:[2,100],79:[1,72],81:[2,100],83:[1,137],86:[1,28],90:[2,100],92:[1,60],93:[1,61],94:[1,59],95:[2,100],97:[2,100],105:[1,58],106:[2,100],108:[2,100],109:[2,100],110:[2,100],114:[2,100],122:[2,100],130:[2,100],132:[2,100],133:[2,100],136:[2,100],137:[2,100],138:[2,100],139:[2,100],140:[2,100],141:[2,100]},{7:139,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{85:140,89:[1,105]},{1:[2,50],6:[2,50],7:141,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,29:[2,50],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],106:[2,50],107:39,108:[2,50],110:[2,50],111:40,112:[1,69],113:41,114:[2,50],115:71,123:[1,42],128:37,129:[1,66],130:[2,50],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,51],6:[2,51],28:[2,51],29:[2,51],57:[2,51],81:[2,51],106:[2,51],108:[2,51],110:[2,51],114:[2,51],130:[2,51]},{7:143,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],14:142,15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,116],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,76],6:[2,76],28:[2,76],29:[2,76],43:[2,76],52:[2,76],57:[2,76],60:[2,76],69:[2,76],70:[2,76],71:[2,76],72:[2,76],74:[2,76],76:[2,76],77:[2,76],81:[2,76],88:[2,76],89:[2,76],90:[2,76],95:[2,76],97:[2,76],106:[2,76],108:[2,76],109:[2,76],110:[2,76],114:[2,76],122:[2,76],130:[2,76],132:[2,76],133:[2,76],136:[2,76],137:[2,76],138:[2,76],139:[2,76],140:[2,76],141:[2,76]},{1:[2,77],6:[2,77],28:[2,77],29:[2,77],43:[2,77],52:[2,77],57:[2,77],60:[2,77],69:[2,77],70:[2,77],71:[2,77],72:[2,77],74:[2,77],76:[2,77],77:[2,77],81:[2,77],88:[2,77],89:[2,77],90:[2,77],95:[2,77],97:[2,77],106:[2,77],108:[2,77],109:[2,77],110:[2,77],114:[2,77],122:[2,77],130:[2,77],132:[2,77],133:[2,77],136:[2,77],137:[2,77],138:[2,77],139:[2,77],140:[2,77],141:[2,77]},{1:[2,32],6:[2,32],28:[2,32],29:[2,32],52:[2,32],57:[2,32],60:[2,32],69:[2,32],70:[2,32],71:[2,32],72:[2,32],74:[2,32],76:[2,32],77:[2,32],81:[2,32],88:[2,32],89:[2,32],90:[2,32],95:[2,32],97:[2,32],106:[2,32],108:[2,32],109:[2,32],110:[2,32],114:[2,32],122:[2,32],130:[2,32],132:[2,32],133:[2,32],136:[2,32],137:[2,32],138:[2,32],139:[2,32],140:[2,32],141:[2,32]},{1:[2,33],6:[2,33],28:[2,33],29:[2,33],52:[2,33],57:[2,33],60:[2,33],69:[2,33],70:[2,33],71:[2,33],72:[2,33],74:[2,33],76:[2,33],77:[2,33],81:[2,33],88:[2,33],89:[2,33],90:[2,33],95:[2,33],97:[2,33],106:[2,33],108:[2,33],109:[2,33],110:[2,33],114:[2,33],122:[2,33],130:[2,33],132:[2,33],133:[2,33],136:[2,33],137:[2,33],138:[2,33],139:[2,33],140:[2,33],141:[2,33]},{1:[2,34],6:[2,34],28:[2,34],29:[2,34],52:[2,34],57:[2,34],60:[2,34],69:[2,34],70:[2,34],71:[2,34],72:[2,34],74:[2,34],76:[2,34],77:[2,34],81:[2,34],88:[2,34],89:[2,34],90:[2,34],95:[2,34],97:[2,34],106:[2,34],108:[2,34],109:[2,34],110:[2,34],114:[2,34],122:[2,34],130:[2,34],132:[2,34],133:[2,34],136:[2,34],137:[2,34],138:[2,34],139:[2,34],140:[2,34],141:[2,34]},{1:[2,35],6:[2,35],28:[2,35],29:[2,35],52:[2,35],57:[2,35],60:[2,35],69:[2,35],70:[2,35],71:[2,35],72:[2,35],74:[2,35],76:[2,35],77:[2,35],81:[2,35],88:[2,35],89:[2,35],90:[2,35],95:[2,35],97:[2,35],106:[2,35],108:[2,35],109:[2,35],110:[2,35],114:[2,35],122:[2,35],130:[2,35],132:[2,35],133:[2,35],136:[2,35],137:[2,35],138:[2,35],139:[2,35],140:[2,35],141:[2,35]},{1:[2,36],6:[2,36],28:[2,36],29:[2,36],52:[2,36],57:[2,36],60:[2,36],69:[2,36],70:[2,36],71:[2,36],72:[2,36],74:[2,36],76:[2,36],77:[2,36],81:[2,36],88:[2,36],89:[2,36],90:[2,36],95:[2,36],97:[2,36],106:[2,36],108:[2,36],109:[2,36],110:[2,36],114:[2,36],122:[2,36],130:[2,36],132:[2,36],133:[2,36],136:[2,36],137:[2,36],138:[2,36],139:[2,36],140:[2,36],141:[2,36]},{1:[2,37],6:[2,37],28:[2,37],29:[2,37],52:[2,37],57:[2,37],60:[2,37],69:[2,37],70:[2,37],71:[2,37],72:[2,37],74:[2,37],76:[2,37],77:[2,37],81:[2,37],88:[2,37],89:[2,37],90:[2,37],95:[2,37],97:[2,37],106:[2,37],108:[2,37],109:[2,37],110:[2,37],114:[2,37],122:[2,37],130:[2,37],132:[2,37],133:[2,37],136:[2,37],137:[2,37],138:[2,37],139:[2,37],140:[2,37],141:[2,37]},{1:[2,38],6:[2,38],28:[2,38],29:[2,38],52:[2,38],57:[2,38],60:[2,38],69:[2,38],70:[2,38],71:[2,38],72:[2,38],74:[2,38],76:[2,38],77:[2,38],81:[2,38],88:[2,38],89:[2,38],90:[2,38],95:[2,38],97:[2,38],106:[2,38],108:[2,38],109:[2,38],110:[2,38],114:[2,38],122:[2,38],130:[2,38],132:[2,38],133:[2,38],136:[2,38],137:[2,38],138:[2,38],139:[2,38],140:[2,38],141:[2,38]},{4:144,5:3,7:4,8:5,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,145],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:146,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,150],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,63:151,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],91:148,92:[1,60],93:[1,61],94:[1,59],95:[1,147],98:149,100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,117],6:[2,117],28:[2,117],29:[2,117],52:[2,117],57:[2,117],60:[2,117],69:[2,117],70:[2,117],71:[2,117],72:[2,117],74:[2,117],76:[2,117],77:[2,117],81:[2,117],88:[2,117],89:[2,117],90:[2,117],95:[2,117],97:[2,117],106:[2,117],108:[2,117],109:[2,117],110:[2,117],114:[2,117],122:[2,117],130:[2,117],132:[2,117],133:[2,117],136:[2,117],137:[2,117],138:[2,117],139:[2,117],140:[2,117],141:[2,117]},{1:[2,118],6:[2,118],28:[2,118],29:[2,118],30:152,31:[1,75],52:[2,118],57:[2,118],60:[2,118],69:[2,118],70:[2,118],71:[2,118],72:[2,118],74:[2,118],76:[2,118],77:[2,118],81:[2,118],88:[2,118],89:[2,118],90:[2,118],95:[2,118],97:[2,118],106:[2,118],108:[2,118],109:[2,118],110:[2,118],114:[2,118],122:[2,118],130:[2,118],132:[2,118],133:[2,118],136:[2,118],137:[2,118],138:[2,118],139:[2,118],140:[2,118],141:[2,118]},{28:[2,54]},{28:[2,55]},{1:[2,71],6:[2,71],28:[2,71],29:[2,71],43:[2,71],52:[2,71],57:[2,71],60:[2,71],69:[2,71],70:[2,71],71:[2,71],72:[2,71],74:[2,71],76:[2,71],77:[2,71],81:[2,71],83:[2,71],88:[2,71],89:[2,71],90:[2,71],95:[2,71],97:[2,71],106:[2,71],108:[2,71],109:[2,71],110:[2,71],114:[2,71],122:[2,71],130:[2,71],132:[2,71],133:[2,71],134:[2,71],135:[2,71],136:[2,71],137:[2,71],138:[2,71],139:[2,71],140:[2,71],141:[2,71],142:[2,71]},{1:[2,74],6:[2,74],28:[2,74],29:[2,74],43:[2,74],52:[2,74],57:[2,74],60:[2,74],69:[2,74],70:[2,74],71:[2,74],72:[2,74],74:[2,74],76:[2,74],77:[2,74],81:[2,74],83:[2,74],88:[2,74],89:[2,74],90:[2,74],95:[2,74],97:[2,74],106:[2,74],108:[2,74],109:[2,74],110:[2,74],114:[2,74],122:[2,74],130:[2,74],132:[2,74],133:[2,74],134:[2,74],135:[2,74],136:[2,74],137:[2,74],138:[2,74],139:[2,74],140:[2,74],141:[2,74],142:[2,74]},{7:153,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:154,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:155,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:157,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],14:156,15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,116],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{30:162,31:[1,75],47:163,61:164,62:165,67:158,79:[1,72],93:[1,113],94:[1,59],117:159,118:[1,160],119:161},{116:166,120:[1,167],121:[1,168]},{6:[2,95],10:172,28:[2,95],30:173,31:[1,75],32:174,33:[1,73],34:[1,74],44:170,45:171,47:175,49:[1,47],57:[2,95],80:169,81:[2,95],93:[1,113]},{1:[2,30],6:[2,30],28:[2,30],29:[2,30],46:[2,30],52:[2,30],57:[2,30],60:[2,30],69:[2,30],70:[2,30],71:[2,30],72:[2,30],74:[2,30],76:[2,30],77:[2,30],81:[2,30],88:[2,30],89:[2,30],90:[2,30],95:[2,30],97:[2,30],106:[2,30],108:[2,30],109:[2,30],110:[2,30],114:[2,30],122:[2,30],130:[2,30],132:[2,30],133:[2,30],136:[2,30],137:[2,30],138:[2,30],139:[2,30],140:[2,30],141:[2,30]},{1:[2,31],6:[2,31],28:[2,31],29:[2,31],46:[2,31],52:[2,31],57:[2,31],60:[2,31],69:[2,31],70:[2,31],71:[2,31],72:[2,31],74:[2,31],76:[2,31],77:[2,31],81:[2,31],88:[2,31],89:[2,31],90:[2,31],95:[2,31],97:[2,31],106:[2,31],108:[2,31],109:[2,31],110:[2,31],114:[2,31],122:[2,31],130:[2,31],132:[2,31],133:[2,31],136:[2,31],137:[2,31],138:[2,31],139:[2,31],140:[2,31],141:[2,31]},{1:[2,29],6:[2,29],28:[2,29],29:[2,29],43:[2,29],46:[2,29],52:[2,29],57:[2,29],60:[2,29],69:[2,29],70:[2,29],71:[2,29],72:[2,29],74:[2,29],76:[2,29],77:[2,29],81:[2,29],83:[2,29],88:[2,29],89:[2,29],90:[2,29],95:[2,29],97:[2,29],106:[2,29],108:[2,29],109:[2,29],110:[2,29],114:[2,29],120:[2,29],121:[2,29],122:[2,29],130:[2,29],132:[2,29],133:[2,29],134:[2,29],135:[2,29],136:[2,29],137:[2,29],138:[2,29],139:[2,29],140:[2,29],141:[2,29],142:[2,29]},{1:[2,5],5:176,6:[2,5],7:4,8:5,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,29:[2,5],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],106:[2,5],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,199],6:[2,199],28:[2,199],29:[2,199],52:[2,199],57:[2,199],60:[2,199],76:[2,199],81:[2,199],90:[2,199],95:[2,199],97:[2,199],106:[2,199],108:[2,199],109:[2,199],110:[2,199],114:[2,199],122:[2,199],130:[2,199],132:[2,199],133:[2,199],136:[2,199],137:[2,199],138:[2,199],139:[2,199],140:[2,199],141:[2,199]},{7:177,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:178,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:179,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:180,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:181,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:182,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:183,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:184,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,154],6:[2,154],28:[2,154],29:[2,154],52:[2,154],57:[2,154],60:[2,154],76:[2,154],81:[2,154],90:[2,154],95:[2,154],97:[2,154],106:[2,154],108:[2,154],109:[2,154],110:[2,154],114:[2,154],122:[2,154],130:[2,154],132:[2,154],133:[2,154],136:[2,154],137:[2,154],138:[2,154],139:[2,154],140:[2,154],141:[2,154]},{1:[2,159],6:[2,159],28:[2,159],29:[2,159],52:[2,159],57:[2,159],60:[2,159],76:[2,159],81:[2,159],90:[2,159],95:[2,159],97:[2,159],106:[2,159],108:[2,159],109:[2,159],110:[2,159],114:[2,159],122:[2,159],130:[2,159],132:[2,159],133:[2,159],136:[2,159],137:[2,159],138:[2,159],139:[2,159],140:[2,159],141:[2,159]},{7:185,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,153],6:[2,153],28:[2,153],29:[2,153],52:[2,153],57:[2,153],60:[2,153],76:[2,153],81:[2,153],90:[2,153],95:[2,153],97:[2,153],106:[2,153],108:[2,153],109:[2,153],110:[2,153],114:[2,153],122:[2,153],130:[2,153],132:[2,153],133:[2,153],136:[2,153],137:[2,153],138:[2,153],139:[2,153],140:[2,153],141:[2,153]},{1:[2,158],6:[2,158],28:[2,158],29:[2,158],52:[2,158],57:[2,158],60:[2,158],76:[2,158],81:[2,158],90:[2,158],95:[2,158],97:[2,158],106:[2,158],108:[2,158],109:[2,158],110:[2,158],114:[2,158],122:[2,158],130:[2,158],132:[2,158],133:[2,158],136:[2,158],137:[2,158],138:[2,158],139:[2,158],140:[2,158],141:[2,158]},{85:186,89:[1,105]},{1:[2,72],6:[2,72],28:[2,72],29:[2,72],43:[2,72],52:[2,72],57:[2,72],60:[2,72],69:[2,72],70:[2,72],71:[2,72],72:[2,72],74:[2,72],76:[2,72],77:[2,72],81:[2,72],83:[2,72],88:[2,72],89:[2,72],90:[2,72],95:[2,72],97:[2,72],106:[2,72],108:[2,72],109:[2,72],110:[2,72],114:[2,72],122:[2,72],130:[2,72],132:[2,72],133:[2,72],134:[2,72],135:[2,72],136:[2,72],137:[2,72],138:[2,72],139:[2,72],140:[2,72],141:[2,72],142:[2,72]},{89:[2,114]},{27:188,30:187,31:[1,75],87:[1,45]},{30:189,31:[1,75]},{1:[2,88],6:[2,88],28:[2,88],29:[2,88],30:190,31:[1,75],43:[2,88],52:[2,88],57:[2,88],60:[2,88],69:[2,88],70:[2,88],71:[2,88],72:[2,88],74:[2,88],76:[2,88],77:[2,88],81:[2,88],83:[2,88],88:[2,88],89:[2,88],90:[2,88],95:[2,88],97:[2,88],106:[2,88],108:[2,88],109:[2,88],110:[2,88],114:[2,88],122:[2,88],130:[2,88],132:[2,88],133:[2,88],134:[2,88],135:[2,88],136:[2,88],137:[2,88],138:[2,88],139:[2,88],140:[2,88],141:[2,88],142:[2,88]},{30:191,31:[1,75]},{1:[2,89],6:[2,89],28:[2,89],29:[2,89],43:[2,89],52:[2,89],57:[2,89],60:[2,89],69:[2,89],70:[2,89],71:[2,89],72:[2,89],74:[2,89],76:[2,89],77:[2,89],81:[2,89],83:[2,89],88:[2,89],89:[2,89],90:[2,89],95:[2,89],97:[2,89],106:[2,89],108:[2,89],109:[2,89],110:[2,89],114:[2,89],122:[2,89],130:[2,89],132:[2,89],133:[2,89],134:[2,89],135:[2,89],136:[2,89],137:[2,89],138:[2,89],139:[2,89],140:[2,89],141:[2,89],142:[2,89]},{7:193,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],60:[1,197],61:49,62:50,64:36,66:25,67:26,68:27,75:192,78:194,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],96:195,97:[1,196],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{73:198,74:[1,99],77:[1,100]},{85:199,89:[1,105]},{1:[2,73],6:[2,73],28:[2,73],29:[2,73],43:[2,73],52:[2,73],57:[2,73],60:[2,73],69:[2,73],70:[2,73],71:[2,73],72:[2,73],74:[2,73],76:[2,73],77:[2,73],81:[2,73],83:[2,73],88:[2,73],89:[2,73],90:[2,73],95:[2,73],97:[2,73],106:[2,73],108:[2,73],109:[2,73],110:[2,73],114:[2,73],122:[2,73],130:[2,73],132:[2,73],133:[2,73],134:[2,73],135:[2,73],136:[2,73],137:[2,73],138:[2,73],139:[2,73],140:[2,73],141:[2,73],142:[2,73]},{6:[1,201],7:200,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,202],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,111],6:[2,111],28:[2,111],29:[2,111],52:[2,111],57:[2,111],60:[2,111],69:[2,111],70:[2,111],71:[2,111],72:[2,111],74:[2,111],76:[2,111],77:[2,111],81:[2,111],88:[2,111],89:[2,111],90:[2,111],95:[2,111],97:[2,111],106:[2,111],108:[2,111],109:[2,111],110:[2,111],114:[2,111],122:[2,111],130:[2,111],132:[2,111],133:[2,111],136:[2,111],137:[2,111],138:[2,111],139:[2,111],140:[2,111],141:[2,111]},{7:205,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,150],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,63:151,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],90:[1,203],91:204,92:[1,60],93:[1,61],94:[1,59],98:149,100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{6:[2,56],28:[2,56],52:[1,206],56:208,57:[1,207]},{6:[2,59],28:[2,59],29:[2,59],52:[2,59],57:[2,59]},{6:[2,63],28:[2,63],29:[2,63],43:[1,210],52:[2,63],57:[2,63],60:[1,209]},{6:[2,66],28:[2,66],29:[2,66],43:[2,66],52:[2,66],57:[2,66],60:[2,66]},{6:[2,67],28:[2,67],29:[2,67],43:[2,67],52:[2,67],57:[2,67],60:[2,67]},{6:[2,68],28:[2,68],29:[2,68],43:[2,68],52:[2,68],57:[2,68],60:[2,68]},{6:[2,69],28:[2,69],29:[2,69],43:[2,69],52:[2,69],57:[2,69],60:[2,69]},{30:152,31:[1,75]},{7:205,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,150],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,63:151,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],91:148,92:[1,60],93:[1,61],94:[1,59],95:[1,147],98:149,100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,53],6:[2,53],28:[2,53],29:[2,53],52:[2,53],57:[2,53],60:[2,53],76:[2,53],81:[2,53],90:[2,53],95:[2,53],97:[2,53],106:[2,53],108:[2,53],109:[2,53],110:[2,53],114:[2,53],122:[2,53],130:[2,53],132:[2,53],133:[2,53],136:[2,53],137:[2,53],138:[2,53],139:[2,53],140:[2,53],141:[2,53]},{4:212,5:3,7:4,8:5,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,29:[1,211],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,192],6:[2,192],28:[2,192],29:[2,192],52:[2,192],57:[2,192],60:[2,192],76:[2,192],81:[2,192],90:[2,192],95:[2,192],97:[2,192],106:[2,192],107:86,108:[2,192],109:[2,192],110:[2,192],113:87,114:[2,192],115:71,122:[2,192],130:[2,192],132:[2,192],133:[2,192],136:[1,77],137:[2,192],138:[2,192],139:[2,192],140:[2,192],141:[2,192]},{107:89,108:[1,67],110:[1,68],113:90,114:[1,70],115:71,130:[1,88]},{1:[2,193],6:[2,193],28:[2,193],29:[2,193],52:[2,193],57:[2,193],60:[2,193],76:[2,193],81:[2,193],90:[2,193],95:[2,193],97:[2,193],106:[2,193],107:86,108:[2,193],109:[2,193],110:[2,193],113:87,114:[2,193],115:71,122:[2,193],130:[2,193],132:[2,193],133:[2,193],136:[1,77],137:[2,193],138:[2,193],139:[2,193],140:[2,193],141:[2,193]},{1:[2,194],6:[2,194],28:[2,194],29:[2,194],52:[2,194],57:[2,194],60:[2,194],76:[2,194],81:[2,194],90:[2,194],95:[2,194],97:[2,194],106:[2,194],107:86,108:[2,194],109:[2,194],110:[2,194],113:87,114:[2,194],115:71,122:[2,194],130:[2,194],132:[2,194],133:[2,194],136:[1,77],137:[2,194],138:[2,194],139:[2,194],140:[2,194],141:[2,194]},{1:[2,195],6:[2,195],28:[2,195],29:[2,195],52:[2,195],57:[2,195],60:[2,195],69:[2,75],70:[2,75],71:[2,75],72:[2,75],74:[2,75],76:[2,195],77:[2,75],81:[2,195],88:[2,75],89:[2,75],90:[2,195],95:[2,195],97:[2,195],106:[2,195],108:[2,195],109:[2,195],110:[2,195],114:[2,195],122:[2,195],130:[2,195],132:[2,195],133:[2,195],136:[2,195],137:[2,195],138:[2,195],139:[2,195],140:[2,195],141:[2,195]},{65:92,69:[1,94],70:[1,95],71:[1,96],72:[1,97],73:98,74:[1,99],77:[1,100],84:91,88:[1,93],89:[2,113]},{65:102,69:[1,94],70:[1,95],71:[1,96],72:[1,97],73:98,74:[1,99],77:[1,100],84:101,88:[1,93],89:[2,113]},{69:[2,78],70:[2,78],71:[2,78],72:[2,78],74:[2,78],77:[2,78],88:[2,78],89:[2,78]},{1:[2,196],6:[2,196],28:[2,196],29:[2,196],52:[2,196],57:[2,196],60:[2,196],69:[2,75],70:[2,75],71:[2,75],72:[2,75],74:[2,75],76:[2,196],77:[2,75],81:[2,196],88:[2,75],89:[2,75],90:[2,196],95:[2,196],97:[2,196],106:[2,196],108:[2,196],109:[2,196],110:[2,196],114:[2,196],122:[2,196],130:[2,196],132:[2,196],133:[2,196],136:[2,196],137:[2,196],138:[2,196],139:[2,196],140:[2,196],141:[2,196]},{1:[2,197],6:[2,197],28:[2,197],29:[2,197],52:[2,197],57:[2,197],60:[2,197],76:[2,197],81:[2,197],90:[2,197],95:[2,197],97:[2,197],106:[2,197],108:[2,197],109:[2,197],110:[2,197],114:[2,197],122:[2,197],130:[2,197],132:[2,197],133:[2,197],136:[2,197],137:[2,197],138:[2,197],139:[2,197],140:[2,197],141:[2,197]},{1:[2,198],6:[2,198],28:[2,198],29:[2,198],52:[2,198],57:[2,198],60:[2,198],76:[2,198],81:[2,198],90:[2,198],95:[2,198],97:[2,198],106:[2,198],108:[2,198],109:[2,198],110:[2,198],114:[2,198],122:[2,198],130:[2,198],132:[2,198],133:[2,198],136:[2,198],137:[2,198],138:[2,198],139:[2,198],140:[2,198],141:[2,198]},{6:[1,215],7:213,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,214],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:216,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{14:217,28:[1,116],129:[1,218]},{1:[2,138],6:[2,138],28:[2,138],29:[2,138],52:[2,138],57:[2,138],60:[2,138],76:[2,138],81:[2,138],90:[2,138],95:[2,138],97:[2,138],101:219,102:[1,220],103:[1,221],106:[2,138],108:[2,138],109:[2,138],110:[2,138],114:[2,138],122:[2,138],130:[2,138],132:[2,138],133:[2,138],136:[2,138],137:[2,138],138:[2,138],139:[2,138],140:[2,138],141:[2,138]},{1:[2,152],6:[2,152],28:[2,152],29:[2,152],52:[2,152],57:[2,152],60:[2,152],76:[2,152],81:[2,152],90:[2,152],95:[2,152],97:[2,152],106:[2,152],108:[2,152],109:[2,152],110:[2,152],114:[2,152],122:[2,152],130:[2,152],132:[2,152],133:[2,152],136:[2,152],137:[2,152],138:[2,152],139:[2,152],140:[2,152],141:[2,152]},{1:[2,160],6:[2,160],28:[2,160],29:[2,160],52:[2,160],57:[2,160],60:[2,160],76:[2,160],81:[2,160],90:[2,160],95:[2,160],97:[2,160],106:[2,160],108:[2,160],109:[2,160],110:[2,160],114:[2,160],122:[2,160],130:[2,160],132:[2,160],133:[2,160],136:[2,160],137:[2,160],138:[2,160],139:[2,160],140:[2,160],141:[2,160]},{28:[1,222],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{124:223,126:224,127:[1,225]},{1:[2,101],6:[2,101],28:[2,101],29:[2,101],52:[2,101],57:[2,101],60:[2,101],76:[2,101],81:[2,101],90:[2,101],95:[2,101],97:[2,101],106:[2,101],108:[2,101],109:[2,101],110:[2,101],114:[2,101],122:[2,101],130:[2,101],132:[2,101],133:[2,101],136:[2,101],137:[2,101],138:[2,101],139:[2,101],140:[2,101],141:[2,101]},{7:226,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,104],6:[2,104],14:227,28:[1,116],29:[2,104],52:[2,104],57:[2,104],60:[2,104],69:[2,75],70:[2,75],71:[2,75],72:[2,75],74:[2,75],76:[2,104],77:[2,75],81:[2,104],83:[1,228],88:[2,75],89:[2,75],90:[2,104],95:[2,104],97:[2,104],106:[2,104],108:[2,104],109:[2,104],110:[2,104],114:[2,104],122:[2,104],130:[2,104],132:[2,104],133:[2,104],136:[2,104],137:[2,104],138:[2,104],139:[2,104],140:[2,104],141:[2,104]},{1:[2,145],6:[2,145],28:[2,145],29:[2,145],52:[2,145],57:[2,145],60:[2,145],76:[2,145],81:[2,145],90:[2,145],95:[2,145],97:[2,145],106:[2,145],107:86,108:[2,145],109:[2,145],110:[2,145],113:87,114:[2,145],115:71,122:[2,145],130:[2,145],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,112],6:[2,112],28:[2,112],29:[2,112],43:[2,112],52:[2,112],57:[2,112],60:[2,112],69:[2,112],70:[2,112],71:[2,112],72:[2,112],74:[2,112],76:[2,112],77:[2,112],81:[2,112],83:[2,112],88:[2,112],89:[2,112],90:[2,112],95:[2,112],97:[2,112],106:[2,112],108:[2,112],109:[2,112],110:[2,112],114:[2,112],122:[2,112],130:[2,112],132:[2,112],133:[2,112],134:[2,112],135:[2,112],136:[2,112],137:[2,112],138:[2,112],139:[2,112],140:[2,112],141:[2,112],142:[2,112]},{1:[2,49],6:[2,49],29:[2,49],106:[2,49],107:86,108:[2,49],110:[2,49],113:87,114:[2,49],115:71,130:[2,49],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,12],6:[2,12],29:[2,12],106:[2,12],108:[2,12],110:[2,12],114:[2,12],130:[2,12]},{1:[2,13],6:[2,13],29:[2,13],106:[2,13],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[2,13],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{6:[1,76],106:[1,229]},{4:230,5:3,7:4,8:5,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{6:[2,134],28:[2,134],57:[2,134],60:[1,232],95:[2,134],96:231,97:[1,196],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,120],6:[2,120],28:[2,120],29:[2,120],43:[2,120],52:[2,120],57:[2,120],60:[2,120],69:[2,120],70:[2,120],71:[2,120],72:[2,120],74:[2,120],76:[2,120],77:[2,120],81:[2,120],88:[2,120],89:[2,120],90:[2,120],95:[2,120],97:[2,120],106:[2,120],108:[2,120],109:[2,120],110:[2,120],114:[2,120],120:[2,120],121:[2,120],122:[2,120],130:[2,120],132:[2,120],133:[2,120],136:[2,120],137:[2,120],138:[2,120],139:[2,120],140:[2,120],141:[2,120]},{6:[2,56],28:[2,56],56:233,57:[1,234],95:[2,56]},{6:[2,129],28:[2,129],29:[2,129],57:[2,129],90:[2,129],95:[2,129]},{7:205,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,150],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,63:151,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],91:235,92:[1,60],93:[1,61],94:[1,59],98:149,100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{6:[2,135],28:[2,135],29:[2,135],57:[2,135],90:[2,135],95:[2,135]},{1:[2,119],6:[2,119],28:[2,119],29:[2,119],43:[2,119],46:[2,119],52:[2,119],57:[2,119],60:[2,119],69:[2,119],70:[2,119],71:[2,119],72:[2,119],74:[2,119],76:[2,119],77:[2,119],81:[2,119],83:[2,119],88:[2,119],89:[2,119],90:[2,119],95:[2,119],97:[2,119],106:[2,119],108:[2,119],109:[2,119],110:[2,119],114:[2,119],120:[2,119],121:[2,119],122:[2,119],130:[2,119],132:[2,119],133:[2,119],134:[2,119],135:[2,119],136:[2,119],137:[2,119],138:[2,119],139:[2,119],140:[2,119],141:[2,119],142:[2,119]},{14:236,28:[1,116],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,148],6:[2,148],28:[2,148],29:[2,148],52:[2,148],57:[2,148],60:[2,148],76:[2,148],81:[2,148],90:[2,148],95:[2,148],97:[2,148],106:[2,148],107:86,108:[1,67],109:[1,237],110:[1,68],113:87,114:[1,70],115:71,122:[2,148],130:[2,148],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,150],6:[2,150],28:[2,150],29:[2,150],52:[2,150],57:[2,150],60:[2,150],76:[2,150],81:[2,150],90:[2,150],95:[2,150],97:[2,150],106:[2,150],107:86,108:[1,67],109:[1,238],110:[1,68],113:87,114:[1,70],115:71,122:[2,150],130:[2,150],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,156],6:[2,156],28:[2,156],29:[2,156],52:[2,156],57:[2,156],60:[2,156],76:[2,156],81:[2,156],90:[2,156],95:[2,156],97:[2,156],106:[2,156],108:[2,156],109:[2,156],110:[2,156],114:[2,156],122:[2,156],130:[2,156],132:[2,156],133:[2,156],136:[2,156],137:[2,156],138:[2,156],139:[2,156],140:[2,156],141:[2,156]},{1:[2,157],6:[2,157],28:[2,157],29:[2,157],52:[2,157],57:[2,157],60:[2,157],76:[2,157],81:[2,157],90:[2,157],95:[2,157],97:[2,157],106:[2,157],107:86,108:[1,67],109:[2,157],110:[1,68],113:87,114:[1,70],115:71,122:[2,157],130:[2,157],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,161],6:[2,161],28:[2,161],29:[2,161],52:[2,161],57:[2,161],60:[2,161],76:[2,161],81:[2,161],90:[2,161],95:[2,161],97:[2,161],106:[2,161],108:[2,161],109:[2,161],110:[2,161],114:[2,161],122:[2,161],130:[2,161],132:[2,161],133:[2,161],136:[2,161],137:[2,161],138:[2,161],139:[2,161],140:[2,161],141:[2,161]},{120:[2,163],121:[2,163]},{30:162,31:[1,75],47:163,61:164,62:165,79:[1,72],93:[1,113],94:[1,114],117:239,119:161},{57:[1,240],120:[2,169],121:[2,169]},{57:[2,165],120:[2,165],121:[2,165]},{57:[2,166],120:[2,166],121:[2,166]},{57:[2,167],120:[2,167],121:[2,167]},{57:[2,168],120:[2,168],121:[2,168]},{1:[2,162],6:[2,162],28:[2,162],29:[2,162],52:[2,162],57:[2,162],60:[2,162],76:[2,162],81:[2,162],90:[2,162],95:[2,162],97:[2,162],106:[2,162],108:[2,162],109:[2,162],110:[2,162],114:[2,162],122:[2,162],130:[2,162],132:[2,162],133:[2,162],136:[2,162],137:[2,162],138:[2,162],139:[2,162],140:[2,162],141:[2,162]},{7:241,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:242,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{6:[2,56],28:[2,56],56:243,57:[1,244],81:[2,56]},{6:[2,96],28:[2,96],29:[2,96],57:[2,96],81:[2,96]},{6:[2,42],28:[2,42],29:[2,42],46:[1,245],57:[2,42],81:[2,42]},{6:[2,45],28:[2,45],29:[2,45],57:[2,45],81:[2,45]},{6:[2,46],28:[2,46],29:[2,46],46:[2,46],57:[2,46],81:[2,46]},{6:[2,47],28:[2,47],29:[2,47],46:[2,47],57:[2,47],81:[2,47]},{6:[2,48],28:[2,48],29:[2,48],46:[2,48],57:[2,48],81:[2,48]},{1:[2,4],6:[2,4],29:[2,4],106:[2,4]},{1:[2,200],6:[2,200],28:[2,200],29:[2,200],52:[2,200],57:[2,200],60:[2,200],76:[2,200],81:[2,200],90:[2,200],95:[2,200],97:[2,200],106:[2,200],107:86,108:[2,200],109:[2,200],110:[2,200],113:87,114:[2,200],115:71,122:[2,200],130:[2,200],132:[2,200],133:[2,200],136:[1,77],137:[1,80],138:[2,200],139:[2,200],140:[2,200],141:[2,200]},{1:[2,201],6:[2,201],28:[2,201],29:[2,201],52:[2,201],57:[2,201],60:[2,201],76:[2,201],81:[2,201],90:[2,201],95:[2,201],97:[2,201],106:[2,201],107:86,108:[2,201],109:[2,201],110:[2,201],113:87,114:[2,201],115:71,122:[2,201],130:[2,201],132:[2,201],133:[2,201],136:[1,77],137:[1,80],138:[2,201],139:[2,201],140:[2,201],141:[2,201]},{1:[2,202],6:[2,202],28:[2,202],29:[2,202],52:[2,202],57:[2,202],60:[2,202],76:[2,202],81:[2,202],90:[2,202],95:[2,202],97:[2,202],106:[2,202],107:86,108:[2,202],109:[2,202],110:[2,202],113:87,114:[2,202],115:71,122:[2,202],130:[2,202],132:[2,202],133:[2,202],136:[1,77],137:[2,202],138:[2,202],139:[2,202],140:[2,202],141:[2,202]},{1:[2,203],6:[2,203],28:[2,203],29:[2,203],52:[2,203],57:[2,203],60:[2,203],76:[2,203],81:[2,203],90:[2,203],95:[2,203],97:[2,203],106:[2,203],107:86,108:[2,203],109:[2,203],110:[2,203],113:87,114:[2,203],115:71,122:[2,203],130:[2,203],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[2,203],139:[2,203],140:[2,203],141:[2,203]},{1:[2,204],6:[2,204],28:[2,204],29:[2,204],52:[2,204],57:[2,204],60:[2,204],76:[2,204],81:[2,204],90:[2,204],95:[2,204],97:[2,204],106:[2,204],107:86,108:[2,204],109:[2,204],110:[2,204],113:87,114:[2,204],115:71,122:[2,204],130:[2,204],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[2,204],140:[2,204],141:[1,84]},{1:[2,205],6:[2,205],28:[2,205],29:[2,205],52:[2,205],57:[2,205],60:[2,205],76:[2,205],81:[2,205],90:[2,205],95:[2,205],97:[2,205],106:[2,205],107:86,108:[2,205],109:[2,205],110:[2,205],113:87,114:[2,205],115:71,122:[2,205],130:[2,205],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[2,205],141:[1,84]},{1:[2,206],6:[2,206],28:[2,206],29:[2,206],52:[2,206],57:[2,206],60:[2,206],76:[2,206],81:[2,206],90:[2,206],95:[2,206],97:[2,206],106:[2,206],107:86,108:[2,206],109:[2,206],110:[2,206],113:87,114:[2,206],115:71,122:[2,206],130:[2,206],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[2,206],140:[2,206],141:[2,206]},{1:[2,191],6:[2,191],28:[2,191],29:[2,191],52:[2,191],57:[2,191],60:[2,191],76:[2,191],81:[2,191],90:[2,191],95:[2,191],97:[2,191],106:[2,191],107:86,108:[1,67],109:[2,191],110:[1,68],113:87,114:[1,70],115:71,122:[2,191],130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,190],6:[2,190],28:[2,190],29:[2,190],52:[2,190],57:[2,190],60:[2,190],76:[2,190],81:[2,190],90:[2,190],95:[2,190],97:[2,190],106:[2,190],107:86,108:[1,67],109:[2,190],110:[1,68],113:87,114:[1,70],115:71,122:[2,190],130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,108],6:[2,108],28:[2,108],29:[2,108],52:[2,108],57:[2,108],60:[2,108],69:[2,108],70:[2,108],71:[2,108],72:[2,108],74:[2,108],76:[2,108],77:[2,108],81:[2,108],88:[2,108],89:[2,108],90:[2,108],95:[2,108],97:[2,108],106:[2,108],108:[2,108],109:[2,108],110:[2,108],114:[2,108],122:[2,108],130:[2,108],132:[2,108],133:[2,108],136:[2,108],137:[2,108],138:[2,108],139:[2,108],140:[2,108],141:[2,108]},{1:[2,83],6:[2,83],28:[2,83],29:[2,83],43:[2,83],52:[2,83],57:[2,83],60:[2,83],69:[2,83],70:[2,83],71:[2,83],72:[2,83],74:[2,83],76:[2,83],77:[2,83],81:[2,83],83:[2,83],88:[2,83],89:[2,83],90:[2,83],95:[2,83],97:[2,83],106:[2,83],108:[2,83],109:[2,83],110:[2,83],114:[2,83],122:[2,83],130:[2,83],132:[2,83],133:[2,83],134:[2,83],135:[2,83],136:[2,83],137:[2,83],138:[2,83],139:[2,83],140:[2,83],141:[2,83],142:[2,83]},{1:[2,84],6:[2,84],28:[2,84],29:[2,84],43:[2,84],52:[2,84],57:[2,84],60:[2,84],69:[2,84],70:[2,84],71:[2,84],72:[2,84],74:[2,84],76:[2,84],77:[2,84],81:[2,84],83:[2,84],88:[2,84],89:[2,84],90:[2,84],95:[2,84],97:[2,84],106:[2,84],108:[2,84],109:[2,84],110:[2,84],114:[2,84],122:[2,84],130:[2,84],132:[2,84],133:[2,84],134:[2,84],135:[2,84],136:[2,84],137:[2,84],138:[2,84],139:[2,84],140:[2,84],141:[2,84],142:[2,84]},{1:[2,85],6:[2,85],28:[2,85],29:[2,85],43:[2,85],52:[2,85],57:[2,85],60:[2,85],69:[2,85],70:[2,85],71:[2,85],72:[2,85],74:[2,85],76:[2,85],77:[2,85],81:[2,85],83:[2,85],88:[2,85],89:[2,85],90:[2,85],95:[2,85],97:[2,85],106:[2,85],108:[2,85],109:[2,85],110:[2,85],114:[2,85],122:[2,85],130:[2,85],132:[2,85],133:[2,85],134:[2,85],135:[2,85],136:[2,85],137:[2,85],138:[2,85],139:[2,85],140:[2,85],141:[2,85],142:[2,85]},{1:[2,86],6:[2,86],28:[2,86],29:[2,86],43:[2,86],52:[2,86],57:[2,86],60:[2,86],69:[2,86],70:[2,86],71:[2,86],72:[2,86],74:[2,86],76:[2,86],77:[2,86],81:[2,86],83:[2,86],88:[2,86],89:[2,86],90:[2,86],95:[2,86],97:[2,86],106:[2,86],108:[2,86],109:[2,86],110:[2,86],114:[2,86],122:[2,86],130:[2,86],132:[2,86],133:[2,86],134:[2,86],135:[2,86],136:[2,86],137:[2,86],138:[2,86],139:[2,86],140:[2,86],141:[2,86],142:[2,86]},{1:[2,87],6:[2,87],28:[2,87],29:[2,87],43:[2,87],52:[2,87],57:[2,87],60:[2,87],69:[2,87],70:[2,87],71:[2,87],72:[2,87],74:[2,87],76:[2,87],77:[2,87],81:[2,87],83:[2,87],88:[2,87],89:[2,87],90:[2,87],95:[2,87],97:[2,87],106:[2,87],108:[2,87],109:[2,87],110:[2,87],114:[2,87],122:[2,87],130:[2,87],132:[2,87],133:[2,87],134:[2,87],135:[2,87],136:[2,87],137:[2,87],138:[2,87],139:[2,87],140:[2,87],141:[2,87],142:[2,87]},{76:[1,246]},{60:[1,197],76:[2,92],96:247,97:[1,196],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{76:[2,93]},{7:248,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,76:[2,128],79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{11:[2,122],13:[2,122],31:[2,122],33:[2,122],34:[2,122],36:[2,122],37:[2,122],38:[2,122],39:[2,122],40:[2,122],41:[2,122],48:[2,122],49:[2,122],50:[2,122],54:[2,122],55:[2,122],76:[2,122],79:[2,122],82:[2,122],86:[2,122],87:[2,122],92:[2,122],93:[2,122],94:[2,122],100:[2,122],104:[2,122],105:[2,122],108:[2,122],110:[2,122],112:[2,122],114:[2,122],123:[2,122],129:[2,122],131:[2,122],132:[2,122],133:[2,122],134:[2,122],135:[2,122]},{11:[2,123],13:[2,123],31:[2,123],33:[2,123],34:[2,123],36:[2,123],37:[2,123],38:[2,123],39:[2,123],40:[2,123],41:[2,123],48:[2,123],49:[2,123],50:[2,123],54:[2,123],55:[2,123],76:[2,123],79:[2,123],82:[2,123],86:[2,123],87:[2,123],92:[2,123],93:[2,123],94:[2,123],100:[2,123],104:[2,123],105:[2,123],108:[2,123],110:[2,123],112:[2,123],114:[2,123],123:[2,123],129:[2,123],131:[2,123],132:[2,123],133:[2,123],134:[2,123],135:[2,123]},{1:[2,91],6:[2,91],28:[2,91],29:[2,91],43:[2,91],52:[2,91],57:[2,91],60:[2,91],69:[2,91],70:[2,91],71:[2,91],72:[2,91],74:[2,91],76:[2,91],77:[2,91],81:[2,91],83:[2,91],88:[2,91],89:[2,91],90:[2,91],95:[2,91],97:[2,91],106:[2,91],108:[2,91],109:[2,91],110:[2,91],114:[2,91],122:[2,91],130:[2,91],132:[2,91],133:[2,91],134:[2,91],135:[2,91],136:[2,91],137:[2,91],138:[2,91],139:[2,91],140:[2,91],141:[2,91],142:[2,91]},{1:[2,109],6:[2,109],28:[2,109],29:[2,109],52:[2,109],57:[2,109],60:[2,109],69:[2,109],70:[2,109],71:[2,109],72:[2,109],74:[2,109],76:[2,109],77:[2,109],81:[2,109],88:[2,109],89:[2,109],90:[2,109],95:[2,109],97:[2,109],106:[2,109],108:[2,109],109:[2,109],110:[2,109],114:[2,109],122:[2,109],130:[2,109],132:[2,109],133:[2,109],136:[2,109],137:[2,109],138:[2,109],139:[2,109],140:[2,109],141:[2,109]},{1:[2,39],6:[2,39],28:[2,39],29:[2,39],52:[2,39],57:[2,39],60:[2,39],76:[2,39],81:[2,39],90:[2,39],95:[2,39],97:[2,39],106:[2,39],107:86,108:[2,39],109:[2,39],110:[2,39],113:87,114:[2,39],115:71,122:[2,39],130:[2,39],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{7:249,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:250,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,115],6:[2,115],28:[2,115],29:[2,115],43:[2,115],52:[2,115],57:[2,115],60:[2,115],69:[2,115],70:[2,115],71:[2,115],72:[2,115],74:[2,115],76:[2,115],77:[2,115],81:[2,115],83:[2,115],88:[2,115],89:[2,115],90:[2,115],95:[2,115],97:[2,115],106:[2,115],108:[2,115],109:[2,115],110:[2,115],114:[2,115],122:[2,115],130:[2,115],132:[2,115],133:[2,115],134:[2,115],135:[2,115],136:[2,115],137:[2,115],138:[2,115],139:[2,115],140:[2,115],141:[2,115],142:[2,115]},{6:[2,56],28:[2,56],56:251,57:[1,234],90:[2,56]},{6:[2,134],28:[2,134],29:[2,134],57:[2,134],60:[1,252],90:[2,134],95:[2,134],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{53:253,54:[1,62],55:[1,63]},{6:[2,57],28:[2,57],29:[2,57],30:109,31:[1,75],47:110,58:254,59:108,61:111,62:112,79:[1,72],93:[1,113],94:[1,114]},{6:[1,255],28:[1,256]},{6:[2,64],28:[2,64],29:[2,64],52:[2,64],57:[2,64]},{7:257,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,27],6:[2,27],28:[2,27],29:[2,27],52:[2,27],57:[2,27],60:[2,27],76:[2,27],81:[2,27],90:[2,27],95:[2,27],97:[2,27],102:[2,27],103:[2,27],106:[2,27],108:[2,27],109:[2,27],110:[2,27],114:[2,27],122:[2,27],125:[2,27],127:[2,27],130:[2,27],132:[2,27],133:[2,27],136:[2,27],137:[2,27],138:[2,27],139:[2,27],140:[2,27],141:[2,27]},{6:[1,76],29:[1,258]},{1:[2,207],6:[2,207],28:[2,207],29:[2,207],52:[2,207],57:[2,207],60:[2,207],76:[2,207],81:[2,207],90:[2,207],95:[2,207],97:[2,207],106:[2,207],107:86,108:[2,207],109:[2,207],110:[2,207],113:87,114:[2,207],115:71,122:[2,207],130:[2,207],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{7:259,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:260,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,210],6:[2,210],28:[2,210],29:[2,210],52:[2,210],57:[2,210],60:[2,210],76:[2,210],81:[2,210],90:[2,210],95:[2,210],97:[2,210],106:[2,210],107:86,108:[2,210],109:[2,210],110:[2,210],113:87,114:[2,210],115:71,122:[2,210],130:[2,210],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,189],6:[2,189],28:[2,189],29:[2,189],52:[2,189],57:[2,189],60:[2,189],76:[2,189],81:[2,189],90:[2,189],95:[2,189],97:[2,189],106:[2,189],108:[2,189],109:[2,189],110:[2,189],114:[2,189],122:[2,189],130:[2,189],132:[2,189],133:[2,189],136:[2,189],137:[2,189],138:[2,189],139:[2,189],140:[2,189],141:[2,189]},{7:261,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,139],6:[2,139],28:[2,139],29:[2,139],52:[2,139],57:[2,139],60:[2,139],76:[2,139],81:[2,139],90:[2,139],95:[2,139],97:[2,139],102:[1,262],106:[2,139],108:[2,139],109:[2,139],110:[2,139],114:[2,139],122:[2,139],130:[2,139],132:[2,139],133:[2,139],136:[2,139],137:[2,139],138:[2,139],139:[2,139],140:[2,139],141:[2,139]},{14:263,28:[1,116]},{14:266,28:[1,116],30:264,31:[1,75],62:265,79:[1,72]},{124:267,126:224,127:[1,225]},{29:[1,268],125:[1,269],126:270,127:[1,225]},{29:[2,182],125:[2,182],127:[2,182]},{7:272,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],99:271,100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,102],6:[2,102],14:273,28:[1,116],29:[2,102],52:[2,102],57:[2,102],60:[2,102],76:[2,102],81:[2,102],90:[2,102],95:[2,102],97:[2,102],106:[2,102],107:86,108:[1,67],109:[2,102],110:[1,68],113:87,114:[1,70],115:71,122:[2,102],130:[2,102],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,105],6:[2,105],28:[2,105],29:[2,105],52:[2,105],57:[2,105],60:[2,105],76:[2,105],81:[2,105],90:[2,105],95:[2,105],97:[2,105],106:[2,105],108:[2,105],109:[2,105],110:[2,105],114:[2,105],122:[2,105],130:[2,105],132:[2,105],133:[2,105],136:[2,105],137:[2,105],138:[2,105],139:[2,105],140:[2,105],141:[2,105]},{7:274,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,146],6:[2,146],28:[2,146],29:[2,146],52:[2,146],57:[2,146],60:[2,146],69:[2,146],70:[2,146],71:[2,146],72:[2,146],74:[2,146],76:[2,146],77:[2,146],81:[2,146],88:[2,146],89:[2,146],90:[2,146],95:[2,146],97:[2,146],106:[2,146],108:[2,146],109:[2,146],110:[2,146],114:[2,146],122:[2,146],130:[2,146],132:[2,146],133:[2,146],136:[2,146],137:[2,146],138:[2,146],139:[2,146],140:[2,146],141:[2,146]},{6:[1,76],29:[1,275]},{7:276,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{6:[2,70],11:[2,123],13:[2,123],28:[2,70],31:[2,123],33:[2,123],34:[2,123],36:[2,123],37:[2,123],38:[2,123],39:[2,123],40:[2,123],41:[2,123],48:[2,123],49:[2,123],50:[2,123],54:[2,123],55:[2,123],57:[2,70],79:[2,123],82:[2,123],86:[2,123],87:[2,123],92:[2,123],93:[2,123],94:[2,123],95:[2,70],100:[2,123],104:[2,123],105:[2,123],108:[2,123],110:[2,123],112:[2,123],114:[2,123],123:[2,123],129:[2,123],131:[2,123],132:[2,123],133:[2,123],134:[2,123],135:[2,123]},{6:[1,278],28:[1,279],95:[1,277]},{6:[2,57],7:205,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[2,57],29:[2,57],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,63:151,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],90:[2,57],92:[1,60],93:[1,61],94:[1,59],95:[2,57],98:280,100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{6:[2,56],28:[2,56],29:[2,56],56:281,57:[1,234]},{1:[2,186],6:[2,186],28:[2,186],29:[2,186],52:[2,186],57:[2,186],60:[2,186],76:[2,186],81:[2,186],90:[2,186],95:[2,186],97:[2,186],106:[2,186],108:[2,186],109:[2,186],110:[2,186],114:[2,186],122:[2,186],125:[2,186],130:[2,186],132:[2,186],133:[2,186],136:[2,186],137:[2,186],138:[2,186],139:[2,186],140:[2,186],141:[2,186]},{7:282,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:283,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{120:[2,164],121:[2,164]},{30:162,31:[1,75],47:163,61:164,62:165,79:[1,72],93:[1,113],94:[1,114],119:284},{1:[2,171],6:[2,171],28:[2,171],29:[2,171],52:[2,171],57:[2,171],60:[2,171],76:[2,171],81:[2,171],90:[2,171],95:[2,171],97:[2,171],106:[2,171],107:86,108:[2,171],109:[1,285],110:[2,171],113:87,114:[2,171],115:71,122:[1,286],130:[2,171],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,172],6:[2,172],28:[2,172],29:[2,172],52:[2,172],57:[2,172],60:[2,172],76:[2,172],81:[2,172],90:[2,172],95:[2,172],97:[2,172],106:[2,172],107:86,108:[2,172],109:[1,287],110:[2,172],113:87,114:[2,172],115:71,122:[2,172],130:[2,172],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{6:[1,289],28:[1,290],81:[1,288]},{6:[2,57],10:172,28:[2,57],29:[2,57],30:173,31:[1,75],32:174,33:[1,73],34:[1,74],44:291,45:171,47:175,49:[1,47],81:[2,57],93:[1,113]},{7:292,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,293],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,90],6:[2,90],28:[2,90],29:[2,90],43:[2,90],52:[2,90],57:[2,90],60:[2,90],69:[2,90],70:[2,90],71:[2,90],72:[2,90],74:[2,90],76:[2,90],77:[2,90],81:[2,90],83:[2,90],88:[2,90],89:[2,90],90:[2,90],95:[2,90],97:[2,90],106:[2,90],108:[2,90],109:[2,90],110:[2,90],114:[2,90],122:[2,90],130:[2,90],132:[2,90],133:[2,90],134:[2,90],135:[2,90],136:[2,90],137:[2,90],138:[2,90],139:[2,90],140:[2,90],141:[2,90],142:[2,90]},{7:294,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,76:[2,126],79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{76:[2,127],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,40],6:[2,40],28:[2,40],29:[2,40],52:[2,40],57:[2,40],60:[2,40],76:[2,40],81:[2,40],90:[2,40],95:[2,40],97:[2,40],106:[2,40],107:86,108:[2,40],109:[2,40],110:[2,40],113:87,114:[2,40],115:71,122:[2,40],130:[2,40],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{29:[1,295],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{6:[1,278],28:[1,279],90:[1,296]},{6:[2,70],28:[2,70],29:[2,70],57:[2,70],90:[2,70],95:[2,70]},{14:297,28:[1,116]},{6:[2,60],28:[2,60],29:[2,60],52:[2,60],57:[2,60]},{30:109,31:[1,75],47:110,58:298,59:108,61:111,62:112,79:[1,72],93:[1,113],94:[1,114]},{6:[2,58],28:[2,58],29:[2,58],30:109,31:[1,75],47:110,51:299,57:[2,58],58:107,59:108,61:111,62:112,79:[1,72],93:[1,113],94:[1,114]},{6:[2,65],28:[2,65],29:[2,65],52:[2,65],57:[2,65],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,28],6:[2,28],28:[2,28],29:[2,28],52:[2,28],57:[2,28],60:[2,28],76:[2,28],81:[2,28],90:[2,28],95:[2,28],97:[2,28],102:[2,28],103:[2,28],106:[2,28],108:[2,28],109:[2,28],110:[2,28],114:[2,28],122:[2,28],125:[2,28],127:[2,28],130:[2,28],132:[2,28],133:[2,28],136:[2,28],137:[2,28],138:[2,28],139:[2,28],140:[2,28],141:[2,28]},{29:[1,300],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,209],6:[2,209],28:[2,209],29:[2,209],52:[2,209],57:[2,209],60:[2,209],76:[2,209],81:[2,209],90:[2,209],95:[2,209],97:[2,209],106:[2,209],107:86,108:[2,209],109:[2,209],110:[2,209],113:87,114:[2,209],115:71,122:[2,209],130:[2,209],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{14:301,28:[1,116],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{14:302,28:[1,116]},{1:[2,140],6:[2,140],28:[2,140],29:[2,140],52:[2,140],57:[2,140],60:[2,140],76:[2,140],81:[2,140],90:[2,140],95:[2,140],97:[2,140],106:[2,140],108:[2,140],109:[2,140],110:[2,140],114:[2,140],122:[2,140],130:[2,140],132:[2,140],133:[2,140],136:[2,140],137:[2,140],138:[2,140],139:[2,140],140:[2,140],141:[2,140]},{14:303,28:[1,116]},{14:304,28:[1,116]},{1:[2,144],6:[2,144],28:[2,144],29:[2,144],52:[2,144],57:[2,144],60:[2,144],76:[2,144],81:[2,144],90:[2,144],95:[2,144],97:[2,144],102:[2,144],106:[2,144],108:[2,144],109:[2,144],110:[2,144],114:[2,144],122:[2,144],130:[2,144],132:[2,144],133:[2,144],136:[2,144],137:[2,144],138:[2,144],139:[2,144],140:[2,144],141:[2,144]},{29:[1,305],125:[1,306],126:270,127:[1,225]},{1:[2,180],6:[2,180],28:[2,180],29:[2,180],52:[2,180],57:[2,180],60:[2,180],76:[2,180],81:[2,180],90:[2,180],95:[2,180],97:[2,180],106:[2,180],108:[2,180],109:[2,180],110:[2,180],114:[2,180],122:[2,180],130:[2,180],132:[2,180],133:[2,180],136:[2,180],137:[2,180],138:[2,180],139:[2,180],140:[2,180],141:[2,180]},{14:307,28:[1,116]},{29:[2,183],125:[2,183],127:[2,183]},{14:308,28:[1,116],57:[1,309]},{28:[2,136],57:[2,136],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,103],6:[2,103],28:[2,103],29:[2,103],52:[2,103],57:[2,103],60:[2,103],76:[2,103],81:[2,103],90:[2,103],95:[2,103],97:[2,103],106:[2,103],108:[2,103],109:[2,103],110:[2,103],114:[2,103],122:[2,103],130:[2,103],132:[2,103],133:[2,103],136:[2,103],137:[2,103],138:[2,103],139:[2,103],140:[2,103],141:[2,103]},{1:[2,106],6:[2,106],14:310,28:[1,116],29:[2,106],52:[2,106],57:[2,106],60:[2,106],76:[2,106],81:[2,106],90:[2,106],95:[2,106],97:[2,106],106:[2,106],107:86,108:[1,67],109:[2,106],110:[1,68],113:87,114:[1,70],115:71,122:[2,106],130:[2,106],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{106:[1,311]},{95:[1,312],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,121],6:[2,121],28:[2,121],29:[2,121],43:[2,121],52:[2,121],57:[2,121],60:[2,121],69:[2,121],70:[2,121],71:[2,121],72:[2,121],74:[2,121],76:[2,121],77:[2,121],81:[2,121],88:[2,121],89:[2,121],90:[2,121],95:[2,121],97:[2,121],106:[2,121],108:[2,121],109:[2,121],110:[2,121],114:[2,121],120:[2,121],121:[2,121],122:[2,121],130:[2,121],132:[2,121],133:[2,121],136:[2,121],137:[2,121],138:[2,121],139:[2,121],140:[2,121],141:[2,121]},{7:205,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,63:151,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],98:313,100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:205,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,28:[1,150],30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,63:151,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],91:314,92:[1,60],93:[1,61],94:[1,59],98:149,100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{6:[2,130],28:[2,130],29:[2,130],57:[2,130],90:[2,130],95:[2,130]},{6:[1,278],28:[1,279],29:[1,315]},{1:[2,149],6:[2,149],28:[2,149],29:[2,149],52:[2,149],57:[2,149],60:[2,149],76:[2,149],81:[2,149],90:[2,149],95:[2,149],97:[2,149],106:[2,149],107:86,108:[1,67],109:[2,149],110:[1,68],113:87,114:[1,70],115:71,122:[2,149],130:[2,149],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,151],6:[2,151],28:[2,151],29:[2,151],52:[2,151],57:[2,151],60:[2,151],76:[2,151],81:[2,151],90:[2,151],95:[2,151],97:[2,151],106:[2,151],107:86,108:[1,67],109:[2,151],110:[1,68],113:87,114:[1,70],115:71,122:[2,151],130:[2,151],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{120:[2,170],121:[2,170]},{7:316,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:317,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:318,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,94],6:[2,94],28:[2,94],29:[2,94],43:[2,94],52:[2,94],57:[2,94],60:[2,94],69:[2,94],70:[2,94],71:[2,94],72:[2,94],74:[2,94],76:[2,94],77:[2,94],81:[2,94],88:[2,94],89:[2,94],90:[2,94],95:[2,94],97:[2,94],106:[2,94],108:[2,94],109:[2,94],110:[2,94],114:[2,94],120:[2,94],121:[2,94],122:[2,94],130:[2,94],132:[2,94],133:[2,94],136:[2,94],137:[2,94],138:[2,94],139:[2,94],140:[2,94],141:[2,94]},{10:172,30:173,31:[1,75],32:174,33:[1,73],34:[1,74],44:319,45:171,47:175,49:[1,47],93:[1,113]},{6:[2,95],10:172,28:[2,95],29:[2,95],30:173,31:[1,75],32:174,33:[1,73],34:[1,74],44:170,45:171,47:175,49:[1,47],57:[2,95],80:320,93:[1,113]},{6:[2,97],28:[2,97],29:[2,97],57:[2,97],81:[2,97]},{6:[2,43],28:[2,43],29:[2,43],57:[2,43],81:[2,43],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{7:321,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{76:[2,125],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,41],6:[2,41],28:[2,41],29:[2,41],52:[2,41],57:[2,41],60:[2,41],76:[2,41],81:[2,41],90:[2,41],95:[2,41],97:[2,41],106:[2,41],108:[2,41],109:[2,41],110:[2,41],114:[2,41],122:[2,41],130:[2,41],132:[2,41],133:[2,41],136:[2,41],137:[2,41],138:[2,41],139:[2,41],140:[2,41],141:[2,41]},{1:[2,116],6:[2,116],28:[2,116],29:[2,116],43:[2,116],52:[2,116],57:[2,116],60:[2,116],69:[2,116],70:[2,116],71:[2,116],72:[2,116],74:[2,116],76:[2,116],77:[2,116],81:[2,116],83:[2,116],88:[2,116],89:[2,116],90:[2,116],95:[2,116],97:[2,116],106:[2,116],108:[2,116],109:[2,116],110:[2,116],114:[2,116],122:[2,116],130:[2,116],132:[2,116],133:[2,116],134:[2,116],135:[2,116],136:[2,116],137:[2,116],138:[2,116],139:[2,116],140:[2,116],141:[2,116],142:[2,116]},{1:[2,52],6:[2,52],28:[2,52],29:[2,52],52:[2,52],57:[2,52],60:[2,52],76:[2,52],81:[2,52],90:[2,52],95:[2,52],97:[2,52],106:[2,52],108:[2,52],109:[2,52],110:[2,52],114:[2,52],122:[2,52],130:[2,52],132:[2,52],133:[2,52],136:[2,52],137:[2,52],138:[2,52],139:[2,52],140:[2,52],141:[2,52]},{6:[2,61],28:[2,61],29:[2,61],52:[2,61],57:[2,61]},{6:[2,56],28:[2,56],29:[2,56],56:322,57:[1,207]},{1:[2,208],6:[2,208],28:[2,208],29:[2,208],52:[2,208],57:[2,208],60:[2,208],76:[2,208],81:[2,208],90:[2,208],95:[2,208],97:[2,208],106:[2,208],108:[2,208],109:[2,208],110:[2,208],114:[2,208],122:[2,208],130:[2,208],132:[2,208],133:[2,208],136:[2,208],137:[2,208],138:[2,208],139:[2,208],140:[2,208],141:[2,208]},{1:[2,187],6:[2,187],28:[2,187],29:[2,187],52:[2,187],57:[2,187],60:[2,187],76:[2,187],81:[2,187],90:[2,187],95:[2,187],97:[2,187],106:[2,187],108:[2,187],109:[2,187],110:[2,187],114:[2,187],122:[2,187],125:[2,187],130:[2,187],132:[2,187],133:[2,187],136:[2,187],137:[2,187],138:[2,187],139:[2,187],140:[2,187],141:[2,187]},{1:[2,141],6:[2,141],28:[2,141],29:[2,141],52:[2,141],57:[2,141],60:[2,141],76:[2,141],81:[2,141],90:[2,141],95:[2,141],97:[2,141],106:[2,141],108:[2,141],109:[2,141],110:[2,141],114:[2,141],122:[2,141],130:[2,141],132:[2,141],133:[2,141],136:[2,141],137:[2,141],138:[2,141],139:[2,141],140:[2,141],141:[2,141]},{1:[2,142],6:[2,142],28:[2,142],29:[2,142],52:[2,142],57:[2,142],60:[2,142],76:[2,142],81:[2,142],90:[2,142],95:[2,142],97:[2,142],102:[2,142],106:[2,142],108:[2,142],109:[2,142],110:[2,142],114:[2,142],122:[2,142],130:[2,142],132:[2,142],133:[2,142],136:[2,142],137:[2,142],138:[2,142],139:[2,142],140:[2,142],141:[2,142]},{1:[2,143],6:[2,143],28:[2,143],29:[2,143],52:[2,143],57:[2,143],60:[2,143],76:[2,143],81:[2,143],90:[2,143],95:[2,143],97:[2,143],102:[2,143],106:[2,143],108:[2,143],109:[2,143],110:[2,143],114:[2,143],122:[2,143],130:[2,143],132:[2,143],133:[2,143],136:[2,143],137:[2,143],138:[2,143],139:[2,143],140:[2,143],141:[2,143]},{1:[2,178],6:[2,178],28:[2,178],29:[2,178],52:[2,178],57:[2,178],60:[2,178],76:[2,178],81:[2,178],90:[2,178],95:[2,178],97:[2,178],106:[2,178],108:[2,178],109:[2,178],110:[2,178],114:[2,178],122:[2,178],130:[2,178],132:[2,178],133:[2,178],136:[2,178],137:[2,178],138:[2,178],139:[2,178],140:[2,178],141:[2,178]},{14:323,28:[1,116]},{29:[1,324]},{6:[1,325],29:[2,184],125:[2,184],127:[2,184]},{7:326,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{1:[2,107],6:[2,107],28:[2,107],29:[2,107],52:[2,107],57:[2,107],60:[2,107],76:[2,107],81:[2,107],90:[2,107],95:[2,107],97:[2,107],106:[2,107],108:[2,107],109:[2,107],110:[2,107],114:[2,107],122:[2,107],130:[2,107],132:[2,107],133:[2,107],136:[2,107],137:[2,107],138:[2,107],139:[2,107],140:[2,107],141:[2,107]},{1:[2,147],6:[2,147],28:[2,147],29:[2,147],52:[2,147],57:[2,147],60:[2,147],69:[2,147],70:[2,147],71:[2,147],72:[2,147],74:[2,147],76:[2,147],77:[2,147],81:[2,147],88:[2,147],89:[2,147],90:[2,147],95:[2,147],97:[2,147],106:[2,147],108:[2,147],109:[2,147],110:[2,147],114:[2,147],122:[2,147],130:[2,147],132:[2,147],133:[2,147],136:[2,147],137:[2,147],138:[2,147],139:[2,147],140:[2,147],141:[2,147]},{1:[2,124],6:[2,124],28:[2,124],29:[2,124],52:[2,124],57:[2,124],60:[2,124],69:[2,124],70:[2,124],71:[2,124],72:[2,124],74:[2,124],76:[2,124],77:[2,124],81:[2,124],88:[2,124],89:[2,124],90:[2,124],95:[2,124],97:[2,124],106:[2,124],108:[2,124],109:[2,124],110:[2,124],114:[2,124],122:[2,124],130:[2,124],132:[2,124],133:[2,124],136:[2,124],137:[2,124],138:[2,124],139:[2,124],140:[2,124],141:[2,124]},{6:[2,131],28:[2,131],29:[2,131],57:[2,131],90:[2,131],95:[2,131]},{6:[2,56],28:[2,56],29:[2,56],56:327,57:[1,234]},{6:[2,132],28:[2,132],29:[2,132],57:[2,132],90:[2,132],95:[2,132]},{1:[2,173],6:[2,173],28:[2,173],29:[2,173],52:[2,173],57:[2,173],60:[2,173],76:[2,173],81:[2,173],90:[2,173],95:[2,173],97:[2,173],106:[2,173],107:86,108:[2,173],109:[2,173],110:[2,173],113:87,114:[2,173],115:71,122:[1,328],130:[2,173],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,175],6:[2,175],28:[2,175],29:[2,175],52:[2,175],57:[2,175],60:[2,175],76:[2,175],81:[2,175],90:[2,175],95:[2,175],97:[2,175],106:[2,175],107:86,108:[2,175],109:[1,329],110:[2,175],113:87,114:[2,175],115:71,122:[2,175],130:[2,175],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,174],6:[2,174],28:[2,174],29:[2,174],52:[2,174],57:[2,174],60:[2,174],76:[2,174],81:[2,174],90:[2,174],95:[2,174],97:[2,174],106:[2,174],107:86,108:[2,174],109:[2,174],110:[2,174],113:87,114:[2,174],115:71,122:[2,174],130:[2,174],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{6:[2,98],28:[2,98],29:[2,98],57:[2,98],81:[2,98]},{6:[2,56],28:[2,56],29:[2,56],56:330,57:[1,244]},{29:[1,331],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{6:[1,255],28:[1,256],29:[1,332]},{29:[1,333]},{1:[2,181],6:[2,181],28:[2,181],29:[2,181],52:[2,181],57:[2,181],60:[2,181],76:[2,181],81:[2,181],90:[2,181],95:[2,181],97:[2,181],106:[2,181],108:[2,181],109:[2,181],110:[2,181],114:[2,181],122:[2,181],130:[2,181],132:[2,181],133:[2,181],136:[2,181],137:[2,181],138:[2,181],139:[2,181],140:[2,181],141:[2,181]},{29:[2,185],125:[2,185],127:[2,185]},{28:[2,137],57:[2,137],107:86,108:[1,67],110:[1,68],113:87,114:[1,70],115:71,130:[1,85],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{6:[1,278],28:[1,279],29:[1,334]},{7:335,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{7:336,8:118,9:19,10:20,11:[1,21],12:22,13:[1,48],15:6,16:7,17:8,18:9,19:10,20:11,21:12,22:13,23:14,24:15,25:16,26:17,27:18,30:64,31:[1,75],32:51,33:[1,73],34:[1,74],35:24,36:[1,52],37:[1,53],38:[1,54],39:[1,55],40:[1,56],41:[1,57],42:23,47:65,48:[1,46],49:[1,47],50:[1,29],53:30,54:[1,62],55:[1,63],61:49,62:50,64:36,66:25,67:26,68:27,79:[1,72],82:[1,43],86:[1,28],87:[1,45],92:[1,60],93:[1,61],94:[1,59],100:[1,38],104:[1,44],105:[1,58],107:39,108:[1,67],110:[1,68],111:40,112:[1,69],113:41,114:[1,70],115:71,123:[1,42],128:37,129:[1,66],131:[1,31],132:[1,32],133:[1,33],134:[1,34],135:[1,35]},{6:[1,289],28:[1,290],29:[1,337]},{6:[2,44],28:[2,44],29:[2,44],57:[2,44],81:[2,44]},{6:[2,62],28:[2,62],29:[2,62],52:[2,62],57:[2,62]},{1:[2,179],6:[2,179],28:[2,179],29:[2,179],52:[2,179],57:[2,179],60:[2,179],76:[2,179],81:[2,179],90:[2,179],95:[2,179],97:[2,179],106:[2,179],108:[2,179],109:[2,179],110:[2,179],114:[2,179],122:[2,179],130:[2,179],132:[2,179],133:[2,179],136:[2,179],137:[2,179],138:[2,179],139:[2,179],140:[2,179],141:[2,179]},{6:[2,133],28:[2,133],29:[2,133],57:[2,133],90:[2,133],95:[2,133]},{1:[2,176],6:[2,176],28:[2,176],29:[2,176],52:[2,176],57:[2,176],60:[2,176],76:[2,176],81:[2,176],90:[2,176],95:[2,176],97:[2,176],106:[2,176],107:86,108:[2,176],109:[2,176],110:[2,176],113:87,114:[2,176],115:71,122:[2,176],130:[2,176],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{1:[2,177],6:[2,177],28:[2,177],29:[2,177],52:[2,177],57:[2,177],60:[2,177],76:[2,177],81:[2,177],90:[2,177],95:[2,177],97:[2,177],106:[2,177],107:86,108:[2,177],109:[2,177],110:[2,177],113:87,114:[2,177],115:71,122:[2,177],130:[2,177],132:[1,79],133:[1,78],136:[1,77],137:[1,80],138:[1,81],139:[1,82],140:[1,83],141:[1,84]},{6:[2,99],28:[2,99],29:[2,99],57:[2,99],81:[2,99]}], -defaultActions: {62:[2,54],63:[2,55],93:[2,114],194:[2,93]}, -parseError: function parseError(str, hash) { - throw new Error(str); -}, -parse: function parse(input) { - var self = this, stack = [0], vstack = [null], lstack = [], table = this.table, yytext = "", yylineno = 0, yyleng = 0, recovering = 0, TERROR = 2, EOF = 1; - this.lexer.setInput(input); - this.lexer.yy = this.yy; - this.yy.lexer = this.lexer; - this.yy.parser = this; - if (typeof this.lexer.yylloc == "undefined") - this.lexer.yylloc = {}; - var yyloc = this.lexer.yylloc; - lstack.push(yyloc); - var ranges = this.lexer.options && this.lexer.options.ranges; - if (typeof this.yy.parseError === "function") - this.parseError = this.yy.parseError; - function popStack(n) { - stack.length = stack.length - 2 * n; - vstack.length = vstack.length - n; - lstack.length = lstack.length - n; - } - function lex() { - var token; - token = self.lexer.lex() || 1; - if (typeof token !== "number") { - token = self.symbols_[token] || token; - } - return token; - } - var symbol, preErrorSymbol, state, action, a, r, yyval = {}, p, len, newState, expected; - while (true) { - state = stack[stack.length - 1]; - if (this.defaultActions[state]) { - action = this.defaultActions[state]; - } else { - if (symbol === null || typeof symbol == "undefined") { - symbol = lex(); - } - action = table[state] && table[state][symbol]; - } - if (typeof action === "undefined" || !action.length || !action[0]) { - var errStr = ""; - if (!recovering) { - expected = []; - for (p in table[state]) - if (this.terminals_[p] && p > 2) { - expected.push("'" + this.terminals_[p] + "'"); - } - if (this.lexer.showPosition) { - errStr = "Parse error on line " + (yylineno + 1) + ":\n" + this.lexer.showPosition() + "\nExpecting " + expected.join(", ") + ", got '" + (this.terminals_[symbol] || symbol) + "'"; - } else { - errStr = "Parse error on line " + (yylineno + 1) + ": Unexpected " + (symbol == 1?"end of input":"'" + (this.terminals_[symbol] || symbol) + "'"); - } - this.parseError(errStr, {text: this.lexer.match, token: this.terminals_[symbol] || symbol, line: this.lexer.yylineno, loc: yyloc, expected: expected}); - } - } - if (action[0] instanceof Array && action.length > 1) { - throw new Error("Parse Error: multiple actions possible at state: " + state + ", token: " + symbol); - } - switch (action[0]) { - case 1: - stack.push(symbol); - vstack.push(this.lexer.yytext); - lstack.push(this.lexer.yylloc); - stack.push(action[1]); - symbol = null; - if (!preErrorSymbol) { - yyleng = this.lexer.yyleng; - yytext = this.lexer.yytext; - yylineno = this.lexer.yylineno; - yyloc = this.lexer.yylloc; - if (recovering > 0) - recovering--; - } else { - symbol = preErrorSymbol; - preErrorSymbol = null; - } - break; - case 2: - len = this.productions_[action[1]][1]; - yyval.$ = vstack[vstack.length - len]; - yyval._$ = {first_line: lstack[lstack.length - (len || 1)].first_line, last_line: lstack[lstack.length - 1].last_line, first_column: lstack[lstack.length - (len || 1)].first_column, last_column: lstack[lstack.length - 1].last_column}; - if (ranges) { - yyval._$.range = [lstack[lstack.length - (len || 1)].range[0], lstack[lstack.length - 1].range[1]]; - } - r = this.performAction.call(yyval, yytext, yyleng, yylineno, this.yy, action[1], vstack, lstack); - if (typeof r !== "undefined") { - return r; - } - if (len) { - stack = stack.slice(0, -1 * len * 2); - vstack = vstack.slice(0, -1 * len); - lstack = lstack.slice(0, -1 * len); - } - stack.push(this.productions_[action[1]][0]); - vstack.push(yyval.$); - lstack.push(yyval._$); - newState = table[stack[stack.length - 2]][stack[stack.length - 1]]; - stack.push(newState); - break; - case 3: - return true; - } - } - return true; -} -}; -undefined -function Parser () { this.yy = {}; }Parser.prototype = parser;parser.Parser = Parser; -return new Parser; -})(); -if (typeof require !== 'undefined' && typeof exports !== 'undefined') { -exports.parser = parser; -exports.Parser = parser.Parser; -exports.parse = function () { return parser.parse.apply(parser, arguments); }; -exports.main = function commonjsMain(args) { - if (!args[1]) { - console.log('Usage: '+args[0]+' FILE'); - process.exit(1); - } - var source = require('fs').readFileSync(require('path').normalize(args[1]), "utf8"); - return exports.parser.parse(source); -}; -if (typeof module !== 'undefined' && require.main === module) { - exports.main(process.argv.slice(1)); -} -} \ No newline at end of file diff --git a/node_modules/iced-coffee-script/lib/coffee-script/repl.js b/node_modules/iced-coffee-script/lib/coffee-script/repl.js deleted file mode 100644 index 4b63d5f..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/repl.js +++ /dev/null @@ -1,194 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var CoffeeScript, addHistory, addMultilineHandler, fs, iced, icedmod, merge, nodeREPL, path, prettyErrorMessage, replDefaults, vm, __iced_k, __iced_k_noop, _ref; - - __iced_k = __iced_k_noop = function() {}; - - fs = require('fs'); - - path = require('path'); - - vm = require('vm'); - - nodeREPL = require('repl'); - - CoffeeScript = require('./coffee-script'); - - _ref = require('./helpers'), merge = _ref.merge, prettyErrorMessage = _ref.prettyErrorMessage; - - icedmod = require('./iced'); - - iced = icedmod.runtime; - - replDefaults = { - prompt: 'iced> ', - historyFile: process.env.HOME ? path.join(process.env.HOME, '.iced_history') : void 0, - historyMaxInputSize: 10240, - "eval": function(input, context, filename, cb) { - var Assign, Block, Literal, Value, ast, err, js, ret, run, ___iced_passed_deferral, __iced_deferrals, __iced_k, _ref1, - _this = this; - __iced_k = __iced_k_noop; - ___iced_passed_deferral = iced.findDeferral(arguments); - input = input.replace(/\uFF00/g, '\n'); - input = input.replace(/^\(([\s\S]*)\n\)$/m, '$1'); - _ref1 = require('./nodes'), Block = _ref1.Block, Assign = _ref1.Assign, Value = _ref1.Value, Literal = _ref1.Literal; - context.iced = iced; - run = function(js) { - return vm.runInContext(js, context, filename); - }; - try { - ast = CoffeeScript.nodes(input, { - repl: true - }); - if (!ast.icedIsCpsPivot()) { - ast = new Block([new Assign(new Value(new Literal('_')), ast, '=')]); - } - js = ast.compile({ - bare: true, - locals: Object.keys(context) - }); - (function(__iced_k) { - if (ast.icedIsCpsPivot()) { - (function(__iced_k) { - __iced_deferrals = new iced.Deferrals(__iced_k, { - parent: ___iced_passed_deferral, - filename: "src/repl.coffee" - }); - context[icedmod["const"].k] = __iced_deferrals.defer({ - lineno: 39 - }); - ret = run(js); - __iced_deferrals._fulfill(); - })(__iced_k); - } else { - return __iced_k(ret = run(js)); - } - })(function() { - return cb(null, ret); - }); - } catch (_error) { - err = _error; - return cb(prettyErrorMessage(err, filename, input, true)); - } - } - }; - - addMultilineHandler = function(repl) { - var inputStream, multiline, nodeLineListener, outputStream, rli; - rli = repl.rli, inputStream = repl.inputStream, outputStream = repl.outputStream; - multiline = { - enabled: false, - initialPrompt: repl.prompt.replace(/^[^> ]*/, function(x) { - return x.replace(/./g, '-'); - }), - prompt: repl.prompt.replace(/^[^> ]*>?/, function(x) { - return x.replace(/./g, '.'); - }), - buffer: '' - }; - nodeLineListener = rli.listeners('line')[0]; - rli.removeListener('line', nodeLineListener); - rli.on('line', function(cmd) { - if (multiline.enabled) { - multiline.buffer += "" + cmd + "\n"; - rli.setPrompt(multiline.prompt); - rli.prompt(true); - } else { - nodeLineListener(cmd); - } - }); - return inputStream.on('keypress', function(char, key) { - if (!(key && key.ctrl && !key.meta && !key.shift && key.name === 'v')) { - return; - } - if (multiline.enabled) { - if (!multiline.buffer.match(/\n/)) { - multiline.enabled = !multiline.enabled; - rli.setPrompt(repl.prompt); - rli.prompt(true); - return; - } - if ((rli.line != null) && !rli.line.match(/^\s*$/)) { - return; - } - multiline.enabled = !multiline.enabled; - rli.line = ''; - rli.cursor = 0; - rli.output.cursorTo(0); - rli.output.clearLine(1); - multiline.buffer = multiline.buffer.replace(/\n/g, '\uFF00'); - rli.emit('line', multiline.buffer); - multiline.buffer = ''; - } else { - multiline.enabled = !multiline.enabled; - rli.setPrompt(multiline.initialPrompt); - rli.prompt(true); - } - }); - }; - - addHistory = function(repl, filename, maxSize) { - var buffer, fd, lastLine, readFd, size, stat; - lastLine = null; - try { - stat = fs.statSync(filename); - size = Math.min(maxSize, stat.size); - readFd = fs.openSync(filename, 'r'); - buffer = new Buffer(size); - fs.readSync(readFd, buffer, 0, size, stat.size - size); - repl.rli.history = buffer.toString().split('\n').reverse(); - if (stat.size > maxSize) { - repl.rli.history.pop(); - } - if (repl.rli.history[0] === '') { - repl.rli.history.shift(); - } - repl.rli.historyIndex = -1; - lastLine = repl.rli.history[0]; - } catch (_error) {} - fd = fs.openSync(filename, 'a'); - repl.rli.addListener('line', function(code) { - if (code && code.length && code !== '.history' && lastLine !== code) { - fs.write(fd, "" + code + "\n"); - return lastLine = code; - } - }); - repl.rli.on('exit', function() { - return fs.close(fd); - }); - return repl.commands['.history'] = { - help: 'Show command history', - action: function() { - repl.outputStream.write("" + (repl.rli.history.slice(0).reverse().join('\n')) + "\n"); - return repl.displayPrompt(); - } - }; - }; - - module.exports = { - start: function(opts) { - var build, major, minor, repl, _ref1; - if (opts == null) { - opts = {}; - } - _ref1 = process.versions.node.split('.').map(function(n) { - return parseInt(n); - }), major = _ref1[0], minor = _ref1[1], build = _ref1[2]; - if (major === 0 && minor < 8) { - console.warn("Node 0.8.0+ required for CoffeeScript REPL"); - process.exit(1); - } - opts = merge(replDefaults, opts); - repl = nodeREPL.start(opts); - repl.on('exit', function() { - return repl.outputStream.write('\n'); - }); - addMultilineHandler(repl); - if (opts.historyFile) { - addHistory(repl, opts.historyFile, opts.historyMaxInputSize); - } - return repl; - } - }; - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/rewriter.js b/node_modules/iced-coffee-script/lib/coffee-script/rewriter.js deleted file mode 100644 index b21814c..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/rewriter.js +++ /dev/null @@ -1,494 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var BALANCED_PAIRS, EXPRESSION_CLOSE, EXPRESSION_END, EXPRESSION_START, IMPLICIT_CALL, IMPLICIT_END, IMPLICIT_FUNC, IMPLICIT_UNSPACED_CALL, INVERSES, LINEBREAKS, SINGLE_CLOSERS, SINGLE_LINERS, generate, left, rite, _i, _len, _ref, - __indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; }, - __slice = [].slice; - - - - generate = function(tag, value) { - var tok; - tok = [tag, value]; - tok.generated = true; - return tok; - }; - - exports.Rewriter = (function() { - function Rewriter() {} - - Rewriter.prototype.rewrite = function(tokens) { - this.tokens = tokens; - this.removeLeadingNewlines(); - this.removeMidExpressionNewlines(); - this.closeOpenCalls(); - this.closeOpenIndexes(); - this.addImplicitIndentation(); - this.tagPostfixConditionals(); - this.addImplicitBracesAndParens(); - this.addLocationDataToGeneratedTokens(); - return this.tokens; - }; - - Rewriter.prototype.scanTokens = function(block) { - var i, token, tokens; - tokens = this.tokens; - i = 0; - while (token = tokens[i]) { - i += block.call(this, token, i, tokens); - } - return true; - }; - - Rewriter.prototype.detectEnd = function(i, condition, action) { - var levels, token, tokens, _ref, _ref1; - tokens = this.tokens; - levels = 0; - while (token = tokens[i]) { - if (levels === 0 && condition.call(this, token, i)) { - return action.call(this, token, i); - } - if (!token || levels < 0) { - return action.call(this, token, i - 1); - } - if (_ref = token[0], __indexOf.call(EXPRESSION_START, _ref) >= 0) { - levels += 1; - } else if (_ref1 = token[0], __indexOf.call(EXPRESSION_END, _ref1) >= 0) { - levels -= 1; - } - i += 1; - } - return i - 1; - }; - - Rewriter.prototype.removeLeadingNewlines = function() { - var i, tag, _i, _len, _ref; - _ref = this.tokens; - for (i = _i = 0, _len = _ref.length; _i < _len; i = ++_i) { - tag = _ref[i][0]; - if (tag !== 'TERMINATOR') { - break; - } - } - if (i) { - return this.tokens.splice(0, i); - } - }; - - Rewriter.prototype.removeMidExpressionNewlines = function() { - return this.scanTokens(function(token, i, tokens) { - var _ref; - if (!(token[0] === 'TERMINATOR' && (_ref = this.tag(i + 1), __indexOf.call(EXPRESSION_CLOSE, _ref) >= 0))) { - return 1; - } - tokens.splice(i, 1); - return 0; - }); - }; - - Rewriter.prototype.closeOpenCalls = function() { - var action, condition; - condition = function(token, i) { - var _ref; - return ((_ref = token[0]) === ')' || _ref === 'CALL_END') || token[0] === 'OUTDENT' && this.tag(i - 1) === ')'; - }; - action = function(token, i) { - return this.tokens[token[0] === 'OUTDENT' ? i - 1 : i][0] = 'CALL_END'; - }; - return this.scanTokens(function(token, i) { - if (token[0] === 'CALL_START') { - this.detectEnd(i + 1, condition, action); - } - return 1; - }); - }; - - Rewriter.prototype.closeOpenIndexes = function() { - var action, condition; - condition = function(token, i) { - var _ref; - return (_ref = token[0]) === ']' || _ref === 'INDEX_END'; - }; - action = function(token, i) { - return token[0] = 'INDEX_END'; - }; - return this.scanTokens(function(token, i) { - if (token[0] === 'INDEX_START') { - this.detectEnd(i + 1, condition, action); - } - return 1; - }); - }; - - Rewriter.prototype.matchTags = function() { - var fuzz, i, j, pattern, _i, _ref, _ref1; - i = arguments[0], pattern = 2 <= arguments.length ? __slice.call(arguments, 1) : []; - fuzz = 0; - for (j = _i = 0, _ref = pattern.length; 0 <= _ref ? _i < _ref : _i > _ref; j = 0 <= _ref ? ++_i : --_i) { - while (this.tag(i + j + fuzz) === 'HERECOMMENT') { - fuzz += 2; - } - if (pattern[j] == null) { - continue; - } - if (typeof pattern[j] === 'string') { - pattern[j] = [pattern[j]]; - } - if (_ref1 = this.tag(i + j + fuzz), __indexOf.call(pattern[j], _ref1) < 0) { - return false; - } - } - return true; - }; - - Rewriter.prototype.looksObjectish = function(j) { - return this.matchTags(j, '@', null, ':') || this.matchTags(j, null, ':'); - }; - - Rewriter.prototype.findTagsBackwards = function(i, tags) { - var backStack, _ref, _ref1, _ref2, _ref3, _ref4, _ref5; - backStack = []; - while (i >= 0 && (backStack.length || (_ref2 = this.tag(i), __indexOf.call(tags, _ref2) < 0) && ((_ref3 = this.tag(i), __indexOf.call(EXPRESSION_START, _ref3) < 0) || this.tokens[i].generated) && (_ref4 = this.tag(i), __indexOf.call(LINEBREAKS, _ref4) < 0))) { - if (_ref = this.tag(i), __indexOf.call(EXPRESSION_END, _ref) >= 0) { - backStack.push(this.tag(i)); - } - if ((_ref1 = this.tag(i), __indexOf.call(EXPRESSION_START, _ref1) >= 0) && backStack.length) { - backStack.pop(); - } - i -= 1; - } - return _ref5 = this.tag(i), __indexOf.call(tags, _ref5) >= 0; - }; - - Rewriter.prototype.addImplicitBracesAndParens = function() { - var stack; - stack = []; - return this.scanTokens(function(token, i, tokens) { - var endImplicitCall, endImplicitObject, forward, inImplicit, inImplicitCall, inImplicitControl, inImplicitObject, nextTag, offset, prevTag, ret, s, sameLine, stackIdx, stackTag, stackTop, startIdx, startImplicitCall, startImplicitObject, startsLine, tag, _ref, _ref1, _ref2, _ref3, _ref4, _ref5; - tag = token[0]; - prevTag = (i > 0 ? tokens[i - 1] : [])[0]; - nextTag = (i < tokens.length - 1 ? tokens[i + 1] : [])[0]; - stackTop = function() { - return stack[stack.length - 1]; - }; - startIdx = i; - forward = function(n) { - return i - startIdx + n; - }; - inImplicit = function() { - var _ref, _ref1; - return (_ref = stackTop()) != null ? (_ref1 = _ref[2]) != null ? _ref1.ours : void 0 : void 0; - }; - inImplicitCall = function() { - var _ref; - return inImplicit() && ((_ref = stackTop()) != null ? _ref[0] : void 0) === '('; - }; - inImplicitObject = function() { - var _ref; - return inImplicit() && ((_ref = stackTop()) != null ? _ref[0] : void 0) === '{'; - }; - inImplicitControl = function() { - var _ref; - return inImplicit && ((_ref = stackTop()) != null ? _ref[0] : void 0) === 'CONTROL'; - }; - startImplicitCall = function(j) { - var idx; - idx = j != null ? j : i; - stack.push([ - '(', idx, { - ours: true - } - ]); - tokens.splice(idx, 0, generate('CALL_START', '(')); - if (j == null) { - return i += 1; - } - }; - endImplicitCall = function() { - stack.pop(); - tokens.splice(i, 0, generate('CALL_END', ')')); - return i += 1; - }; - startImplicitObject = function(j, startsLine) { - var idx; - if (startsLine == null) { - startsLine = true; - } - idx = j != null ? j : i; - stack.push([ - '{', idx, { - sameLine: true, - startsLine: startsLine, - ours: true - } - ]); - tokens.splice(idx, 0, generate('{', generate(new String('{')))); - if (j == null) { - return i += 1; - } - }; - endImplicitObject = function(j) { - j = j != null ? j : i; - stack.pop(); - tokens.splice(j, 0, generate('}', '}')); - return i += 1; - }; - if (inImplicitCall() && (tag === 'IF' || tag === 'TRY' || tag === 'FINALLY' || tag === 'CATCH' || tag === 'CLASS' || tag === 'SWITCH')) { - stack.push([ - 'CONTROL', i, { - ours: true - } - ]); - return forward(1); - } - if (tag === 'INDENT' && inImplicit()) { - if (prevTag !== '=>' && prevTag !== '->' && prevTag !== '[' && prevTag !== '(' && prevTag !== ',' && prevTag !== '{' && prevTag !== 'TRY' && prevTag !== 'ELSE' && prevTag !== '=') { - while (inImplicitCall()) { - endImplicitCall(); - } - } - if (inImplicitControl()) { - stack.pop(); - } - stack.push([tag, i]); - return forward(1); - } - if (__indexOf.call(EXPRESSION_START, tag) >= 0) { - stack.push([tag, i]); - return forward(1); - } - if (__indexOf.call(EXPRESSION_END, tag) >= 0) { - while (inImplicit()) { - if (inImplicitCall()) { - endImplicitCall(); - } else if (inImplicitObject()) { - endImplicitObject(); - } else { - stack.pop(); - } - } - stack.pop(); - } - if ((__indexOf.call(IMPLICIT_FUNC, tag) >= 0 && token.spaced && !token.stringEnd || tag === '?' && i > 0 && !tokens[i - 1].spaced) && (__indexOf.call(IMPLICIT_CALL, nextTag) >= 0 || __indexOf.call(IMPLICIT_UNSPACED_CALL, nextTag) >= 0 && !((_ref = tokens[i + 1]) != null ? _ref.spaced : void 0) && !((_ref1 = tokens[i + 1]) != null ? _ref1.newLine : void 0))) { - if (tag === '?') { - tag = token[0] = 'FUNC_EXIST'; - } - startImplicitCall(i + 1); - ret = forward(2); - return ret; - } - if (__indexOf.call(IMPLICIT_FUNC, tag) >= 0 && this.matchTags(i + 1, 'INDENT', null, ':') && !this.findTagsBackwards(i, ['CLASS', 'EXTENDS', 'IF', 'CATCH', 'SWITCH', 'LEADING_WHEN', 'FOR', 'WHILE', 'UNTIL'])) { - startImplicitCall(i + 1); - stack.push(['INDENT', i + 2]); - return forward(3); - } - if (tag === ':') { - if (this.tag(i - 2) === '@') { - s = i - 2; - } else { - s = i - 1; - } - while (this.tag(s - 2) === 'HERECOMMENT') { - s -= 2; - } - startsLine = s === 0 || (_ref2 = this.tag(s - 1), __indexOf.call(LINEBREAKS, _ref2) >= 0) || tokens[s - 1].newLine; - if (stackTop()) { - _ref3 = stackTop(), stackTag = _ref3[0], stackIdx = _ref3[1]; - if ((stackTag === '{' || stackTag === 'INDENT' && this.tag(stackIdx - 1) === '{') && (startsLine || this.tag(s - 1) === ',' || this.tag(s - 1) === '{')) { - return forward(1); - } - } - startImplicitObject(s, !!startsLine); - return forward(2); - } - if (prevTag === 'OUTDENT' && inImplicitCall() && (tag === '.' || tag === '?.' || tag === '::' || tag === '?::')) { - endImplicitCall(); - return forward(1); - } - if (inImplicitObject() && __indexOf.call(LINEBREAKS, tag) >= 0) { - stackTop()[2].sameLine = false; - } - if (__indexOf.call(IMPLICIT_END, tag) >= 0) { - while (inImplicit()) { - _ref4 = stackTop(), stackTag = _ref4[0], stackIdx = _ref4[1], (_ref5 = _ref4[2], sameLine = _ref5.sameLine, startsLine = _ref5.startsLine); - if (inImplicitCall() && prevTag !== ',') { - endImplicitCall(); - } else if (inImplicitObject() && sameLine && !startsLine) { - endImplicitObject(); - } else if (inImplicitObject() && tag === 'TERMINATOR' && prevTag !== ',' && !(startsLine && this.looksObjectish(i + 1))) { - endImplicitObject(); - } else { - break; - } - } - } - if (tag === ',' && !this.looksObjectish(i + 1) && inImplicitObject() && (nextTag !== 'TERMINATOR' || !this.looksObjectish(i + 2))) { - offset = nextTag === 'OUTDENT' ? 1 : 0; - while (inImplicitObject()) { - endImplicitObject(i + offset); - } - } - return forward(1); - }); - }; - - Rewriter.prototype.addLocationDataToGeneratedTokens = function() { - return this.scanTokens(function(token, i, tokens) { - var column, line, nextLocation, prevLocation, _ref, _ref1; - if (token[2]) { - return 1; - } - if (!(token.generated || token.explicit)) { - return 1; - } - if (token[0] === '{' && (nextLocation = (_ref = tokens[i + 1]) != null ? _ref[2] : void 0)) { - line = nextLocation.first_line, column = nextLocation.first_column; - } else if (prevLocation = (_ref1 = tokens[i - 1]) != null ? _ref1[2] : void 0) { - line = prevLocation.last_line, column = prevLocation.last_column; - } else { - line = column = 0; - } - token[2] = { - first_line: line, - first_column: column, - last_line: line, - last_column: column - }; - return 1; - }); - }; - - Rewriter.prototype.addImplicitIndentation = function() { - var action, condition, indent, outdent, starter; - starter = indent = outdent = null; - condition = function(token, i) { - var _ref, _ref1; - return token[1] !== ';' && (_ref = token[0], __indexOf.call(SINGLE_CLOSERS, _ref) >= 0) && !(token[0] === 'ELSE' && starter !== 'THEN') && !(((_ref1 = token[0]) === 'CATCH' || _ref1 === 'FINALLY') && (starter === '->' || starter === '=>')); - }; - action = function(token, i) { - return this.tokens.splice((this.tag(i - 1) === ',' ? i - 1 : i), 0, outdent); - }; - return this.scanTokens(function(token, i, tokens) { - var j, tag, _i, _ref, _ref1; - tag = token[0]; - if (tag === 'TERMINATOR' && this.tag(i + 1) === 'THEN') { - tokens.splice(i, 1); - return 0; - } - if (tag === 'ELSE' && this.tag(i - 1) !== 'OUTDENT') { - tokens.splice.apply(tokens, [i, 0].concat(__slice.call(this.indentation()))); - return 2; - } - if (tag === 'CATCH') { - for (j = _i = 1; _i <= 2; j = ++_i) { - if (!((_ref = this.tag(i + j)) === 'OUTDENT' || _ref === 'TERMINATOR' || _ref === 'FINALLY')) { - continue; - } - tokens.splice.apply(tokens, [i + j, 0].concat(__slice.call(this.indentation()))); - return 2 + j; - } - } - if (__indexOf.call(SINGLE_LINERS, tag) >= 0 && this.tag(i + 1) !== 'INDENT' && !(tag === 'ELSE' && this.tag(i + 1) === 'IF')) { - starter = tag; - _ref1 = this.indentation(true), indent = _ref1[0], outdent = _ref1[1]; - if (starter === 'THEN') { - indent.fromThen = true; - } - tokens.splice(i + 1, 0, indent); - this.detectEnd(i + 2, condition, action); - if (tag === 'THEN') { - tokens.splice(i, 1); - } - return 1; - } - return 1; - }); - }; - - Rewriter.prototype.tagPostfixConditionals = function() { - var action, condition, original; - original = null; - condition = function(token, i) { - var prevTag, tag; - tag = token[0]; - prevTag = this.tokens[i - 1][0]; - return tag === 'TERMINATOR' || (tag === 'INDENT' && __indexOf.call(SINGLE_LINERS, prevTag) < 0); - }; - action = function(token, i) { - if (token[0] !== 'INDENT' || (token.generated && !token.fromThen)) { - return original[0] = 'POST_' + original[0]; - } - }; - return this.scanTokens(function(token, i) { - if (token[0] !== 'IF') { - return 1; - } - original = token; - this.detectEnd(i + 1, condition, action); - return 1; - }); - }; - - Rewriter.prototype.indentation = function(implicit) { - var indent, outdent; - if (implicit == null) { - implicit = false; - } - indent = ['INDENT', 2]; - outdent = ['OUTDENT', 2]; - if (implicit) { - indent.generated = outdent.generated = true; - } - if (!implicit) { - indent.explicit = outdent.explicit = true; - } - return [indent, outdent]; - }; - - Rewriter.prototype.generate = generate; - - Rewriter.prototype.tag = function(i) { - var _ref; - return (_ref = this.tokens[i]) != null ? _ref[0] : void 0; - }; - - return Rewriter; - - })(); - - BALANCED_PAIRS = [['(', ')'], ['[', ']'], ['{', '}'], ['INDENT', 'OUTDENT'], ['CALL_START', 'CALL_END'], ['PARAM_START', 'PARAM_END'], ['INDEX_START', 'INDEX_END']]; - - exports.INVERSES = INVERSES = {}; - - EXPRESSION_START = []; - - EXPRESSION_END = []; - - for (_i = 0, _len = BALANCED_PAIRS.length; _i < _len; _i++) { - _ref = BALANCED_PAIRS[_i], left = _ref[0], rite = _ref[1]; - EXPRESSION_START.push(INVERSES[rite] = left); - EXPRESSION_END.push(INVERSES[left] = rite); - } - - EXPRESSION_CLOSE = ['CATCH', 'WHEN', 'ELSE', 'FINALLY'].concat(EXPRESSION_END); - - IMPLICIT_FUNC = ['IDENTIFIER', 'SUPER', ')', 'CALL_END', ']', 'INDEX_END', '@', 'THIS']; - - IMPLICIT_CALL = ['IDENTIFIER', 'NUMBER', 'STRING', 'JS', 'REGEX', 'NEW', 'PARAM_START', 'CLASS', 'IF', 'TRY', 'SWITCH', 'THIS', 'BOOL', 'NULL', 'UNDEFINED', 'UNARY', 'SUPER', 'THROW', '@', '->', '=>', '[', '(', '{', '--', '++']; - - IMPLICIT_UNSPACED_CALL = ['+', '-']; - - IMPLICIT_END = ['POST_IF', 'FOR', 'WHILE', 'UNTIL', 'WHEN', 'BY', 'LOOP', 'TERMINATOR']; - - SINGLE_LINERS = ['ELSE', '->', '=>', 'TRY', 'FINALLY', 'THEN']; - - SINGLE_CLOSERS = ['TERMINATOR', 'CATCH', 'FINALLY', 'ELSE', 'OUTDENT', 'LEADING_WHEN']; - - LINEBREAKS = ['TERMINATOR', 'INDENT', 'OUTDENT']; - - IMPLICIT_FUNC.push('DEFER'); - - IMPLICIT_CALL.push('DEFER'); - - IMPLICIT_END.push('AWAIT'); - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/scope.js b/node_modules/iced-coffee-script/lib/coffee-script/scope.js deleted file mode 100644 index 940f12c..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/scope.js +++ /dev/null @@ -1,150 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var Scope, extend, iced, last, _ref; - - - - _ref = require('./helpers'), extend = _ref.extend, last = _ref.last; - - iced = require('./iced'); - - exports.Scope = Scope = (function() { - Scope.root = null; - - function Scope(parent, expressions, method) { - this.parent = parent; - this.expressions = expressions; - this.method = method; - this.variables = [ - { - name: 'arguments', - type: 'arguments' - } - ]; - this.positions = {}; - if (!this.parent) { - Scope.root = this; - } - } - - Scope.prototype.add = function(name, type, immediate) { - if (this.shared && !immediate) { - return this.parent.add(name, type, immediate); - } - if (Object.prototype.hasOwnProperty.call(this.positions, name)) { - return this.variables[this.positions[name]].type = type; - } else { - return this.positions[name] = this.variables.push({ - name: name, - type: type - }) - 1; - } - }; - - Scope.prototype.namedMethod = function() { - var _ref1; - if (((_ref1 = this.method) != null ? _ref1.name : void 0) || !this.parent) { - return this.method; - } - return this.parent.namedMethod(); - }; - - Scope.prototype.find = function(name) { - if (this.check(name)) { - return true; - } - this.add(name, 'var'); - return false; - }; - - Scope.prototype.parameter = function(name) { - if (this.shared && this.parent.check(name, true)) { - return; - } - return this.add(name, 'param'); - }; - - Scope.prototype.check = function(name) { - var _ref1; - return !!(this.type(name) || ((_ref1 = this.parent) != null ? _ref1.check(name) : void 0)); - }; - - Scope.prototype.temporary = function(name, index) { - if (name.length > 1) { - return '_' + name + (index > 1 ? index - 1 : ''); - } else { - return '_' + (index + parseInt(name, 36)).toString(36).replace(/\d/g, 'a'); - } - }; - - Scope.prototype.type = function(name) { - var v, _i, _len, _ref1; - _ref1 = this.variables; - for (_i = 0, _len = _ref1.length; _i < _len; _i++) { - v = _ref1[_i]; - if (v.name === name) { - return v.type; - } - } - return null; - }; - - Scope.prototype.freeVariable = function(name, reserve) { - var index, temp; - if (reserve == null) { - reserve = true; - } - index = 0; - while (this.check((temp = this.temporary(name, index)))) { - index++; - } - if (reserve) { - this.add(temp, 'var', true); - } - return temp; - }; - - Scope.prototype.assign = function(name, value) { - this.add(name, { - value: value, - assigned: true - }, true); - return this.hasAssignments = true; - }; - - Scope.prototype.hasDeclarations = function() { - return !!this.declaredVariables().length; - }; - - Scope.prototype.declaredVariables = function() { - var realVars, tempVars, v, _i, _len, _ref1; - realVars = []; - tempVars = []; - _ref1 = this.variables; - for (_i = 0, _len = _ref1.length; _i < _len; _i++) { - v = _ref1[_i]; - if ((v.type === 'var') || (v.type === 'param' && v.name === iced["const"].k)) { - (v.name.charAt(0) === '_' ? tempVars : realVars).push(v.name); - } - } - return realVars.sort().concat(tempVars.sort()); - }; - - Scope.prototype.assignedVariables = function() { - var v, _i, _len, _ref1, _results; - _ref1 = this.variables; - _results = []; - for (_i = 0, _len = _ref1.length; _i < _len; _i++) { - v = _ref1[_i]; - if (v.type.assigned) { - _results.push("" + v.name + " = " + v.type.value); - } - } - return _results; - }; - - return Scope; - - })(); - -}).call(this); diff --git a/node_modules/iced-coffee-script/lib/coffee-script/sourcemap.js b/node_modules/iced-coffee-script/lib/coffee-script/sourcemap.js deleted file mode 100644 index 158c7d9..0000000 --- a/node_modules/iced-coffee-script/lib/coffee-script/sourcemap.js +++ /dev/null @@ -1,163 +0,0 @@ -// Generated by IcedCoffeeScript 1.6.3-g -(function() { - var LineMap, SourceMap; - - - - LineMap = (function() { - function LineMap(line) { - this.line = line; - this.columns = []; - } - - LineMap.prototype.add = function(column, _arg, options) { - var sourceColumn, sourceLine; - sourceLine = _arg[0], sourceColumn = _arg[1]; - if (options == null) { - options = {}; - } - if (this.columns[column] && options.noReplace) { - return; - } - return this.columns[column] = { - line: this.line, - column: column, - sourceLine: sourceLine, - sourceColumn: sourceColumn - }; - }; - - LineMap.prototype.sourceLocation = function(column) { - var mapping; - while (!((mapping = this.columns[column]) || (column <= 0))) { - column--; - } - return mapping && [mapping.sourceLine, mapping.sourceColumn]; - }; - - return LineMap; - - })(); - - SourceMap = (function() { - var BASE64_CHARS, VLQ_CONTINUATION_BIT, VLQ_SHIFT, VLQ_VALUE_MASK; - - function SourceMap() { - this.lines = []; - } - - SourceMap.prototype.add = function(sourceLocation, generatedLocation, options) { - var column, line, lineMap, _base; - if (options == null) { - options = {}; - } - line = generatedLocation[0], column = generatedLocation[1]; - lineMap = ((_base = this.lines)[line] || (_base[line] = new LineMap(line))); - return lineMap.add(column, sourceLocation, options); - }; - - SourceMap.prototype.sourceLocation = function(_arg) { - var column, line, lineMap; - line = _arg[0], column = _arg[1]; - while (!((lineMap = this.lines[line]) || (line <= 0))) { - line--; - } - return lineMap && lineMap.sourceLocation(column); - }; - - SourceMap.prototype.generate = function(options, code) { - var buffer, lastColumn, lastSourceColumn, lastSourceLine, lineMap, lineNumber, mapping, needComma, v3, writingline, _i, _j, _len, _len1, _ref, _ref1; - if (options == null) { - options = {}; - } - if (code == null) { - code = null; - } - writingline = 0; - lastColumn = 0; - lastSourceLine = 0; - lastSourceColumn = 0; - needComma = false; - buffer = ""; - _ref = this.lines; - for (lineNumber = _i = 0, _len = _ref.length; _i < _len; lineNumber = ++_i) { - lineMap = _ref[lineNumber]; - if (lineMap) { - _ref1 = lineMap.columns; - for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) { - mapping = _ref1[_j]; - if (!(mapping)) { - continue; - } - while (writingline < mapping.line) { - lastColumn = 0; - needComma = false; - buffer += ";"; - writingline++; - } - if (needComma) { - buffer += ","; - needComma = false; - } - buffer += this.encodeVlq(mapping.column - lastColumn); - lastColumn = mapping.column; - buffer += this.encodeVlq(0); - buffer += this.encodeVlq(mapping.sourceLine - lastSourceLine); - lastSourceLine = mapping.sourceLine; - buffer += this.encodeVlq(mapping.sourceColumn - lastSourceColumn); - lastSourceColumn = mapping.sourceColumn; - needComma = true; - } - } - } - v3 = { - version: 3, - file: options.generatedFile || '', - sourceRoot: options.sourceRoot || '', - sources: options.sourceFiles || [''], - names: [], - mappings: buffer - }; - if (options.inline) { - v3.sourcesContent = [code]; - } - return JSON.stringify(v3, null, 2); - }; - - VLQ_SHIFT = 5; - - VLQ_CONTINUATION_BIT = 1 << VLQ_SHIFT; - - VLQ_VALUE_MASK = VLQ_CONTINUATION_BIT - 1; - - SourceMap.prototype.encodeVlq = function(value) { - var answer, nextChunk, signBit, valueToEncode; - answer = ''; - signBit = value < 0 ? 1 : 0; - valueToEncode = (Math.abs(value) << 1) + signBit; - while (valueToEncode || !answer) { - nextChunk = valueToEncode & VLQ_VALUE_MASK; - valueToEncode = valueToEncode >> VLQ_SHIFT; - if (valueToEncode) { - nextChunk |= VLQ_CONTINUATION_BIT; - } - answer += this.encodeBase64(nextChunk); - } - return answer; - }; - - BASE64_CHARS = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'; - - SourceMap.prototype.encodeBase64 = function(value) { - return BASE64_CHARS[value] || (function() { - throw new Error("Cannot Base64 encode value: " + value); - })(); - }; - - return SourceMap; - - })(); - - exports.SourceMap = SourceMap; - -}).call(this); diff --git a/node_modules/iced-coffee-script/media/detail.png b/node_modules/iced-coffee-script/media/detail.png deleted file mode 100644 index fc9eefb352954a69227ccfc62ec1bc1ad38a5a2b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 42586 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%0D^uPkm;O$7Emz}Hkp!I4H|`7e*RKG{!i=Ni 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 diff --git a/node_modules/iced-coffee-script/media/post-rotate.png b/node_modules/iced-coffee-script/media/post-rotate.png deleted file mode 100644 index 7237741007b01562bce685703d563c2440515ba5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 104388 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 - - - - ApplicationVersion - - com.omnigroup.OmniGrafflePro - 138.33.0.157554 - - CreationDate - 2011-12-10 05:12:29 +0000 - Creator - Maxwell Krohn - GraphDocumentVersion - 8 - GuidesLocked - NO - GuidesVisible - YES - ImageCounter - 1 - LinksVisible - NO - MagnetsVisible - NO - MasterSheets - - ModificationDate - 2011-12-11 16:42:40 +0000 - Modifier - Maxwell Krohn - NotesVisible - NO - OriginVisible - NO - PageBreaks - YES - PrintInfo - - NSBottomMargin - - float - 41 - - NSHorizonalPagination - - int - 0 - - NSLeftMargin - - float - 18 - - NSOrientation - - int - 1 - - NSPaperSize - - coded - BAtzdHJlYW10eXBlZIHoA4QBQISEhAdOU1ZhbHVlAISECE5TT2JqZWN0AIWEASqEhAx7X05TU2l6ZT1mZn2WgRgDgWQChg== - - NSPrintReverseOrientation - - int - 0 - - NSRightMargin - - float - 18 - - NSTopMargin - - float - 18 - - - ReadOnly - NO - Sheets - - - ActiveLayerIndex - 4 - AutoAdjust - - BackgroundGraphic - - Bounds - {{0, 0}, {756, 553}} - Class - SolidGraphic - ID - 2 - Style - - shadow - - Draws - NO - - stroke - - Draws - NO - - - - CanvasOrigin - {0, 0} - ColumnAlign - 1 - ColumnSpacing - 36 - DisplayScale - 1 0/72 in = 1.0000 in - GraphicsList - - - Bounds - {{346, 248.625}, {9, 15}} - Class - ShapedGraphic - FitText - YES - Flow - Resize - ID - 89 - Layer - 0 - Shape - Rectangle - Style - - fill - - Draws - NO - - shadow - - Draws - NO - - stroke - - Draws - NO - - - Text - - Pad - 0 - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 HelveticaNeue;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\b\fs24 \cf0 A} - VerticalPad - 0 - - Wrap - NO - - - Bounds - {{346, 172.5}, {9, 15}} - Class - ShapedGraphic - FitText - YES - Flow - Resize - ID - 88 - Layer - 0 - Shape - Rectangle - Style - - fill - - Draws - NO - - shadow - - Draws - NO - - stroke - - Draws - NO - - - Text - - Pad - 0 - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 HelveticaNeue;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\b\fs24 \cf0 A} - VerticalPad - 0 - - Wrap - NO - - - Bounds - {{346, 89}, {9, 15}} - Class - ShapedGraphic - FitText - YES - Flow - Resize - ID - 87 - Layer - 0 - Shape - Rectangle - Style - - fill - - Draws - NO - - shadow - - Draws - NO - - stroke - - Draws - NO - - - Text - - Pad - 0 - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 HelveticaNeue;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\b\fs24 \cf0 A} - VerticalPad - 0 - - Wrap - NO - - - Bounds - {{346, 4}, {9, 15}} - Class - ShapedGraphic - FitText - YES - Flow - Resize - ID - 86 - Layer - 0 - Shape - Rectangle - Style - - fill - - Draws - NO - - shadow - - Draws - NO - - stroke - - Draws - NO - - - Text - - Pad - 0 - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 HelveticaNeue;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\b\fs24 \cf0 A} - VerticalPad - 0 - - Wrap - NO - - - Class - LineGraphic - Head - - ID - 103 - - ID - 136 - Layer - 1 - Points - - {419.83569, 37.414627} - {661.16437, 103.58539} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 133 - - - - Class - LineGraphic - Head - - ID - 97 - - ID - 135 - Layer - 1 - Points - - {351.1936, 37.437035} - {110.8064, 103.56297} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 133 - - - - Class - LineGraphic - Head - - ID - 104 - - ID - 134 - Layer - 1 - Points - - {385.5, 46.500011} - {385.5, 94.499992} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 133 - - - - Bounds - {{346, 10}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 133 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 131 - - ID - 132 - Layer - 1 - Points - - {551.99615, 300.60721} - {590.00385, 356.39279} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 107 - - - - Bounds - {{562.5, 356}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 131 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 127 - - ID - 130 - Layer - 1 - Points - - {622.07526, 389.92178} - {683.42474, 438.57822} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 131 - - - - Class - LineGraphic - Head - - ID - 125 - - ID - 129 - Layer - 1 - Points - - {581.92474, 389.92178} - {520.57532, 438.57819} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 131 - - - - Class - LineGraphic - Head - - ID - 126 - - ID - 128 - Layer - 1 - Points - - {601.77014, 392.49969} - {601.2298, 436.00031} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 131 - - - - Bounds - {{664, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 127 - Layer - 1 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f6()} - VerticalPad - 0 - - - - Bounds - {{561.5, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 126 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 break} - VerticalPad - 0 - - - - Bounds - {{461, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 125 - Layer - 1 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f5()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 119 - - ID - 124 - Layer - 1 - Points - - {385.5, 361.25} - {385.5, 392.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 122 - - - - Class - LineGraphic - Head - - ID - 122 - - ID - 123 - Layer - 1 - Points - - {385.5, 292.5} - {385.5, 324.25} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 108 - - - - Bounds - {{346, 324.75}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 122 - Layer - 1 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 120 - - ID - 121 - Layer - 1 - Points - - {224.00079, 301.19794} - {202.99921, 355.80206} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 106 - - - - Bounds - {{156.5, 356}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 120 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{346, 393}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 119 - Layer - 1 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f4()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 111 - - ID - 118 - Layer - 1 - Points - - {216.0753, 389.92178} - {277.42471, 438.57822} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 120 - - - - Class - LineGraphic - Head - - ID - 109 - - ID - 117 - Layer - 1 - Points - - {175.92476, 389.92175} - {114.57525, 438.57825} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 120 - - - - Class - LineGraphic - Head - - ID - 110 - - ID - 116 - Layer - 1 - Points - - {195.77019, 392.49969} - {195.22981, 436.00031} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 120 - - - - Class - LineGraphic - Head - - ID - 107 - - ID - 115 - Layer - 1 - Points - - {411.13647, 212.1042} - {514.36359, 268.89578} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 105 - - - - Class - LineGraphic - Head - - ID - 106 - - ID - 114 - Layer - 1 - Points - - {359.86353, 212.1042} - {256.63651, 268.89575} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 105 - - - - Class - LineGraphic - Head - - ID - 108 - - ID - 113 - Layer - 1 - Points - - {385.5, 216.50002} - {385.5, 255.49998} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 105 - - - - Class - LineGraphic - Head - - ID - 105 - - ID - 112 - Layer - 1 - Points - - {385.5, 131.50002} - {385.5, 179.49998} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 104 - - - - Bounds - {{258, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 111 - Layer - 1 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f3()} - VerticalPad - 0 - - - - Bounds - {{155.5, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 110 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 continue} - VerticalPad - 0 - - - - Bounds - {{55, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 109 - Layer - 1 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f2()} - VerticalPad - 0 - - - - Bounds - {{346, 256}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 108 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 await} - VerticalPad - 0 - - - - Bounds - {{500.5, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 107 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 if z} - VerticalPad - 0 - - - - Bounds - {{191.5, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 106 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 if y} - VerticalPad - 0 - - - - Bounds - {{346, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 105 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{346, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 104 - Layer - 1 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x2} - VerticalPad - 0 - - - - Class - Group - Graphics - - - Class - LineGraphic - Head - - ID - 100 - - ID - 99 - Points - - {694.5, 216.50002} - {694.5, 264.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 102 - - - - Bounds - {{655, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 100 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f7()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 102 - - ID - 101 - Points - - {695.28235, 131.49973} - {694.71759, 179.50027} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 103 - - - - Bounds - {{655, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 102 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{656, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 103 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x3} - VerticalPad - 0 - - - - ID - 98 - Layer - 1 - - - Class - Group - Graphics - - - Class - LineGraphic - Head - - ID - 94 - - ID - 93 - Points - - {75.5, 216.50002} - {75.5, 264.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 96 - - - - Bounds - {{36, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 94 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f1()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 96 - - ID - 95 - Points - - {76.282356, 131.49973} - {75.717644, 179.50027} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 97 - - - - Bounds - {{36, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 96 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{37, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 97 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x1} - VerticalPad - 0 - - - - ID - 92 - Layer - 1 - - - Class - LineGraphic - Head - - ID - 44 - - ID - 81 - Layer - 2 - Points - - {419.83569, 37.414627} - {661.16437, 103.58539} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 78 - - - - Class - LineGraphic - Head - - ID - 38 - - ID - 80 - Layer - 2 - Points - - {351.1936, 37.437035} - {110.8064, 103.56297} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 78 - - - - Class - LineGraphic - Head - - ID - 45 - - ID - 79 - Layer - 2 - Points - - {385.5, 46.500011} - {385.5, 94.499992} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 78 - - - - Bounds - {{346, 10}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 78 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 76 - - ID - 77 - Layer - 2 - Points - - {551.99615, 300.60721} - {590.00385, 356.39279} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 48 - - - - Bounds - {{562.5, 356}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 76 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 71 - - ID - 74 - Layer - 2 - Points - - {622.07526, 389.92178} - {683.42474, 438.57822} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 76 - - - - Class - LineGraphic - Head - - ID - 69 - - ID - 73 - Layer - 2 - Points - - {581.92474, 389.92178} - {520.57532, 438.57819} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 76 - - - - Class - LineGraphic - Head - - ID - 70 - - ID - 72 - Layer - 2 - Points - - {601.77014, 392.49969} - {601.2298, 436.00031} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 76 - - - - Bounds - {{664, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 71 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f6()} - VerticalPad - 0 - - - - Bounds - {{561.5, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 70 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 break} - VerticalPad - 0 - - - - Bounds - {{461, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 69 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f5()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 60 - - ID - 68 - Layer - 2 - Points - - {385.5, 361.25} - {385.5, 392.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 66 - - - - Class - LineGraphic - Head - - ID - 66 - - ID - 67 - Layer - 2 - Points - - {385.5, 292.5} - {385.5, 324.25} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 49 - - - - Bounds - {{346, 324.75}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 66 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 64 - - ID - 65 - Layer - 2 - Points - - {224.00079, 301.19794} - {202.99921, 355.80206} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 47 - - - - Bounds - {{156.5, 356}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 64 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{346, 393}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 60 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f4()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 52 - - ID - 59 - Layer - 2 - Points - - {216.0753, 389.92178} - {277.42471, 438.57822} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 64 - - - - Class - LineGraphic - Head - - ID - 50 - - ID - 58 - Layer - 2 - Points - - {175.92471, 389.92178} - {114.57523, 438.57825} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 64 - - - - Class - LineGraphic - Head - - ID - 51 - - ID - 62 - Layer - 2 - Points - - {195.77019, 392.49969} - {195.22981, 436.00031} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 64 - - - - Class - LineGraphic - Head - - ID - 48 - - ID - 56 - Layer - 2 - Points - - {411.13647, 212.1042} - {514.36359, 268.89578} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 46 - - - - Class - LineGraphic - Head - - ID - 47 - - ID - 55 - Layer - 2 - Points - - {359.86356, 212.10419} - {256.63647, 268.89578} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 46 - - - - Class - LineGraphic - Head - - ID - 49 - - ID - 54 - Layer - 2 - Points - - {385.5, 216.50002} - {385.5, 255.49998} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 46 - - - - Class - LineGraphic - Head - - ID - 46 - - ID - 53 - Layer - 2 - Points - - {385.5, 131.50002} - {385.5, 179.49998} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 45 - - - - Bounds - {{258, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 52 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f3()} - VerticalPad - 0 - - - - Bounds - {{155.5, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 51 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 continue} - VerticalPad - 0 - - - - Bounds - {{55, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 50 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f2()} - VerticalPad - 0 - - - - Bounds - {{346, 256}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 49 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 await} - VerticalPad - 0 - - - - Bounds - {{500.5, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 48 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 if z} - VerticalPad - 0 - - - - Bounds - {{191.5, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 47 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 if y} - VerticalPad - 0 - - - - Bounds - {{346, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 46 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{346, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 45 - Layer - 2 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x2} - VerticalPad - 0 - - - - Class - Group - Graphics - - - Class - LineGraphic - Head - - ID - 41 - - ID - 40 - Points - - {694.5, 216.50002} - {694.5, 264.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 43 - - - - Bounds - {{655, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 41 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f7()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 43 - - ID - 42 - Points - - {695.28235, 131.49973} - {694.71759, 179.50027} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 44 - - - - Bounds - {{655, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 43 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{656, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 44 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x3} - VerticalPad - 0 - - - - ID - 39 - Layer - 2 - - - Class - Group - Graphics - - - Class - LineGraphic - Head - - ID - 35 - - ID - 34 - Points - - {75.5, 216.50002} - {75.5, 264.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 37 - - - - Bounds - {{36, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 35 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f1()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 37 - - ID - 36 - Points - - {76.282356, 131.49973} - {75.717644, 179.50027} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 38 - - - - Bounds - {{36, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 37 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{37, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 38 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x1} - VerticalPad - 0 - - - - ID - 33 - Layer - 2 - - - Bounds - {{137.00021, 5.0000305}, {518.99976, 485.00015}} - Class - ShapedGraphic - ID - 91 - Layer - 3 - Shape - Bezier - ShapeData - - UnitPoints - - {-0.10693675, -0.5} - {-0.10693675, -0.5} - {-0.14161879, -0.071133643} - {-0.14161879, -0.071133643} - {-0.14161879, -0.071133643} - {-0.40944257, 0.05051589} - {-0.40944257, 0.05051589} - {-0.40944257, 0.05051589} - {-0.5, 0.31649494} - {-0.5, 0.31649494} - {-0.5, 0.31649494} - {-0.48458618, 0.5} - {-0.48458618, 0.49999976} - {-0.48458618, 0.49999976} - {-0.29961511, 0.5} - {-0.29961511, 0.5} - {-0.29961511, 0.5} - {-0.27649364, 0.26288617} - {-0.27649364, 0.26288617} - {-0.27649364, 0.26288617} - {-0.18015492, 0.067010641} - {-0.18015492, 0.067010641} - {-0.18015492, 0.067010641} - {-0.020232081, -0.029896706} - {-0.020232081, -0.029896706} - {-0.020232081, -0.029896706} - {0.17437387, 0.095876276} - {0.17437387, 0.095876276} - {0.17437387, 0.095876276} - {0.29768735, 0.2628867} - {0.29768735, 0.2628867} - {0.29768735, 0.2628867} - {0.30539495, 0.47938085} - {0.30539495, 0.47938085} - {0.30539495, 0.47938085} - {0.50000006, 0.47938085} - {0.50000006, 0.47938085} - {0.50000006, 0.47938085} - {0.50000006, 0.24639207} - {0.50000006, 0.24639207} - {0.50000006, 0.24639207} - {0.36127168, 0.029897034} - {0.36127168, 0.029897034} - {0.36127168, 0.029897034} - {0.10308123, -0.081443101} - {0.10308123, -0.081443101} - {0.10308123, -0.081443101} - {0.064546824, -0.5} - {0.064546824, -0.49999988} - {0.064546824, -0.50000018} - {-0.10693675, -0.5} - - - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 1 - - - - - - Bounds - {{326, 4}, {118, 301}} - Class - ShapedGraphic - ID - 84 - Layer - 4 - Shape - RoundRect - Style - - fill - - Color - - b - 1 - g - 0.8 - r - 0.4 - - MiddleColor - - b - 0.588235 - g - 0.917647 - r - 0.568627 - - TrippleBlend - YES - - - - - Bounds - {{38.000275, 67.999847}, {714.99963, 433.00006}} - Class - ShapedGraphic - ID - 90 - Layer - 5 - Shape - Bezier - ShapeData - - UnitPoints - - {-0.014685571, -0.49999997} - {-0.014685571, -0.50000036} - {-0.10559458, -0.45381027} - {-0.10559458, -0.45381027} - {-0.10559458, -0.45381027} - {-0.11398581, -0.16974851} - {-0.11398581, -0.16974851} - {-0.11398581, -0.16974851} - {-0.30699325, -0.045034766} - {-0.30699325, -0.045034766} - {-0.30699325, -0.045034766} - {-0.5, 0.42609698} - {-0.5, 0.42609698} - {-0.5, 0.42609698} - {-0.44965035, 0.47459596} - {-0.44965035, 0.47459596} - {-0.44965035, 0.47459596} - {0.0062935948, 0.4999997} - {0.0062935948, 0.50000006} - {0.0062935948, 0.50000006} - {0.45244771, 0.46535784} - {0.45244771, 0.46535784} - {0.45244771, 0.46535784} - {0.50000006, 0.40762061} - {0.50000006, 0.40762061} - {0.49999946, 0.40762061} - {0.30419588, -0.033487022} - {0.30419588, -0.033487022} - {0.30419588, -0.033487022} - {0.11398667, -0.17436829} - {0.11398667, -0.17436829} - {0.11398667, -0.17436808} - {0.069231153, -0.45842981} - {0.069231153, -0.45842981} - {0.069231153, -0.45842981} - {-0.014685869, -0.5000006} - - - Style - - fill - - Color - - b - 0.811765 - g - 0.435294 - r - 1 - - - - - - GridInfo - - HPages - 1 - KeepToScale - - Layers - - - Lock - NO - Name - A Labels - Print - YES - View - NO - - - Lock - NO - Name - Final - Print - YES - View - NO - - - Lock - NO - Name - Graph - Print - YES - View - YES - - - Lock - NO - Name - Pass 3 - Print - YES - View - NO - - - Lock - NO - Name - Pass 1 - Print - YES - View - NO - - - Lock - NO - Name - Pass 2 - Print - YES - View - NO - - - LayoutInfo - - Animate - NO - circoMinDist - 18 - circoSeparation - 0.0 - layoutEngine - dot - neatoSeparation - 0.0 - twopiSeparation - 0.0 - - Orientation - 2 - PrintOnePage - - RowAlign - 1 - RowSpacing - 36 - SheetTitle - Canvas 1 - UniqueID - 1 - VPages - 1 - - - ActiveLayerIndex - 0 - AutoAdjust - - BackgroundGraphic - - Bounds - {{0, 0}, {756, 553}} - Class - SolidGraphic - ID - 2 - Style - - shadow - - Draws - NO - - stroke - - Draws - NO - - - - CanvasOrigin - {0, 0} - ColumnAlign - 1 - ColumnSpacing - 36 - DisplayScale - 1 0/72 in = 1.0000 in - GraphicsList - - - Bounds - {{609.42883, 467.47443}, {53, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 91 - Layer - 0 - Line - - ID - 90 - Position - 0.41918012499809265 - RotationType - 0 - - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 0.8 - r - 1 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 71 - - ID - 90 - Layer - 0 - Points - - {604.43402, 468.48761} - {679.56836, 509.01151} - - Style - - stroke - - Color - - b - 0.4 - g - 0.8 - r - 1 - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - Width - 2 - - - Tail - - ID - 70 - - - - Bounds - {{247.13251, 456.6409}, {53, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 89 - Layer - 0 - Line - - ID - 88 - Position - 0.49565210938453674 - RotationType - 0 - - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 0.8 - r - 1 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 52 - - ID - 88 - Layer - 0 - Points - - {215.28778, 462.22324} - {333.00085, 487.27643} - - Style - - stroke - - Color - - b - 0.4 - g - 0.8 - r - 1 - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - Width - 2 - - - Tail - - ID - 51 - - - - Bounds - {{456.758, 264.49744}, {53, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 87 - Layer - 0 - Line - - ID - 86 - Position - 0.49565210938453674 - RotationType - 0 - - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 0.8 - r - 1 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 48 - - ID - 86 - Layer - 0 - Points - - {425.49759, 282.20352} - {542.03174, 282.79648} - - Style - - stroke - - Color - - b - 0.4 - g - 0.8 - r - 1 - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - Width - 2 - - - Tail - - ID - 49 - - - - Bounds - {{253.32349, 264.52615}, {53, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 85 - Layer - 0 - Line - - ID - 84 - Position - 0.45652154088020325 - RotationType - 0 - - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 0.8 - r - 1 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 49 - - ID - 84 - Layer - 0 - Points - - {224.65329, 282.80084} - {345.50232, 282.19916} - - Style - - stroke - - Color - - b - 0.4 - g - 0.8 - r - 1 - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - Width - 2 - - - Tail - - ID - 47 - - - - Bounds - {{500, 95}, {53, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 82 - Layer - 0 - Line - - ID - 81 - Position - 0.49565210938453674 - RotationType - 0 - - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 0.8 - r - 1 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 44 - - ID - 81 - Layer - 0 - Points - - {425.50003, 113} - {655.5, 113} - - Style - - stroke - - Color - - b - 0.4 - g - 0.8 - r - 1 - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - Width - 2 - - - Tail - - ID - 45 - - - - Class - LineGraphic - Head - - ID - 38 - - ID - 80 - Layer - 0 - Points - - {351.1936, 37.437035} - {110.8064, 103.56297} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 78 - - - - Class - LineGraphic - Head - - ID - 45 - - ID - 79 - Layer - 0 - Points - - {385.5, 46.500011} - {385.5, 94.499992} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 78 - - - - Bounds - {{346, 10}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 78 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 76 - - ID - 77 - Layer - 0 - Points - - {582.0293, 301.5} - {582.0293, 355.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 48 - - - - Bounds - {{542.5293, 356}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 76 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 69 - - ID - 73 - Layer - 0 - Points - - {561.51398, 389.80014} - {498.01974, 438.70108} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 76 - - - - Class - LineGraphic - Head - - ID - 70 - - ID - 72 - Layer - 0 - Points - - {581.21851, 392.49597} - {579.31134, 436.00406} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 76 - - - - Bounds - {{666, 505}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 71 - Layer - 0 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f6()} - VerticalPad - 0 - - - - Bounds - {{539, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 70 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 break} - VerticalPad - 0 - - - - Bounds - {{438, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 69 - Layer - 0 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f5()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 60 - - ID - 68 - Layer - 0 - Points - - {385.5, 361.25} - {385.5, 392.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 66 - - - - Class - LineGraphic - Head - - ID - 66 - - ID - 67 - Layer - 0 - Points - - {385.5, 300.5} - {385.5, 324.25} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 49 - - - - Bounds - {{346, 324.75}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 66 - Layer - 0 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 64 - - ID - 65 - Layer - 0 - Points - - {184.65109, 301.5} - {184.63783, 355.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 47 - - - - Bounds - {{145.15562, 356}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 64 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{346, 393}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 60 - Layer - 0 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f4()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 50 - - ID - 58 - Layer - 0 - Points - - {163.90605, 389.73416} - {99.246216, 438.76492} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 64 - - - - Class - LineGraphic - Head - - ID - 51 - - ID - 62 - Layer - 0 - Points - - {183.35385, 392.48959} - {180.28973, 436.01022} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 64 - - - - Class - LineGraphic - Head - - ID - 47 - - ID - 55 - Layer - 0 - Points - - {356.09601, 210.44415} - {214.05963, 270.55582} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 46 - - - - Class - LineGraphic - Head - - ID - 46 - - ID - 53 - Layer - 0 - Points - - {385.5, 131.50002} - {385.5, 179.49998} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 45 - - - - Bounds - {{329.78827, 477}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 52 - Layer - 0 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f3()} - VerticalPad - 0 - - - - Bounds - {{139.5, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 51 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 continue} - VerticalPad - 0 - - - - Bounds - {{39, 436.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 50 - Layer - 0 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f2()} - VerticalPad - 0 - - - - Bounds - {{346, 264}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 49 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 await} - VerticalPad - 0 - - - - Bounds - {{542.5293, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 48 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 if z} - VerticalPad - 0 - - - - Bounds - {{145.15562, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 47 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 if y} - VerticalPad - 0 - - - - Bounds - {{346, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 46 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{346, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 45 - Layer - 0 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x2} - VerticalPad - 0 - - - - Class - Group - Graphics - - - Class - LineGraphic - Head - - ID - 41 - - ID - 40 - Points - - {694.5, 216.50002} - {694.5, 264.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 43 - - - - Bounds - {{655, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 41 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f7()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 43 - - ID - 42 - Points - - {695.28235, 131.49973} - {694.71759, 179.50027} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 44 - - - - Bounds - {{655, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 43 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{656, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 44 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x3} - VerticalPad - 0 - - - - ID - 39 - Layer - 0 - - - Class - Group - Graphics - - - Class - LineGraphic - Head - - ID - 35 - - ID - 34 - Points - - {75.5, 216.50002} - {75.5, 264.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 37 - - - - Bounds - {{36, 265}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 35 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f1()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 37 - - ID - 36 - Points - - {76.282356, 131.49973} - {75.717644, 179.50027} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 38 - - - - Bounds - {{36, 180}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 37 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{37, 95}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 38 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 while x1} - VerticalPad - 0 - - - - ID - 33 - Layer - 0 - - - GridInfo - - HPages - 1 - KeepToScale - - Layers - - - Lock - NO - Name - Graph - Print - YES - View - YES - - - Lock - NO - Name - Layer 1 - Print - YES - View - YES - - - LayoutInfo - - Animate - NO - circoMinDist - 18 - circoSeparation - 0.0 - layoutEngine - dot - neatoSeparation - 0.0 - twopiSeparation - 0.0 - - Orientation - 2 - PrintOnePage - - RowAlign - 1 - RowSpacing - 36 - SheetTitle - Canvas 2 - UniqueID - 2 - VPages - 1 - - - ActiveLayerIndex - 0 - AutoAdjust - - BackgroundGraphic - - Bounds - {{0, 0}, {756, 553}} - Class - SolidGraphic - ID - 2 - Style - - shadow - - Draws - NO - - stroke - - Draws - NO - - - - CanvasOrigin - {0, 0} - ColumnAlign - 1 - ColumnSpacing - 36 - DisplayScale - 1 0/72 in = 1.0000 in - GraphicsList - - - Bounds - {{407.44391, 341.46844}, {53, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 91 - Line - - ID - 90 - Position - 0.41918012499809265 - RotationType - 0 - - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 0.8 - r - 1 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 71 - - ID - 90 - Points - - {402.44086, 342.48489} - {477.59482, 383.00098} - - Style - - stroke - - Color - - b - 0.4 - g - 0.8 - r - 1 - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - Width - 2 - - - Tail - - ID - 70 - - - - Bounds - {{254.75797, 138.49727}, {53, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 87 - Line - - ID - 86 - Position - 0.49565210938453674 - RotationType - 0 - - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 0.8 - r - 1 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 48 - - ID - 86 - Points - - {223.49757, 156.20346} - {340.03171, 156.79623} - - Style - - stroke - - Color - - b - 0.4 - g - 0.8 - r - 1 - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - Width - 2 - - - Tail - - ID - 49 - - - - Class - LineGraphic - Head - - ID - 76 - - ID - 77 - Points - - {380.0293, 175.50002} - {380.0293, 229.49998} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 48 - - - - Bounds - {{340.5293, 230}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 76 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 69 - - ID - 73 - Points - - {359.51303, 263.79996} - {296.01627, 312.70004} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 76 - - - - Class - LineGraphic - Head - - ID - 70 - - ID - 72 - Points - - {379.21838, 266.49597} - {377.31091, 310.00403} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 76 - - - - Bounds - {{464, 379}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 71 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f6()} - VerticalPad - 0 - - - - Bounds - {{337, 310.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 70 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 break} - VerticalPad - 0 - - - - Bounds - {{236, 310.5}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 69 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f5()} - VerticalPad - 0 - - - - Class - LineGraphic - Head - - ID - 60 - - ID - 68 - Points - - {183.5, 235.25002} - {183.5, 266.5} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 66 - - - - Class - LineGraphic - Head - - ID - 66 - - ID - 67 - Points - - {183.5, 174.50002} - {183.5, 198.24998} - - Style - - stroke - - HeadArrow - FilledArrow - LineType - 1 - TailArrow - 0 - - - Tail - - ID - 49 - - - - Bounds - {{144, 198.75}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 66 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 block} - VerticalPad - 0 - - - - Bounds - {{144, 267}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 60 - Shape - Circle - Style - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 f4()} - VerticalPad - 0 - - - - Bounds - {{144, 138}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 49 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 await} - VerticalPad - 0 - - - - Bounds - {{340.5293, 139}, {79, 36}} - Class - ShapedGraphic - FontInfo - - Font - AndaleMono - Size - 10 - - ID - 48 - Shape - Circle - Style - - fill - - Color - - b - 0.4 - g - 1 - r - 0.8 - - - - Text - - Text - {\rtf1\ansi\ansicpg1252\cocoartf1138\cocoasubrtf100 -{\fonttbl\f0\fnil\fcharset0 AndaleMono;} -{\colortbl;\red255\green255\blue255;} -\pard\tx560\tx1120\tx1680\tx2240\tx2800\tx3360\tx3920\tx4480\tx5040\tx5600\tx6160\tx6720\pardirnatural\qc - -\f0\fs20 \cf0 if z} - VerticalPad - 0 - - - - GridInfo - - HPages - 1 - KeepToScale - - Layers - - - Lock - NO - Name - Layer 1 - Print - YES - View - YES - - - LayoutInfo - - Animate - NO - circoMinDist - 18 - circoSeparation - 0.0 - layoutEngine - dot - neatoSeparation - 0.0 - twopiSeparation - 0.0 - - Orientation - 2 - PrintOnePage - - RowAlign - 1 - RowSpacing - 36 - SheetTitle - Canvas 3 - UniqueID - 3 - VPages - 1 - - - SmartAlignmentGuidesActive - YES - SmartDistanceGuidesActive - YES - UseEntirePage - - WindowInfo - - CurrentSheet - 2 - ExpandedCanvases - - - name - Canvas 1 - - - name - Canvas 2 - - - Frame - {{50, 89}, {1085, 789}} - ListView - - OutlineWidth - 142 - RightSidebar - - ShowRuler - - Sidebar - - SidebarWidth - 120 - VisibleRegion - {{-97, -40}, {950, 634}} - Zoom - 1 - ZoomValues - - - Canvas 1 - 1 - 2 - - - Canvas 2 - 1 - 1 - - - Canvas 3 - 1 - 1 - - - - saveQuickLookFiles - YES - - diff --git a/node_modules/iced-coffee-script/media/rotate1.png b/node_modules/iced-coffee-script/media/rotate1.png deleted file mode 100644 index e9560ac379170c0d0627d12052cfc7a98daa82f0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 67189 zcmce-byU<*_b*Jhgmjk@A}w8`fS{xxA>BxKH-dDjbSt6KlF~JR!~lYHBRO;Q@u@|W;t_zP1LSEi@eRLS>PHdTmuhtK6q@ghDE(Z1k+w=T6w0eSQ}M~1{) z5$y#svWHAn(A@g=sW{YcTH49sw8^iwzBcRh(CZ>&s;GHwR97Y2XY=JVIk_-aVq_x1 z|9D~12mRd3d&`}X#aCT!l3xO@!AAd zZR=J4^*F>t-|atlfS^5j4l$9)L;o~#_dn<%M2>L#^u4 zB^E-t59wI7ACT@|p=_2C6TDQ7pMIp?H|>61CDXX5`T&KA0cDYqp@J^!+fZtFAQlO4 z5L){TS}jLUYVEn=n=G9B#49XAoGX$K9oW~Z4^{ew=idhk(L9$0Okt1$r#bAnS^lNePOkqKRh46_6?xNnv+?;`|gN;p*t(Dam zTRXdwR&N6@uk$g3CU1e{#6*GQq$I(g6-uZzUy91g3@2x1F3B`~sGF&-@qIZmmS#HZ z^Y>SsQbWWTdk`#S=mz_{;OT_{d)d^g{`FGD(V<<6R0mvFppKpQ)nbKytC>z409V@!pGr5Xjcz1 z5IGHGj1f>5nGS*E9GWOar)zUZ4NRkMXk+bQzs9DYpO~nFuMdBsjuYFdOY6WFE4J;4 zXPjGTdo%eQ=4UoAM73L@-!P_IX%SQVdiZ0_cb7H6hp)bcpj4|X9k4G6n`J(F8mcr(QG7;qEt|zyvn^H%)m~ot3SsVboHfpRQwi- zxSd|v;7EzlR06%ywTY^M?cco=lDUq_xb!CWJBd|ny{^yK`kd@{M}PgA4JG!axW>+C zZz84Bs#Bl?X#va4-6@yH!w_#fj{FJt3{%o>cfMrd$niOO1{&f_eK3lT_{R&iNkO)#HK^mwO!)9}2DrC=yn+ z>S_iYiMQ;9D67AbE40By!Ey&r0Ye1yoB@4pHnc|tiq9j39Zh+2p!fR_xA*S^N&@u%~?I~XWXpEQ)cR!Be=tW^6ajvP|Cf>Rsw)OuL&LN zTU51mqD0@Slr=6+n8rz549n4%L}El7l^z_8s5qb%y;>ji=R6t3a(JtooFaH^s1jFJ zLk1Xu57@l^p}O=Xa8TGNycPB(8)Gr11io#Dp+a+>pH6cgzj-$irTMDXKC^bW`CyGaM|u2Q-m~|O)DDeK7wGKDvtDbNbB$hyYBOKgir$mzj7yY@-SOGDQ`-#h z(wsH+j@8lyEiRSfOryQ;`f>QMm$-6*ex^N$j~jvY&+iGp{swpRUU8@ zcUs!~$GWUEwj=p;h6kg5e4?jodxkPKXf{IlV}u*=)+MXFiG+u2E^`F%JUrm|C|u&1 z^}wvVuvH;x+J!lJMrDu|ICl(<2;VX$?#|wt)Xn#9vdbv0`W_p53-aC0lOL?vp2>R7 z5txs+t`;z*_5|%`CcYmG2+eQ#!FA^kny+z%^x zD{ihYe2jz$FAB8Rq6%n8yF$FJwvzGs-vMesK)c7pP>~fLy&46M92v+=Q*wbDdLQb2 zKg8v@HE1+rQXq2uRi%a3>zEU=huT9cZ4(j}muLHPs088*wEky{bcI%f5~mrbZJ6|) zr?Vc}9=lDa-k7wms%!ppey93|ordm{=5wE4d=g!=zsU-HaoH&M%AS0@(j7^0P-fHtX5>0==)@uWSL6&hUkg#ftAKCn|PhrmylV&1nyR=>FA} zA#}igu7%eLfcy8b>t>BEH5VcyBai!%xd%n@VsoDO>_f?$qzNRd`j%s;gNj{eh&~$0E;1amGr-;k-)#>~oZ+oS6?VNX!$L5zpZ}@oGX;(C54W4FhKxT?XT#_GSEr zF-Hz7bHwiuhI<%i97o#X&xbFcaR;7kjD$_*73b!*6_r}l?rq=UcK0qRDRD}+Yq4-I zFj+3HU7!!6U&bn0ur;2*2IMjU1E?G$@Cg=1-#)&4K(f_7Ae=K6$nn_r8j)?k^_DCsf3dfxR< znS7~$=H2=`nH(_?@MnpBIwRDIh=*|lr(Y8(N9fo1xJtJ|TK<_lQF7lBVJqP~TA^82 zb8VJCc5wCTCBzw`=Xa@A1o>lzd;@R@> zTjs2t`6+JNg;!mU`qsvb4+Qs;^=m%j!`jv#(K|Tq2lvJHQPENnrsgUKF=-NeqQeC| zFgA)ZE;sa^Za|4_rpbG5>9l~Ui3#D|i}@RW;-{KcD^VFQg!mY6d8{zfLl^dm8W~rW z=Gf0AsS9$Iuw+}^^8}XYnIKz@CoLl!PY#PTY zEi)F6;oV9&j6?ISuer;XA}1nYT%0VUB1#}#oUS@HT&wq; zJSy5N>o^}A)>^m>>mlDI+_{l4_$zrrW`>H=ufV{79ucmQjLCJd6S5DTC#vDN{CQ6p zk?~Jaj*SgIZleM z61SVDw9(^YyS5gVE>wng6H+|?%^de4uabI5Q#e#ex0Ah-eQbkd%~nY$!Jczxi_G*% zCU@$$vU({h#bUjQU*RiHSPhdRB2MEHk`c^5J zr`xH`l77e~Em;ah6RQR1)#}VBS+nP8sQ#*M^z(Z(T(!7THVd}VWMgA<+M%U-LF3%R zxc#0*=>ffzzwZwaOB%oN?MU9dT_kJDS4j58C40D@BH`l>UxG#{7#{=I;7x2kP}pWSkqR4>sOj!rr?(uPchqC6H!ylubeW97;%rhK?hxJ5EQ<_yzx$8jnc zTe3Hq8_RuNIAjSISFhcK(EnQ)e`m?5IY|z$+hb?N_)lg;cwjU*M+zG)s=}F$u~Eio;5* za`o8G#9J8!MuqW~lJ|3Ck8AlqAm6+fAyHmbmO^SK+(~BdJ#_nho3p8og18!0@9E}r zD}q(F*tih&o)>umM9Pe7Vx2UymT{ac6YLa|z4rPhY>n*g>^%cYRW4~>_0^@i&nxUN zjGnK33nJN}z;GsaP5si=^MLf~;?!ZX{;4U)xuBEQNELK16x++4j zwvihZ8OB}gUaw>-EE~ri%B5RVd<<-HSG&?e=|@)#bstagfTpY6m+pf&_8})a`gZox zEBgl(AH0C1+_gTdN2jZQR!>QXc*u&@I-Mqut#j8YH;?v>xpO%1`J%uM{Bc&cJ714LJHRl)br8<*Qz*|y=x>7S;kVGvy*hj(_zPMS=-CiPaY=y;Fvt=bQh2)*z7yBI$HzJPU1eNV`L`#9uSx%@^q+?_WgNoS={uO_=;9qZ4P@b(@}0VN^z_|e~gBR9+FHwuCRYHla%UNdn`OJ zBdJao_a#9519hRaafRs9VP$#(LyTI{8j7{v!`rk0vJTi5Rx0oM)l(Y5X&Y*5ULA5m;ap|lrHQ|d>;JG9_ZGi zvPsgpj9AuQ{e+!T2b0VT=UF?A+RH2LHCtg>uOT3$cz1QJ&$@GJS|1t?6*F{uuUx#= zWkq4UrMVA1x`Is?Smw@`9>Q6!90+%P*|l!NmKTVoS>#)ypKpU{4r5*K@&XBjc!6Va z%==m#jZoJXcMj!a=D2D56zMd}j8Zi4)*DH++rrsi9zK!IFn%Z4hc2&2kW66NN>Oy`Bd=Z4?6 z>1j{~VZ8Prwgs&WT4Xj78h#jc22$;qi~q0-*9^G`p)hqkw~mWU1IK6fI;Gd@+9M)^ zq)YpqBI7Q#m5tlfQDUKDhYAay7Fotb2GJV03wZ5Nrqzh21T#TxExz{HRxIF#O{+lG zhE1!PO&s5T$&c_L;7z50R8XH`)Za?OD-gW11gfSgutMkM4x(L~_H6wkNi5ho7mrK@|6LxOWu z)*%(BJDz08_)@qyyPuIzuFi#BRydQ4D2Y^QFFC&S`)&U0u471popJE2PM^s9D!MvX<(R929i80ZMmKf}_LiOc z^mYBAqBw?nsnua!vlK*B{gpFX#s#$V0%H5)0$Ly3ls=PHm4wKveU%|5S1x3|AaVyH zp@mJWc~V$Zj_Bk_#~K^u@$}wEo^+H7<*}0?tGp zg22Acd%f-+Tiw^gE3f(Ci({MDc=q>WMVW8;$AvgcgdDDH&KWjDbt||iFFxmUPG#a4 z8q^mH=eP|$-d7uik$(5(1k(=Ar6*iIzB~V2dTPbD-#)ZsoBo6c-0j=+i#NZd+P|~o z(#oRY#DLmDZej{d3+?AUq+!({eHwhBVIGq6bH_-GSo;c$FU5gaIf`EngG;v2UD1cd zG(Nn#Q~F1~uDbD5Zmzz6Px7}9)@{t#1LbmT83q(I2D}rfIbda2Zly{|CHfUQ7HaGz zBZn#?r06)ptUlpk8vncwTIJT?nulGyzmef?mH3)mi~NKI<;6Xac-q%S3xCm85FCwbS$n6zW1&Ls<$uV)P+8pJeXz`@-{LnMd0b2DC1^j|p zwFh=z^i&cm7Vy}IdZ)IY*$@BJpI;5!Qy>X|3hsBoqvjxA7tgs^c);FUay+9mKGA-X zr!yYef`706NyM>?S!94@Z&u4!T9jJN(}2gBv4*~CquXni31Al^7fq(lK5^@tDxuN?th(ypJ2G=!I;&Bpf5g4^wcb>fshUXG zg(OTMZ$1;@I^wB|@MsRd#+xDE4Ht4?Kv+K(C6HY_X}EIX8TLZZ56S#7Z%K@Etp(t6 z`RGzEsqz)k-mw6O^yop!Iw7QKk+AF%eX!F9^I&Wf#>G#y^lLG{80&WwP}t@kcwGWx zAm$ReQ^g67IQ}`OV>3x_*pja}p%8Oo1SAev$tgvqpet4m-FDcsD2+>K9Cx8k@lyAQ zPUf#f&;;or0<_QsjTwKGMj{Yz9ugpfh0*B_g>|fTu*24tg1U{z%$VYN!Lcc&ztBRl zgq`~R(amUlsCI5}!i8&|mxE-)8DxACLK$u!m{%r4A~5{Y1OgPU#9k#v?$6BJrxa%lt>p%1@Ashj^6f;j>QnG*vyAmuPvA2Wssa#bYbPLr{eDM_*8ex(~_ zh0s+31j7h31UdXsJdy6SYaE2Cf(vT`k)&hsu`eUqA5BHD#x7zF12j?OFr>}p(iGQ* z{_?2k>Uyb|i7&~Uv!X&o4i9QrzJ8> zUT*l!5IUJSU|TSEdygK{rZp*-yEurdJ3Uh9r%sLHV}-xH5inTJ$75 z>PkF-vf4<@qk7dWouTNL2s1Epc!KrO4MXaY2q6*{$&}YqR!diezt3$7BTn+n5+&(PNE<7q*D%p5!~gJegM2o zI8m#FOAzX^R6=&U)eMJ+@L?ipP!DlylNturiSHtRR~9$PuI~$t!Q`ItcaEr_lk9O$ zay-cxcwo+GbABdz0AKLmllhuN0Ev%)h@;ebu$W%IalGJypdaWi(nu_)6VmIkAoh_) zI*q$>wuzBW)E>~81?m6%Jb2SNP>#8sCph`DN`2C8|;$TKRwtdZ@RF#lbeyw=VOm%3~n9!}z*Mnuj; z{bxyv2gL}x^nk2ADZsD<{_39da*=%I*tQEO}sjgj|f!%5>q;7npEqD zrVFksb%K!7H6Dt--@f_~X(Z$Xd%?gIcj1#`Z1CgdQ#9o^i4&oK=K;VG{$vTf*utH_ zZm^o-kBL#eb=TPieuUeIk}pHW+#LqaoirpA%8NfSZ$N_2@H;#B0);jx`fBLbd$9!i z#Cvt;B4%BdggexkutITQl>fT^sUPlxJ zsMhr|=AW*1rN-TuOMXY2Vv~7~RIEHDpuKh>&=yp*yz9DN(*-6@dSS%}f&1~^)>lc@HY;K1lXO{SE-B7u0hIc}z#U25QEx}#wRvwn!puOSU2M&cmF zyFIFk(|6W7&rTjM%q(?r786ILkQwDZBkJKmo`EL{BfC?ZGgJ6s`ATMR?q2eYQ02)Q zQ%2fEXv^@GhH4hGve^>PlrmV(t@7XT{)iGMH2)w-qW|GaeG<4c3cpOFTswpZ`CIdM zqy=-!#S4QpZcU?Sde9970`8(8fSY_W5MEAchH7deHWZX_%!UQNiKY)RX0EZTW+gj^ z`*>X84)@tgb4Q(xa__>OaQ+jopf8BgG0E4cWp2JA2})WW6=gavPts0KP9Avc>gt*+ zC@7eNdL8=MVQ*hw?aAh^qVL7U#ow!{suE)DsLGY}{rs+aS640Ek3Ii|SI97#IrcoP z6MxxGJN{Jj(;^-Y$@-^-!H%mj7rj%P+knC+4+wt4Q_VrdW`Yt#IEyu;tk7ZOghNbt&G^C%(ijC(o4txpPrW4MIaJk03p#xSy zzVIQ<+53rT3YJV@potC|&Z}fLlvnoxDcZS-KCAMs5tRWPfg|+NMdsX3R5+jMF+@%)`Fp-F*eN1*!ILp@i3i3ky*TjX`W<=Rf6D zJ6)#8w#5} z@GozLE+!bVcU6`rvl8R(JSYuiEnRt8jEN(e0HF`;_0VbYziJLE<#+>{StBS;bCxr8 zPWL}OjX4;yLytOi5EgRS_?bV~9*QfL<+&8WIVg4GD+1fjtgo+caC9f=m|RQ$3#IV5 zKh4Ps79Bon*&Stu6EKQr}+Xy-wGLx_SYmvEp%r$Jd01n`jGm%+NIuBruT{knKziwIn-S&(6`Np+>rS* zHFoK1oe!o)hH9IA!(T8^S5#muFn`x{lJE1%ZQ)QfP`lG`l(o$v^w8!j^}@{^De~rq zKAaKycIw5gyJEfiUE+#PoOw>1a85WZ;5Ex=>Nkzd;A*Lm?fb72B8Wut^f~`*Sp~elr)OdDgJszUkm7dyD%3e#O5?%>Dy@`NzN(e1DOQ z_N*@ID4|)c#Y+`$?lWIu-NDIfbI#klR4iccefkmSm-j@YLAn}^v@}B3qf;aXQqT(y z-C4uXdHlkfoSmGSIx>G3#Q@dx;S{wfoxFkd(u{63>lS=*9LA4zjVl*Q z?l%z}paMk6fai;<>wf-0TTggNt+rO=PfxqLuBVskYH>EJ0nLpGXY*Jhdj*SSc$eMg zgqP-LWWo;jT$IZ~3^D0ZxlunCe@H#yDE*2W0Y}=AfIfN|PhvtveS3%WeHEh%Ll>%p_b$1BV>wSxXD`2{i(+jHQ@$DEB8yyWsh=OXTzwVYz+;2g=sox zX3?YHkU7`yaA=jvG)2Fy2)&Bthkfp@uH?RWp}VjXV|G57^C9uu zPk*R}Th4`O;Ds75C*jwI2&+V+u8G~+p)av+hTWa=(cfNSI&w>*3n?uY45Zvp`ikr6 zl-QIc)z&5L+w;IwIxlzMz%)%Uwx*i>F6Lh8>iUfKr}7uEDy9U!8U|P1HJ`N%7v+uu zrSz;>09XiRP{jn@Rt#aNXzKpk#Wyfp4nB>#gmZNS@`xb}jbx<$Z6-rOMG^fnrTPu4 z8MLfa#QF=CSrn29g&W1L+_llqxc>( z@9KHBvgk&h{?7NctT~D@n#zwh6!ddq0Q!Tw9 z88eS*SM*5~ce5w1WL0N2rRQNeIw-Wi@UBCCtB7{+7(@%V%}A>+{3B#c@d_xBHeW{Y zrSzp?z$Uzk zl$=TxR~~#F=2~PmQjpj>y_Yy9wMZf|*Vxnwv2b-xMDQjH@P;6Tqu>}RO^_&PhDcQp zjtuwhuC8im>%3MsT~tS)YMwqQ{x_;-WiRBLJywSQpN$wJ0bIYlQs`Hjn-31JZ(%jZ zTNpWHQINO5RLwE+Omr-St+|=vvBAkxbJo4i!N*3=5y%j;1jtryK}r&fDYWVlvb76Q4+Ao^ zy1%twOV8BCGj8BqWk6@_Jg|t3lM~9b`MyaJ*DX3{$+)K@WX%WQBs_h6#%+JRyHD38 zc&aRUZkC=nOk&7#?0;Nj==BDZ;6(B)kX!q7rU&Py!pq2m>ZCoFx@5P$!W?mz=wS-_ z++Agv?KgG-{P3bu3Qya`wnIQVnH99EYR}4)sD~n`bcxDMuEM4o0li?&a10J3!fu+s zxjwaW8qm^HP$=#$o#-^PB&Ucj^zVITnJL?r`eM4=_|1)Rj*M8QkVDoQKCX!O(d%6R zTR9T6|NUAvRVaFCY4<1)6@PDPdW0S&6QC(%wnjVavE8qU?^R@-GmX;8GSCI}bX76O z3b#vphzCR)<5j{11P?Hqv(imKvC&F_sD~%0&S@LqHn_Y?Y7n>^(C@AbTm05^vD-4= z5kF~_J0Pd0pjBMCSAwFds#dO*<94CIEEl^)6n#49GuW^c$st;6XJ>aj@!Y{3_(t#f z>yz3m&DZI*t{ZB*V)N&*9zBd<6UAKWt^)fA)KRS4j`1V1xDoxyfa|pYZvyeXz!n_c z^i^0Ue2uaA%7v4AZGm%xnuBEGPG^?q;hwDe&IAJg(_(sy|1pbz z@w)*q+iW8#ho)z???D6ta<~g1E~h;50c){m#M@1$b8zD=_W%hl3IXd6r-nDjhID^^ zc$AK7Se$H4&LuY=-H&7P1*oBuo8?XGt6jfcpSkRtOUstTtTp!;r*5}u@3`ay%1L8P zU-t2*p^RGfv$j2mn*pB^gm>HJqs>_H{sct@ytKhzvJz7ST15vLVPT4YjH2IJ9m!@@ z-Ez6YJy&kr(FlG4>yQt%B643!6DW2Ld-cnD#!J)!gX?sNOyt8L_M-sZM6$faYxkm zDW(nl&sVBm16GErP)9*nuw?{GB@TL+n+q;$11^Midzj-Ci!?IP-^JC9Ejx{wUgb<| z4RTR$)vBZJ;;v~4hvm9maz$4fBy2DSmG4GV%DEWVrn?#n_H<%HyLN2b`bwp`t~J|) zS|zgL{@_#3f-u1AJyciyP3RV!tRR--T1v;-F4x#w_$Zn^xk zS_O#XH#W$=2EV!}L93KWdcz?U*fI}FU+m^{jJ^s-_;o_K?TT9_YHY13R(sfLf2z8W z-Bi^PCIf3Oli*RNL$_nRXIYWCAlLhcKZ-e%sB{%?uhWp$9T5f65w(!glq4pe5-yZU zR-^rXHE1KcD(KF3(eP-dsXVt`I>8h-=g0LJwXJm$E^1QyYBFnIg7>NdmXc<9_S6TdP|r*d_+BbL7oI4 z_}n<6;G=xmhwnM=X9_Ar`D z^^JLb+TkXrTl3{^WE^V5jI=A@6+n5wWl#SAJmObYFhSYwhIMn1rcV zW^X`v9{l{E;5a(H=5CN4AqdZuD?HF@#p>ee!3VZP1(jl$=qydh*IY6i^gO<>dc=)# z?~}|60fUQpMrob4B?+E$W?^3>R_0{#^H;bHZf_V(uRR_j;u;;YO^}+R!n+C+9Z+#g z!}=W;xTF)g)j>A?lN8=~=}Az{(qDM?yZp^+w()vK7?0pNF^FwXMm7qw=jyG0J$M-V zc3;P9pz-8%KH$cx+>=JjUYehZx#pvK-TQ28cQ`jkEJjt2H-CGJT*M;pe@V$!UWW)6z zc;7j4o=H_AhnBbJk0j0w0~;GSd@tE4lVpWPEYpfMqeDQgPWSmfY=vd&N^;BlJBbW1 z2XkN!`lyeoQr&PUMtDSw~qV+uY9`QEGmTRON*CD#! zKj2a(6XnMDDO$e$f8eD$l2vri45`rmP*ZHLXy_5i3hVZR>?=wXU2L{svR=7%%*{R2 zDp?y79SAWMiOxFG3b_ls8PZ6CP?&S-YD|Mj)%5y4Xs>m|l)!x_M@hFx;3TVKAL+oy zv8z9_8%~8P8nGDA{27Spv<}53g&E z!RaVVI$(W>sPHFYcOocRC{9UQ=`EOzf`BVksULnl;j+$@NoMlyW%tuIWlIkVQQg5= zO?MlO!oWfo%JE!n%P8^85ufp~v7NB8oD-4X@zPVQN@*`8{V&hnz4-Lh%iMg%@A^&$ zr}yn}0>I_+`kZWyL8Hi%+c$w5IAVdtX7021Zr^|LKjBZ{yc^aUm;jRaJ2KED%|xek zGcA@GJ zf`XbyyYnq}WlblW6uc%fKdvu!cUy`hR}7StVs_UJMpD6MPbX*!cYM@8{=$e>T0OY2>VYZ z1Z*i?UMOfM(zSGVM}>0}iHSJKmq=9+e%VS(%}6qGph4uO_cw`dkFuF0gcqx^Or)w6 z>q@!fAmEJl-sR@ANlp?qz5D{h_Q9DKW^)o44dZ85i;TZ*hL`lCxY)%DTJF>re;6Kq zqRcIO<3P~fK+9cIldG#lJ!U7paxBYNKW8heaL$6|43~qAIoJ4KaCRKFDIMR%7h+#QSnigYdPk*P}i=;PW?UXVupJifpY&S=~5D6d>0pzQZ zf6&cV-y=XABe6b2^f8e9AQ}!sLBbG#FY&&NwBjXv4pOHjVz$6K7J%HO)y?9!w#8q? ze?g>Zn}#JgY{S`2@)c-xDGN-g)n#fxJty7qRJV_JVDy@(T$7u4hh<7B$EBHa@D1mT ztw`bI`FAfdEw0MOOrg-~@7QiOltTMlIHozalar5R#NvBur<>hpbejsC;Q8d3Ua*DZ6Ejv3S+F*(=0of7m@o(0u?aR z&Idu+F|_Xq@I}!Daf6>Eu0bB#pv=gKsKY*~T5hh|QLd$GTRwNPXFp^a(jSH`ALP~` z+gmVA!Uo9)wR@~e9ROy@?NHnS#t&w;>{16zt?j!Pn=efjKEDCl@kE&+k9XiRdjTbO zKH^GDjAfLiINRFNWMRFx?p|cPx zpp^W_jtrbAjNvw;&wpB1sP%u>uAMR^MIP}b`lZv%Zl5Y7^a5r@yp&Xb80?kaWs!NY zBu>SFWXMypEYt7HNZXg*TH^(p4sKkvR;x;R)4tPL$-RvrKh6oq7C!YVHBEb=p~!z1 zF^S0cgdxr;rv8}@r)H!VG2EW>xhf@FDf?RxbL^j1z|2{CW;Yg`(=DvKwzZLMQ~1*z zf~RBzREz;`glP@65bz%ShtBVvk z6+0Pm^D6fQw5mQUkMB&6jd_*H#6#pa^hGoWH z`l9w(5%f9lsELGuQj|J-9o0pYi>JKS2NzB8W!xg)&rWqcT%2tc^)pX#Z|CDkQLvX3 zAMg&~7p7R;`vAs+ekVdOmeQ)y)PvDk$1+{!8Pju|ZENX{#5o|kd~o~PTo&N75EGd1 zV$^L5<=tm#41@8(=oNw@c^1uQMRsAf4Li)!-0XAghQ;K&D;tRP{2#BJ9Bla_5$M{< zvAYGinBT`PcMr zj~E!fEyTJvMoZBPC>**{YgWfl1ATHn}WY!@U?a zjrpGpU_dcuWGEqH!;UycliYdLRLCRt ze^bd6F9alod<5PgJ^hcv^-u$F1d{jvCGrjFQn-45ynFX<*~z$LL9FA#uCMn?B`Qu*5UqQ!X`y=A`Jfe{ikAgR)*FY66$ z3%S+USgUP59p?<6PJ?RO%ijKl6%S!ulCt(=U`1J7CVub&P~pE1;@v*@bsg%xb>JZP ze0Ybd?23B_Zv`Q_973#xwi@)cbKx0(&fix2Qq+BdE&#*^q=)_#qQm((NLIFZuqD?w2@ezrKYr zl28Rv3LBqj6gsX+EB`%_0@%-gnDlZehGM7i-gUkk%q@97Zf(k5<5i&U15$=h;FZf!jVEy z$p|@o3&-$BA}y}ukZszR6*~aJVVZMf1D<1@JqQn4J?rS(RY!obMkM5JObe|>VnbMz z*;~k4le8+G9N{PIW5gsi*!EiS{0@hy7!r=(^_8dI4A7Y+IWza^_ImBaTV?D%Ykdzw zx(ovys2zL0e53-Ht_sQ;E2Y1mFpn*hh=$f-yAUt3z(fYbrO@~&B=MU6v%ieB_P_U+ zIa&_Z5|fi1wtp3Axx9My%EijcikWu$kx_d6pFe*R2L=WljH_!GixziVMLItn;|jFF z9?glvk{wyWbw?%xT(>(p0=W#)BBQ}I-yi2unTdqzdIZ|{h?8r`qXLiJIk}fwTJ+!C z;A0ZhR8;O%JVua_p2m2Oxj1|Aow{Xfe6g+a zZ;aA!5mzn?4iUsCiNKt0gL_G>#Q>GV@JCrYmo=E3_BS(Z^h-JI8p8G)N*X^RU)qc( znWo<{WyK5q_PG`jb+V*7}j|lhpga}9wv8vwmPO{8Zr1rw`t3qy2E?A7VdPNS`ClN;%`{luX zRzbMF_9^OzaQI;jyL!P(X3ozI7XM>XK69CV@QrSOhmD&@!;kevP1XG3_oP8~Xe8q# z`>O4MldZ4E);~)weX$%mq%g$fC;2BeSrOFSc7OkPFA;tcTzW&e zF<1PmvTCCT_4IdOg?0*a9Jo_B~fY*3FO=jI4PT?qJFEu|({Ze;# zw(i~FH!Vd}KW)a>)zO$4kVW-qwLGm3o5lCMYJC`0IYPyH-<_%Ru`a>97ML`95ocp* zcv87QGx=8BdLhx#`5?dxDK`p2;p%)zXs=Rnt=_Q{KoTY4IIHhCUi1 z&`vScdD_q>ER|or5&`;>kwgCneaYICB<)R)n8#Pfh#n3Ws6kbi;y!jNv60kyfMU#? zlQQNO5yD$wu2|fSGp~~k+KnXe|Iu!2w0SPu{Lj()5inJLvY*7M3+r6r3Ue8yz&U{p zoBkcH_hQMZOVUFIm96e)HW83!M6FZ{#W>1}k_g~dvTx_?{r_Xm{^x(q+0*MR4dQ_j zM}fDnB=?%)L#R3!-bCDjA48lF{X?6-{X-6*oL}DKD<~@HLb`FX(I&pa##~R<@(5bx zxepy$uH3QRTCu2ZyuKg(RJbs?zI$dQ9N?UDm`#&}J6xbIIT43&eX_v3*ioWNO6$h> z?WdljeFJTa63dW|yCZz}G5%-&Q1t1be3<8d_75L<1W)%Bm_23HLR6Ie5$(soh9w<> z94lg?o;$!?=^XK;6%~r86}O}W$RtDat3R9Owf9oTN}Nno()%5~z*k{|zimYn5wSEY zhvYVOil_cDlAKQ@tp@;hakc#kzqhGj*+tr5T1h>ryz&m{R#+Hbcbm z3~6ZWJs?ZUuc+V;%9q)2h&S3&R~2Xn>FRLGmYzpiZG^j#HEST@JjXe~Icm86*^|Jm z-gr5xb#p!#uv3A4^D&C+AT50oDpuil@Pl>kezM*Y^0(7B^p~yEbxx*hy+ZT8m8T^R zetj1wwJqCCK&vtjQ`>~s=R8_2^bz2h>4CwnPDBAKXcn@SE<|AXXyr2x%cz3s4XtSs zLfN8#1LVWfa{3FITBigo&vJ@t&}cT%iR-Lqrui;`e*gc?biaEgBL|}9je+PIt83`} zFUsCBppK|n*Tg++2u|>z!GpVn0Kp+ZaEIXT?ye!Yg#>qZ*Wk{^-Q68-?|kRXoS8d6 z?)*o(SFf(FRqtC*4e=`0Am&y66==&{3$VC7fx)$vMxCkYGl_ZFqNECQ(ycK!ME~K#EVFJ~`cRd*Cw-2ip za|53YJRR50P4?qt!)U2HihYhbt<4PQ1+x`+Jgv3zv}2*6M8?YwYuISR=hx%*TE`*U z#r(482;oXV>mO1G`%q?#^vC%<%s%jml8({corUZil+HCD%m06P00d@bSD&X%QUSTC zXu*f?pl+%MuT_Yt>U7>T>v$=c;Kq&=$W~q_51dh}SVT z&_;3~ijt51-}@PRYI1X~(Fw9$`qX2t19GyBlHS}b7W&-=UDBs_vy47yHJO1&I*w-l^i%00-v9N_yJQHi1z+-?%G+V{&()fFCK@TY1uT+sHrL@BrO0ta}3Fu z{6_lBjigMhDyP8)N)S8Tdl=_c^`eGE3RgfSgYaR1{S;xG_#rQMfNzzRk+{-2XmlFR zg#O|x4@g4yU8SnAN$EE1)rny6MXNUR(yR}B6Epe4wuHE8K#ZwxJQfhBFrrmpk)`%mpZ= zPwEjTCeQxHlW|*iweu6rdY=TR`|&u$4c1_VbYJ0$QRwi`54oaF@HWu`CgA$bSS3gXdq?VX_kMZu-RM3)E{hfw1HLS=Fx8bgAYi zl2IB&LqiH-%rnk0UiMbik6_7I=AmWTe{<$S{^iwo9eWN!nlgTQx=+TD-K+%>eEfsM zumbx-$RDPiMEo-o^F=kT^C5y*-_z+g$OF)ZnMKKMNTf9Ft^LQJ82|xs4ZAUfH81!5 ztlRxL$#>ag`SZW}h$BMX6=5L2ao{1p7V%3qKg-+w5XjCexC#3#wmB{r0NLz0zjMP* ztMSdq_egL!&@=5#4K6J&3FOTNZbJgNjd-ZihD12ZQVN)Lvh3uZe5cjq|g+jCJy{f6gp@^DA(<`}oRkFLD8x3@#OK-~9RKUxpV+uJWoAm2n%ouH3GUOeH5QHBQNvjm>JS9;Fw8{M-Egr2xNTQ>ZB|;| zK4DdnculIU9nE#kzovJkF>Yh8V;?$(JJjW+N!H`KIFoNYaDG5**BH#IfD{^e$S^!A za^KD2zxHifj6U1F-p~g(ATXt!{WN|-xg%QINZ8OoKlp=A4zrH|*s6a(Y=;*)Gm~!* zOZZ29EP{=d6VpjRiWwzrg&OitgMc`?@-LM#L#*EcmckI?alre zW&*&TqNdunyt~S>lmqeG*mU~C6IYv#S!=hr+R>GGPvyzlY#f0JvT~Mm-fnkCf;jW2 zr`5qbKuxvWQ3%DxW8+c1kW=6-aANg<7=a~F-52T0Tah#-Ztg|;~29XxecG?o% z1kI%W?MP}`5SmF~iZBSM11xZQCAyR*Vpfh05){K(J~%&CD7#+j8|S1KhB*&8q#i0P zX?~foUN%pjY_uc`@&J%?OnV=dr@-KE6YV~)b*(oG#fCkhm{fp2s{OUX`4yXRN#8oWuCvt{@LvOE~gW~Y-#BFJYPoxF6RWxrH0Cqq9Swprn+gv`+n>_ z?idgdS%^x5yauhNaV=zbyB7j^dmDI2zhmzJqDS25_40F}II%FW$>m|B`f7bTd2~=# z-@HJ1^6;bBS&?e(?w`r8wFkVyjwa_dAOQW9Nbmh94O92oILkByQCJykG#IB0GMuL1 z_>{w)c!Rj%xluE6(nkGvYeuRQ4(A^LnW@4dYq+^z@0PG#h(FMiQKQ9Tf-0#0s$L}N z=SNXUEVcQ`9?F ze0fXE8K*yrvUr7o4U`3E0OHK_f9JqZ_NMcReqEyL)OXB}%~_nTjJ3~;*qh`D{x0fl z#SaF@kk*Vicl3ZEDJf*3E5Ks!m-5dg#r zwx%$Lon@!x=X3n82mG0k^b`(mqlW`s%7f~u`K);z`=G`2H_;rv=~&Q@zg}l)Ywy-^ zSFz`y0Ct)_83s*WkZf@VtX$qyD8q$Dls}hY-;Ul18ct^qb-wmL&9BU}La`y0yB?7^ zf0-Y?Da^UOmS{?*)E=;ari0$dvE{Pbm|TMG<*F|oF8N*DTP}A3`j-Nm03z1`(D$EO2%t6jUXh}%<)m6CfU*jMWGFfIu@Kny7C#*O2lT!l+HVgHcj;hw;P0l`c z#T~~J-dW)pX~1fh4B3A=Nnv2f=gjk_BtHNv2}&yYR`S+%HBg2PdKfYN(e`&J2efrg zI^C{f0h`lhUF4-q$)ebl5{eVnUfK74%T?|#0V!cdeC@8lLp5m1;JPfkys)os1Z_NJbN9A^I|}l@1Z=q0R%!Y{Hl) z#DCP@Cd=TT zkV*j)s_wDr2#yydd{MrG#T{b+j7frJelsw(yhu-@zGI&<-}gNsU8?;{PXm1+N@Srz z{)9j8h$E%)l(O}KIm?iyMukRH7uR88HU&mzm<3FYr$VGdTbSce^WjJNPU}nB;S>;x z<0zYE3>s=GZV3|hxMMNKFu*9_nX<~1U9`_jtT*m_-WR7A23j-mk%eVWA}^b8Uu3y^ zAGk2v!MRq5%36y|a@3I)ta7h!>Oim*&At2~Phyhgy;U*JP{jnms=1M4R3(4T_zdxI zr-wR`a)os#OJwKy=BAM4RphSab>x9L)`=}ScJJ*|=tpIf7db^Lqh}!K0`_PnJsjVu z;l)PV1-QE0x?zY`iFHwzX|;R0T9$f^#+Uy#P z`37F^Gz$#AF^sh-Vb8I5IcavwCZyeQvSv!AFM4WbJv!d@n^p4;k=1tXGGqqGYW~tR z53)G~vnK(Gfm6!Ei59&Hq+;Lo32HsWZjSbg z5swm@uBaF4y+zzglyPJ(M0Ul%7;unra3VBIE;CqhOdhE=%PVsL=mxqK8jQa;OJF}~ z1pCU`$+E?bv;fa=@nf3JROdDztJZ>8E7SO3-Eb*J7+vM6W6f085*Ix4>SRq1zS{a2 zLL>qtQs}@mLp#-1(sWXxX0+SmhlAb%_L2ZeMz;drXuffUhwj@?lifui~VjUQ1@ zjVo^yPhW;Y0X0?HYGK5|>vj%C4k8$eeu1jo9^&Q*?-n^DeXl1JUkseJrDX^$ zUq~6Yh2)R=N8=22v?t`;PDXQd3#$dI?dV``t-s6J9v~P8iU)`#(U|ylCM$l%t})f7%DKKq#kPv^UK@Z zcTQ(QYdR)beJ!aQ3%hrn{-^J~@Lt&N*v_(YB4_`-N7zVyXgxKbc%$HCBGk< zoUtF3A%DNo+|cgz9vJx9easHAUwBUjN5(i$5ofTrr@FArNWVsl<*8NeGK}W7-GlTR z{uof}M&c($9eD<6yLs+FKBTj%Zn#cvb#lMdM1oG9DH z7T5sKF7yk(bgm55tn#m1CqNhzm4fQ@vPlQao=uz;uiaj@+y9BHpcPoMEw!{vL3^co z;<$@Zm@WtU^Z}F^v>$D6wNZrhvfh6|MMWi`M7&Jz9Mi$5th-4PrxEcAAnjj9_krK) zDiH0Gz1E+$ji3)Yh|kt;*{VFL7g{0W=T8pA~3X%V{1Y^xjyiMM8H07L)=Y123~34S-8R+RO_6H7ob8=q`r5l zq4e8Bzb(aE(O`~yziYYOsv&cB{KDInLSbkN54<)adkR^*;QrUsvvE3l@0=dDF107+r+mC{<%w>qegmi+h+i#S~cJ?NJKotL=&qJ5ED{qMNQ(&st;u zgG3VnvSGJH==mMmy(L@y`@ zvm;Ig&AK5o_8IlukQhTJHE1~nf#@UD zd63EoBS7KHTT9|i&bgCh4GL)wYr2)?7OJ5H-T(|gg}39VdA}HNGZK(K1wgS+;%n`X zoPFSya9x+MHllkAC6PV#orTU|bAT25h^RQj;2DL*6XBm_I|dGW%sG&`vh6+A2oO03 zK0?D8dJpVS-+kv6vUU`jp;l4{A?>Zt!hRp6UgIM8!PS-2$+Lm+SP#)|FNB59EhY92 z5UT&{x8Fj;t!J!HRX*Yww1EOAyxtzD!;~#ki$h)8;HoYmzFwUWoaz^;6KQp?JqhZ{p3c7x-}%6N}^tD}ggxmLgN z;T~stY^~gHe{7AT*K50p7g&M{EP#SrkrX>B4~LQr6^Q;0Zq#oT%7?_83U0#uPZc|o zFMYEWKax%8aURURDz^mhJvF{V3d0*1J$|IL>J<{$c3f|~hC1?p^{+Ufe{IR!9kE}W zq4H!>0g&Ix4B;?rX$(xC{I{sNbxFKf4f$g3qC4t97@laS#KN7rADp2NNBr(nkI11b zNP=Y5W7^?f$-yu3AeY<`V5#B(WI)?{N~eJNxUe1I9sMH$<#!{@-OfAMw*QK_v15yT z&F^R#*9Y|46BT`42F>=$p1%b(O0)91|;v>Mv<%;4$R^f+?^>^X{92EdbG zBxNQ5+&mMqaK{mJJ086NvJaf#TZ)iE20l3NnB`f%<~-kBesx@=Y^JeJ>sNR|tQY)) zhk!G3b?yY`f6j+RVC%1ostN~o_?b&rUi#Gc%NBd6=0abFV!V8-<3Mn+FG zW3vFgJxw2`VCwjc0L%yky*F)ePOrFkcpp7!kL4i#3SPik;*z8A8CWnJ7!2ChwO8gk zb>vleZ>Ydw4qK;gNq8gB9K30?^y`1bc!&UBB?XoKZf(jo64SPg>kRiKh_s8o+D+?N z6Zj*kzXy&o7JW%yb_%4=8S-1%>Biz{gB8!k63%hl;Hgc}uWUI}#&+F_Ej=MV^-gC<9o_W@WSpN zO7bjiUSysqFNO(!MOyRTpEkE9xs~6l)tIO+`Q34S9$T=q-!vMD)$v@-=IqY+Bm<X}c))J63I_n|XC!9}C0mfrLvFfVmka5wiJ^3@#?ld*(`E z2;1FunIBO&aHRjHu*07M{d1(iz}dDpXp9J6^wk!v$GwhgV)rx&lDuu81MD-)n8IsD zTzJv+UOZ2NcVp!2oWM;ALlMb3E~;X0P!G%crg87GUs{rrq5;&?;;Jh9tkl%hb+8KK zV0K<>Yisz*?5yhiIWsd8V!7PpG!`5joPmIbKhEyBH(m#HE0)gzf>Lqtn|C|eG*LmqEr~m7~vsv#7B9`Yr?@8Rv@@XU5 zy^+(_Ue@NW3|Ws%!#=Tm6HUK_qbx!#B{su@UVPEESx)e@dzVYeHDgYpkN{_J{#Aof zZvV4r$z_uwvbWz>6@Gy3dzKf(Y~1<6w%+RBgs%sRWh-%)=p%m4^agYwsBEQ@_Du}>~p?Y8N9t{^C6K!%A{FlG3~V)|vGw~hY!Ty*b$>p%a* z2lStd{PCzZ8M(-l&uSxhUgaA{@!`_>_WY)7Pno-UAUW;3jYLMgo_y7SV*Z^d*SjBk zZ0mEdq?${{5Ug+m9&k+wuALE1yPX8P7fdyM>H9{O6s^pk z4H{u6yaws=)0Hl&vz|-LfjYg^QSF(k`v1o;gJ5l*yRl4U=lV3XV#lrr9apuiWwy`6 zn+k<$?&DnNHBl%a!)sciyE(dtu?@S-MKBCIXM7Y?es&T_L6`HnS)B`4M z`0zSCprzTr&c8MxsSnYVzuL*UTXZMDa2bJ=`X4t$*GB_H)~%RsmffcQyxyykS-7o@ zbiUI#6mT6jwB)|SbUK``v1A^eehECE(s=v#)SLXOCea!$Xn-J2(O&v+=z{+HT4tRk?(oRb^% z-4j^g)Mk7bdE+PS6`_{pN|8=l;*oB1X~{P#-S@;BfKy7MK4_Ivzpc8vWLEa0qHW#% zPOlwZ2R8p4hzU=t=drKvOb6m=)}}#;9IZrhl{$;{HmiJq?iiEG<6v?fh>XQT{||5d z{~G!LckrK~59MP89@fPD6zVQ;3b%a*8O`gu{)^9AF;7lwe|6b6Qk6{`mkh9!9nx&h zBc>ATr|UMWjD`&oepcr$CAw*5+RI4qtZC5P!ogfe(yYxg7fYm%wan0H_RBWPRpFMC zIB|@XhP=2pxL7_L#iZO`|HtWuI)}1|GJ}#nHGKJt2Qa#o#e_ilGw{IpJ(ygF(TyCW z%m}5O#2AYiw&6XH#{f2VzvLaUnpD!F&I%Vt?}FfH%)7z+e4d?`5c3MAUI&4PXYbIq zj-fbRR9#|+8|XU3H29P?cG##9u^Wf==tExQAFpprBbqEu^$nPNXzN@D<0&zUX-e=M zEfh_c2*Oy$UL7Z;XpZfHhR&-#*KMhjN^Z68yR#Lb=XEhq_2xf){$=sksZLS;8HKv0 zimxSdhKVJ7FaByNx@!)!cC^KiTFWn+!3S+KHo`MV2lLom0Y&;_^ zHH2h;4Bc=bSeCB?s+hD1SLd;gU-&i_d9S0KqD-KBWmEEL`Je=y^D~?ucqd!W*-tN9aC z#55AKdcUk@b5(0dv2EqmDTJ6aRO&@LQ@g z6YgG+I0otW=5D%!<6ooOvbH+R04$OBRp{+uc5EQeb@0f-*VAMeMt68iiu|93K^te z3Svrpy4L`w^2NTNZTOYfI;ev=08dlKkoXrMu*0=zEJ9Pi+u`SH^f2RqE4H+ibIan7 zvE-Et1@Sw0>{SNPXd;498n8S+KD-Ksork5hok!VB{}#v0|N4y+=+Q&ZpDS& zfMjCYY;{a=(&wx|KDyi+AF2n1%pVbTi!HXKXJ3p8PCa+iR-aE? zG<7`Bdz(IT4oWxDwI7`OA9R;tzLV<5Igr#YWOG7<75cLU+8r^$YAr24>k76MTBbH+TfSS5j$I%=##0?{fS-AqF^U^S-p#$opJJFBbUw)nX+jwV>{nuOiSX}=}~ znu9#vm#Cx?U$o`LsM(8>8RJ) zf&hhRgFH_Cy7kYKv0O%klt*+-xYopD+sY-L^;+N?5p-AZ&Myx`uTF=RtNkg&hTg;7 z3q5#~E^B(aA;QkMHqtJ1g3z||G!M*36hbcz;NlPKvG`ESRDWxgQbg%apE(ZFNo8fX z$NQAXd5+Qu-T!UWNYfPhEzd0dW%+Br8^w{s*WD~+M1+!r8fMPR!Xx@glBi4@0~>Iv z6z(USu0fjB(+n2?Tb1#kC_i&6IOE}B#%hL`mQMNiP0rU1 zZ&90 z7I@Are3ic(o#>bY&TPa5o68Opz0ito`GaJy%)&%IXQ`%mP*$?Z%fYD?>zAp{Mt?N( zMBoc;3Y!{ya2-(5Fy!0qduJnK-Mw?6x0ROrWH#I(prPJu1u^Bfo-TTQ270u zRdIJ6(Z7~UIu;#(VcfeTS2sqMC}?kto{^ek9GUj3|s5- ze5h+mfPFet(+)&nx0X6t%ei)_V<5;oJ+{9-pGG?XqmWLVTlq8k9iAX_aJW_JK--9X z`U6Sat12Boyg{D(mt8No_3e98ysUkDQ-cxz0Q5@BC&Bi+UgkRqO-)NZYvvaM~>K=!mXa2^xKrJoB1p14(jXOv8Ru$x(;nnC7zN~9(nm?1JkQG5A<--YdUe+IxrOq};g=&D`NOi0ErL}b7>&0YznVp58x zKd|UjY`|kk_6V@qY7#@QX2`HWl4rB_fHdugEEm@qryQUUNiSsjuUe89bT$`lqZ} z)C$~sHqp)*cWB*G+gf|4WNXi{REAQKtxUie?riJ1UG7lKz5nM2f|Ta9)8HyS&JY%v zBAQ7JD&dwCSpr5yWvSm>8=L7UzxBa%;RZ*`dn~0$sD|&dY(4vW#EWVu3j9R=zmH{C zFvvT+LI{({0jp2jK3Dtw8{LhPWY4>9($TByF`2`$%k~;GW}_JuwUH|&@48tH3uAz3 zqSHV8;Jj#+^t>p1zdzc@AHOSKTIIdl$I=!`H!E$z? zeUH>+H3I(pdc)Z@k>4669^*NYe~py4@oC9{gwHOZ80Y<03N2JEM%`kK97}=lFD7t% z-^pYPUapDkE|{8+?AL%d|F7-|`7&Ade2rVzfy;yDBOI0SLIm@(2-*d6gBd1{u9by5 z9XJha}?Aosol0nWz%j(93E_=k>4Dt---r2h@x|>8B za*F!>KYF@odSDz7Tgmb6Dl^F+n!lr??ptvZ!G{+0D5{^Y-fHJLS%KirnWp{%8ajmd;%m!Kdri1oolyP5$Z+Py1bre87M!6?9eZIsM#PHFw!`@30ET{0lO@k{KPNVw!GB#uDEi2)(tIX44of*8RQC*E| z_&2rOv1^@PyB;hLHNEHioY-C(^%y)NwjA##BwWZE38Us{=rAN*+Bp5!DKZ}VZ?M#< zOtZpKpB602jTczSrxHyGn9azCiIBmT*E4EknRC)|xmyX!a({P*_5CHL`0bHFfxb8v zDnZ&!{C=a0#BNg{+pt!B3tCX{?@!q`5ZCtztlZ2Nv(LTs&GV^`PuSu41*tT_Hv@$s z7@gFCB~4?aek>s(e=~2P{%-v~2AiU%>X|(CBkA(SomBM)Y4(@>w`AI3b=j6vAK%Gt zFdL<5RKQg~Wu=KW(3wuE+@gmGmcoX6djQXYsHcyV zT~ul;eNnMy9w8~eTe|d&$tBo>O#Dlr=ay&97uNkH%kp+^k25Gy*a&Dq66-@%D=z7h z5UrN+07jw&zq3w$l}$&C&!R7Dpiow4E5#c#v_rPw`4V5`uER2QeLIaKu=rQ^9K+eV zB#-8>3+h~qybeK-bexH%bfG!ZP~``36cL!3BTs5rnNict$N)Hz^X0&|kt%vd3C`8* znQhC=g3gGTp%dE!cGD!$ARfKbl5WpaU^=Ue_#1+SmD|*hkynqr#S29rY$DJWD^DLacELq^?Zgc*R)LHTYa=nJ#i&q=^o2+b$Zm84w*yObISiZY#$6h-` zeD9VfO;{@b`yak;JF1|3Gx2G+a*~nAYat$q##wcECU}leZIg6=%(ys1@M}7tX24(cGdppy3$PaR>^Yd6HzwSOun@r;5 z!gBtPhH5U5xk{!9tEM5f_x^joInJbZd^f&t7v)jKfWmk`S; zbZD>7AWhQ)<^=+avc?_JIA?K=d<7TGTwMM05|+kqwEWEQzp#PoYBa(4Kh8bD)gOtP zPLSHI)6&9vs|b=zX2Cd|Bx+ixINkVRZSd$=g~fziA=w?3cF%74tOG47rfcPV6VVLw zr{1^CN5DTBL##JfTho1gz-!1QEoRj(my}RGez?AL&kg0R?v9n8rB~OTYi3OX>NeGL z`@I|Y?&`%sKHI6{1@umNCqR{IX*IA{rKq{yg#9x^(7vT}Yf4zt$RGWOnszSsWP z?E+ZxiF~IpcUgy8p_lWRc_5c;?5WybGq$-3%8U10L1f zLl~V0kGPP`kla|u3Ykik3ZC4GE5x_R`Vv#>PLec>5+cHGJP4xUkG_fC>G`)NdHX>( zxX%`oN-Z*9%m_ahymR_fR&*&=^hi~-^j^tUsuG&??1F)BDC`@;6l?X0rF7}nTLY=n z=BAS=uzFzI_9p48>x@fJxL_%`BoV25$~?YUQ7KX3=Ylw6!@}Y*I6sC0XI6w2T&?iK ziBIL#bzo6tFD<4V7Q$K_aXG=~b^PA9%2JYK(A>YBs99!{a&umC=PEhZ@plEkhwSIm zaZi(X9nn^px%x5QGY)15@fgo8QKG~2&x*Pqc&Ys{q`EWqq@HJ;s6LO2t1D$OGf!Xe zyrmveu_yP{6%rY0LM261)8tT|)2z^&yG~q_;F^DzzoY4WDQU6oV6}8ezhPaK@fvh@ zKQ)fMAtdcftdfD5u09h)Xtv4#kd~+^ifTUhq0SzSx=#MFhRoifkn_as(EN zRc@-ZLp~}yA7nT2m-8;?HEI9hT`7w=i$%m`@DhoF^%i;(*-#hk`IzI_%KPJ>676#~ zrBc0@hEhn8$8Y&VtgX(O2dG71``VY$61Kyy{MBvg8`R%dFMa@RkKpYZQoBY{qz5Rw zgpZI=6O9{4-u%?h2>SG#CGf7oo|7Xe~a)>E*Bf%_Q65=C-a#Q*~kj5iJF4 z%ff5;-h$J6%P5tO-P;dWrISgy;6<6jzc_*Hx@BaV7K15HGg-M(XeL+MPXT*Am{vaf z49%u9l~M_YVA^Mj>by)IZB3v3vSKN}+&Zv91n2RLkYn?4-MbnfZkmV%$3AzUlEwq? z-s4BFrDAQ;7xRg?qF!9aXGB&Zj|E@)4LlbS{w!MD+3o-4uRh6TiFKW9el_@};&L>^c34-qHwBfK^wvvb&KKu_l{b0KTw84Bcml{r7#R18S?CRNFO3lf7wzvi z$!W|H%AbJ_6@xWYI?)bN&;*%VD!LmBtk@=UUZ$6S{-SPzayO!t=~wpNzUoXDi>wO{ zfVFoIGM_E~kfl`4)#F5JLJ`KeuyduO@7O*Vli`+!MWk3EuGE6p%RbM1n~K1@DRS)Q zE3%a27iWb@#1kz#Op1$!%xwGGm;$qoVZf2Eulf;ro6T?~rCPt_hTHSI9L~Ipr5w+6s{N2ABWa3t%r$ ze%MaCO&{06hfpRo-eTDXF%_C51NMoiYNaTz(3TY z&GD|xAz0!Ed+7avy&d@(A558)cR`5ST*q;ykY7=8qFqRJ5-5!Y*M*=28 zq8_~p-U5xI-l86dE7{+EFfFErHC6Unxl186BlEhFQ>!&xx%51~1pN^OPXQTv6j>v08)I|668Bo!!bBf}a{XU2Hj9~Vz<=4Z8W!~mJ zUef)-g_1Pm%sS05DfH1|`D3;i@s7o43w9B4zDA8!$#Y?TCgC4~zvNB2E0YfoJiIR- z;k1q2M3}ZxdGBBBeKJ=~!?_gmGgU)*uRn^&OY!b!kh*eQM>ZW~eQ|G}&$E=D$j_C; zx&JLs5A)}c%-N(iOWNzXudp!Z{5~DTYQOY6|q!crk8x>)X_Ny zFSy3V2#A}nI4nEOy_u1z%1PA(zcEYFb~eK#e~bxTdm|dKtx{O(wtQ~cD^_K1Po){a z&fYq3A~StYKR;w}Cq2)cDcUim?kLHt-#<@9To)0skZJdKv8=rFf>1|KTZ8S)kYiIo zRa&XR#5~ovSkh#XsIu-FTsjBGKT*Lsb>D%$Xf+YTX@={l* zUH$}b8ZV4<2A;C8rm)6t>PtaT(qT6LE4vkG-UBoBL_Q;eL<}r&EX6P4{{@r;gS-R0 zX&N-5zuXGd_RTzq%@XvmK!G&SzM!$tT7;1wHVjE7;rJ~`Fi({hwPOVzS9GI?n+vXQ zl0OhR3_()nl7XpO(GF(gb-&k_6+^BqlKJ3%?v+=5MAJjCl;4A6^ZxktIptW^bGS*B3Bu~!0VgNqFzVh1n3H8BdCu_q4or$hY-jdy;=NX z^=7eqRyDTtm@ETHU+Am@7I9`U!t;Y>@2Be06eRaY)_x*bsVwqVS-gN`LkqBqxdoUc zRS^Fp)}T@lnJNs$&#r7!m=Q~E^PD%(`OH0>vVG)7fy7A|`gsWSe;t zNV~l3U8Zk$%{M_n$ilX+KcyLcx6VJmaYwwn1|Y|^JFKGtynlj+l6xIt^)p}!pi z!ULIevO?g$D>6nE9HB9$NHSn$Jde#zyA1%<-5O;wmvj%*U3eXV$U1tJVl2*;_?^)` z4>WBgLF(lgVUlgKAGBClv_7Ro@-T@ghUM5yba|1K{{8PX**9`33|chL%o)yKoQ-|G zoW~nnbp3m0d`E`p9%NMVi#&GJM_NPParg})dwKHzF*G6GfRmCSm!oqtz=%q!f)oAKeJ+D-)PHzd!C^TB z3ajHf!U&o!XPRuwH8JbEj7tTMX^sOsX*Okh4~fkuO%dhowHKX)&kkozk`Dq!n3Tjy zy}Skku3u)|Xlq)}$3xxou)A=o4`6=L(_uKr>qVTv%2p}o9D7et5+`wEJxre(_V9qT z%Jk>vE;*cN+XYYLb0s$ELZv=3X@r+}a0*Zz)7^7IGgI%R?;4JF&;5O*AH?`XjNOz7 zlwf;k9%`sq_ibc;jpe-BW?^fUjtX9>qF0S}tak=M_kV`+^QRTbQ9%0oWtLolHqatR zM^sy$xuf*YSb04uFgaeV_Gbz(jrz`Y0AN;3*gs#-fhNTUq}|=F^nZFcC3KGY5Q}xm z#ai`sN8lG>s@|P3^B|Jcs@HbHL&LErb6f2YOu0ud6`jHt3Etr zs2={@DXhm-lTIFX&VsmJFVwHqa9$ zoqNrztSK^@=oIT$t-lOTac}&cV)BmOeg1l-CYlrS)@y+x8z{B>>bzoA?;Q0nMs9cC zHh-o#1j|m&yV~l{B5{a()c>f+Rp`$jN3A_4#Vts3aicqd=FM8nguzh7g@t$YW+T8r znzm=D`lya=F>4CE!fj(*YPKuJQCwOz_~F=Hfv5$j5unclrO0Xxic>TrfHq>0_uQy? z%>upwCM=*$T9e9K2t4?xgzV;K-d~CRa@kp1AL}8)%Q*=VpG+=r@RqCd|Bf7{^W_(4 zi@_30%4+(>;GNY0YXr^%ai(0dd1MOI zm$9^y*|DCI*i3A{v1*LfJ^4Qg2ew2BhjqU3==7EuurdWF|7|)RsSEa6SUD>%@TA$J z|3M!C$D9vhsI>yX{Q-BATR_n*~U5>A;|Dc*`yu2O?&O;YX;r?Q3WPGc31S!rYkIN$q3g(OGmeg)g6eNT(@FzD8o z^kC)TVs#Tx*wj&?)@O3&aEriypYevaq`D3`~10xBN>tx0foZrC4t=42T@~ z%#uZtrLp&WX>J_Eih!Wz;c*SSY09hf(d4a3)zGWzkG^73{D4dE%E@Ov^?@c7i>{Lt zG6d2w_dH6JkML&xGV}7u?D&z|XajDpH(`cYcL%zRz^WMk>cn5dAKO4S`MK&=@4!_ts)1y;2bt7B{~Y{fL?>*mn!*RitLYEF0bpKUBJQ9r7r zIQzY2lImv@&|p(D@mXvZz~Ot|qiKpIDhX&36=Cl5MDD zew6Lwd%qj);vR0+6>V`}&Tqj-#ziw3#1atJ-GH9vxp>V=$|dFEDM9j+BpGGf`U4O^ zdZ|g5B891ELxz*^o)cZ>;;TouqVreDg!roH;Wy^@J zmgddxw}Ai_~IAgzFsLgX^smE!v44BnxE{RC7<_Y zVx3$8U$S$Z(pKT};op6BnM~V^6-r{foRNTErqff1Vx#$ncKMrf0~zsSRHPRnv7N1C z2IM%x*?PrWmhn-moW|`jJS*!Po9LT)7ydBxxtJHtva(fXHa$$|cgMUdc7n_<-JgBS z?3F5hKp#)B@3Y8lzJE4c0TqorPUTk~SsU*Eo}|eI;}CkK3ggpCJ-m;}=$yfoHEG!Z=&nrJ8@)uX_-Mu*O82IYl`@$c5cSTtHl@$c|`?6_qiX^_D&4cV1& zd}q{6FNu;zx3v;qx2~myV07BTJGc4SfhqIHuI*F{sa2|0op6j8M^l^q5vzCI2*>rU zjba_2E7-Iw`^{l=EG-IKdEV@DgD$Abmreo#k;6VnO-ek37mk3fiPVnlqo=@5M9g^A z@7~7Wm_``j##BG!H2-lTa@&%(YapKy$wqi!w;Xkr$UT-nW`uuVk0lPa<_;#^#$V(- zrF+i6eitPuONLfM`5e2*N6bg?hz|-VZWSQ!yW`2$dZVPod7jNIb{2Ig(#=(F^A!mW z4p~Xj5<5^f=WNmTOj)Nm3Uv-4L^|IzoD~?leZk@2L0%ex^*7HA>i#hMBN^{GOM~qs* zD@vN0%5F={#t6urZI*`4m#`0Z%^M3gj~|%0oH|J?EgI^Go9nomsrS$~<2h9>SeZIP zg#H&(Zy69}7j+FQk|HH29YcpQB0Y3>Dj?k;EvYmJNOuWH2s6?lEnU*(&@J6acYKHY zd7tt>AQ)1g84JP@F$m93Ml=UZT+BhwYs}u8zw*& z^~?60C9{3+14s;)0;#jp<7k2%cX_RxU`zR&#(E|GecYxFHvYp}vA=#k`MtaC#?|yU zvT^dIxlJ{P9#NXpH@nE%jWM}K2$i{*b98Eq41 zw`5vwCvHCx*wdwy&9}$5?!VpO=q>aS7PdvBe*T8TzfCH^Z=Iqh=X0YANblZ0O=hX%1D66Gr`H}Gm6$Lwo!?oMd z-_gUTvRft}tACF2sL#mnvYaHB^G4~8T$_kKx5H#JlI79%PE`~B9_4WNoO+Rv$6onPjaD{Fe`fmt)AqUmVqzl zU)*vZ@c&#u?WGU>w4|7@1EWf=6wXw}@tyv~QY$GFpDcG!o3AeY5q6qpTfa;*9~j0* z-Bl$5PAc9k6-!pbc7@jYb5A_-EW6ePx7cX~Sof4ey+XTim!mGTNcEfR+)+0shv3Mm z(~yeEPRut{sJvA;*VW{+OCPMAs|9jD>gbJ5Ji@((_>0o0=3>76uHr0`bQ+%}C8!=H zylcBEvGA+rSkcOnT9YCQ95@z&PeBtAX@b1`|c$1G2Z_T}#~;##=Vd|wwU`Khcj3yS$u^9eX6Z)Cf}(#oe@_XI7b`+_mTsQoAEGk!|@VqO>;p4vYemcM>4A%S!GlO+_@jo!*u zJjqn4@Z_4Diq?hnNljVHde$iC27T&}lk1&NJU+hXZFNj>iaB9?zrI&6*(DrlJGbx) zPjR`f{zccE$l^z84T|+ovwiSufs0-G#)k0DSrQ5VY&Y-{GdMvfBv$|T-gje~Br=(4 zOTw@3$aP!$)OFD?Tg)U-$WopmVQ?-N(fWsrPkr`p$HiCuT!lFk5o2Bo9|f{x95^pxCe&L}XAZa7@{8K@S=L?Vg%K^HliG=e z?_&G7pJtxTg@f!Q6Etg-k*P6U$HSY?4thK7xbPU4@ckIvar+p+q{-vo7%{5(@^G~76J z{{f^dw4^V5Feh$XVls9Bj(-*-n)QP~)%P+L^9oGrJX*Df?|(~W|3Ji!PeOxgV&U!Q z7u`2bj6x=RTktd!(Yo^SMB|Xzj*gbb)2y&k3$Yxw#jCKmYL;uO_WUX4C>l!ndWFVt zP`PfCJ=s&R&{CLihn44)VMa9gn0|ifabqj5bluSwV?>Rz2$*9IoZsS}=2!m}frivA zV%9|A^XZt+DS7V>y1rdwwx&_#M*Uo*g|k(wN=f&jtk)-2cK9=_M1rXC0RM0Nu9wo( z=er5qKyu7cY?LkCiU>_%V9Sj@MsNBlgizeq+A>YZM&0i4@|_nlpPF(>~$JqrpN?Qz1A=0w(khsTR+e^Ca4m|X-lpZtwpHDCYjUB}Ne z57uyR?$`V@&i4Oy-o?FW@Na3Mp*a$f(lU^-s2gfYpeZ549DR-=Hf~cWl+DE*=!YHw zE#T9&X*oB;dBzro6vd>6QzYK;bwinx%=4S#H4gl3+EDAC=2!wov@06g`~Wy5xgeR-Hr^p@}`M zeX_?QuRC;=y?8kX8z#*jR^~>TuB$r!Gwo|`36?vV<>8*GTR;nTN6Z#o)|I)beqz%{ z!jsk^&1O``-=bM=_dg(DN5GH|LFH1l&0f-D(U*7>6sm}m_$o{t4>N&g0};aKI6k(L z8G=jBV_S>>J?pvZ>T2%wq0HtbFz8dJvbq|Vl*6Q>EtXEEI{V(dHhxXDKV5(~9#?Z3 zvhf)W`wI?m|I)a4p*o$6jz@Lfu`^XcttlM68JG)Kkz7C26ly%v{6*(z@XZ?SQ&}JDyhP;^jTo-_WaaP% z2cNURb*@(VhBgM@1**#-)>XyMcDJYSvT78AtFJNLqAA&T;cC9dsAkTMykgF62IpSw zP_fvqU7#!U=Y=Pa>(=NFXvAC9gE3$0v#t~RskPJxVHQUYBkk=A8gYe)20a`OUiYqO z4@O~&Su|-`bNu1Pswk&5h?Db4m&)5U)>s=F)tUQc+&v9N;(%qOVB9XpurZo+f51+i zTttWZG%h2~8B@U60xekXBXB{Sg&eI7%-93fhKE93Ww(1Zm?~&Vy9?73ppDCz10HKr z#WnM(`oO8F;_h;y8F#|=wm8*aL>FOpvoFp9rwl4et6fI7!U&8h2XVLa-n_2nIAAN; zS(oifind)e>k9!0y!J>!DRL52u2*`cY?(P?i*f!eVtQ5M1RH^HPjv6R-e&)#YA@-% zTbtr4O}7pLnHSvdhbv-zFDev^wd{T#Hi7|0Q@~*pc^htcZ88WLgXsLtl6~gvF;{+- zQjZ;3GaT`d?5g6u!J>;k%b6K$qtH9J7r%@$P$k{(lkhZ)yfyBY{)>%qZ7O-1*o z*;X1ERr`}cll5>B6`^Yo&>AZwM{Sjg;g2Gh+r}bhnU76>Gv}dpbPNoci*D|TV#i__ zlnE?CR^-Zm7l1jX@o815w_S;Wfw!VSzn5I%bqx{HehJq~CQ&fyC1!BUYnm+2t)t#G zXJoON)})Rs%`g$ze}^!ODEU)>@{^{ORC-#|)HXA%OQx=g(m=Lioa0@`bD~GsUoIdI z0`AE;cz4ISB0}$G7h%mycIRoU+(;cT<}K|}OtwzbVJ}+>=GC*b=VT|w%|c{E`}E!O zH-Uzzp22>5U5T!p&G7oXQi~oF@QVAGuO0+}Y=c`cPUrP>v=kSAJ>u-ld*eep27n=Bt6Mm6I5tG4B8(MJC20Qi0;k`j`F&+{e zue&g~1^*IV3rq+9FHW>*e-W=2p$wYmXFu4@f&3ZVB^x~;MTnmX$;E?$W|A`aLxK;I zMLltO{05i@pT+`tp0R1baNjhrpnN$47{SCu7L&PKE(S=8FO&m@!eci<A8UYg7_BDNL)UAW4c3{w<%Q-$t#Gzc7>bl+x-Si6bE6*N zYYIL5*i7yq%Ykw>_tCI=%Hl%4Jiui~deduD7a5hT5>SR68RG-V@=S2G9D#fGW$bNe zS=|4?^8NKq=8(l~Foyp%o9*}0uLZ68#KhUYlG?rgCG0mdfSgAusW0lnm}XzcLjAV) zm;=F?V1bjL+)*}2E$IF{Z1oiiE&Ly5NLF)pg$VQ}vfs*O!8wfc5$F`sbbIX9s3nxO zVdmv5yh}Dp`l8(u!zhy_O0P5u`HFwCcz^btHC+riRPO-JyHb?u+1P}Z&eXIc{FQ)Q zzKk4X=kNPA;ab0nY;hF-CJ0U2EmOLC+A)TAA7t<`>eeIwcj#}ZcZm>1z`Xz`m*q-p z^dR$HYoFw}>t|7a$#UVj6({(0S|qi|?I8#)`-CP%9VTwlNs}#w+$mr+E9d|Q-C^X> zpUJPwYU2j1*-W|){oZ?pLI>Vw!-q1rLL2vpH06zA?J;F9U&WXEB!$fW8?(+TfvU_< z!~zpyUGvu3F-Qlz(ddmj3;Xt{jO?ip7~%CrBQDPuP|^qH0qKu{25GQ{AxO&bzgz&) ziVJYgpC$2)r-cBT)WpSM!u+?9%t|0b0V5Uf@(vfVFw8B|F(7x&5C;&>A5kTTr)KUd zkVh_eQM;bXjutek_4k1>R~;Isf#Kiin&Ml~o~f4J_giD@!9zm15C+yJey_8hc&6pj zuVser`Jrnn-dk@{k`~M=EL!C**)Ngr!8FH)Cd=Vq&Q?V{n85bi5E!2DZ4F9W2GpPn z98Q-3etT);F1MM8+*77ZK3@Dh4Ui*FSocZ(DhO(6tsCYat9z%HyGCHdGsn4&ui(-F z1Pt6hfI40LX|efc*TFLLyo0G63{a)jE%VdI6G+xF^ya=oCO0KeerS0(r6xm}2rL0@ zC`0d!Z7neMZRV{%#mswSs1*Az)}X1QHRL@E^|69cUXA6rBV@as zm|c(0E5-cpx5Q_%$M;+{#j9G&npr1>WMWhI{(FMAzRMq}DZrVICy=%W{Fj}3wjn#{ zN$&uk`=NP2h;&^!0*#gx`rAIkTc^&Fmx<2-*V||#S#iYtl@D`)3kU5Tqo`=bb_3yo z)n=&+6gq*b5yi?wfbv8d9E26LEwHZev7V1YK~qH=T$YVO6@&ey$*oVw!DvL=5PO7q zqK0M<^ykG|n&sJipFm1_9}dtRc>jGw-A(AVA3)uYN#}bxl39-Dd*1%5`FwrWnYc0a z=e)=EGwK3alKwzS`p0edf^XUkVz;G#421e!9{T33HOH3Qy$3<`>i*dO`XeIj81>;m z?G9A``>Sx6)cUuv_#;}S5s2slrsGhu?Gq}&^Uytux<3Em2vW{N+r7GN_9oP6^CX~` zwBvDQrN$a42MSG2x`;A6+N*ELaiF&=d_P&Z0`w~@M;FVx{8P$u)ET!o4~h3d81+WB z@Fp&P*;-~w7Fh0v;*&6gD8eH;bet=o%No22sM{z-!~}c7Z6RGrh7f;00zXkCG&OI% z#9*&s7ZCK7elHAOPYwAbdGDA~_ZfI>B5VO8-z~-I4Dd_4ff@2f6rX{hIaa?v1{JVj zJ@`#(Y}Q5jv44^$w(8$>B*WF!!ridOj-Gol} zXPt>5e2g(RW! znRIB>IHNu@mZeeUL3I(h%Y3~YgwvGc_%q?^Q<0)w^(?3=OYrx1r1hV{9+P7p6Zo5W zTB;Zb=N;&~?xdY?4Dl5o5ctV9F(BQi5?J0m-w5gn8?WV+O#hrfyVYS+-=iGu*bs*z zJzeei{Z+l@1c%XI5_m&gi_AO=iioCBb^;D6%~l0`MpgpFA5<^ z|D0WKySQeMCo^7+wX*cZHmEf5cIx%##k?w+!Z&~J{XP*THt&MkY;L@^f!M9v_A(tg z{`vW`yxiBfcqn*JA?6aI@z9vVF1U^CMe>uy%<^}S<0S}0PQHk;9I89Gh8&`!2uWp4 z&O^2giM7s6RX9Ujlmh%h5uuC~?buZ-D|)+ihvS~}M^sV438JDiD%HwqJa#`$Lq$}A zoo81L{Q^Gc{=|-BeE9g%Eh z^?PtSs1W$7Xd7Xe^aT}#AEn1BVoBSzn-NY}zli9bqRp{11i>EdP`mgA``+1Kio3?o zzwG@Vs(&Lv0}esx&^R*at4AV)Zu{J7HHCQ~4g@hb41C1+?I1T%L{~@6d@*Om2=FNS z5RiVAq!#jH7pBCNn3dt4N8QY&mRM zCS$p^+M{sKZrA0RGK&ua-;^G?Kq*%m7%|8GL?`ffdfN)CaZ9+D(XmCVQcv9-LN>Xb zSsTQu0qUpoBrn5X_6+He)x}uXZR&(IZ(rg^3M!pMNEcQ;I(mF&Vw9h*roUn$ye=aj zG#bRov4}WYHN+=}9dYOl6?iH8G@-ZmKVIoOaC+MF1-YTLldU{sW65?LQqoT)Qeh2C8@xZvb> z;pl-rNsPK9CB~{;7fu)9`lot|?foOv#g4`M*aZ2FN^F>mxfH0u_moG{>Den5pI`05 z(xX4F*To+j{&VBUYmQ#l;a@wa+0Xj(SG`PWlE-jPa!TnlDL;=$@VAP&+{9Os8O!mK zlQqvB4c7qak?isy+EX#DkN*@samscRT{hbI8t)Xj57Q7 zOTw$(0~db9olMigts#h_1pe9`aK?zLz$XQ!~j!=$$X4B zt#5bv%6mESVuK7gpXjPoBoPvQp@_7%u^~q^Fa04vqhz&;WoObrYUNhnjW@AT=QuYRw=&eu%Co&l6=x;HS>cA59kod4`O*qZ46J`T*W1SD zoSU+Mwy|XNicy-p_wLp_NqbGgX<~fuDO>2@CkDXneZ6^G{~uW;C={c0EYn@TG%Zv9 z>*n&D`tBvkh-1?UtvKYCy>bK`(p5zP_ew#Kc^n<9>s^hghzO~h!Ih|!7K7kj2B69% zUzukQiVz)q12l36vclU`a8m-3$V1&S-ELT)aRWU%?O+^9UBs3I79 zbJlz}0Q3Z@?!0p@Zwh5^RD~YQmTI4G|6YasJClWw$GBLag@<}7Pd}sP-1AdiS1&W3 zDZ&!q!6pb!keG#RY7W&cJAHFu+X#^(wFSu>%N3GTh}jIx22#97m)7y}oMw(=bI0!`z!Jy*kb2(T z8$lOG>&;n*LeL-=(`A4Q86G_0XQ!t;c}>#IZyfJ(jqjP@HOU@KdP1`W>MTnT{N-=8dEtXZaqUl&$G8eV#T;|$I5!;wQ4)s}04gM`L%)3to(5dH zgmmxIsp5MT?wU>YjnV*522-Q#`aq1hq~LN?THM)&9-3^Ud~5XxLdW2#DaxL<%TTq% z?3-+R6BQ3OchT8jBKz?cQu;RT1ZxK#%oJ=hWG|NlZGS)0vegD^(CI1{i;!D?GB^Ww zfcBwA!?D6Tw0bt2!SKby5o99I%U$bOL{`Dn!jTNY10ms zZX6RM@IQ$7i4VWY#?|IMV9(It)#ml@k`hYGh(8wh5p3_2C~CpnB=aZHGSS{ef59AOqO0r z4=N(}JCVdiO{=b*@7IjG9cacFfw08DGSh`>yYA{9-ime2c<1`4E_2ezp>`?AHuFXH zZ#?w=wxic_ ztDX^ML3|yKZ&pMEqP347iAf8F+mTn=)N`fqhy|N1cPYuiZe1+qP}4tMQ#gC9|K=X% z`RS4@(+biIE|eZ`i?p+wPH%!`d1IE8vi+O11ua3eWN9uFax{QwMIz%gie!-Ak+G@o z{u;_vc;<$D{db!~ee-@4Kv$#t)WHwkgz=phgNUnrSJ@0xauL`Q))%e3&u$uskVvt2 zk8f#qP_n5IfpasbN|O+xbEVlg6O!qCNYTIP&nMQMs1bqbHt!&7%*Nq2GAv7$>Uh(& zr<_?ifxBJ5g&iSlllfS%gBXWkrBS7RrCv%ESi(`!N*ZAp&6YSVTo+Y-){t=voE^2u z;u|iH&)|9I+ssQ0ze{i%`!S1hZM^+Bs(9(1d3&5IbG;udLeW|`Z9zDW`X7Ku z3p%WLwf^>w91%l82q9!(6jy|&;{2#9b@(G>B;ML+)Mm%au*|q=MREI@!c6u>K4vVKV6Mw8vzhUYVB;upaS&CCq3!k{BOGn^-o$Ho< z7e8;<@NT53kG;NsqF^yIq>^*SYRObRe&_?b@joc?(MO*}gRBReH!VV9lYuiUmAPp4q;N zhCCD%NO`uzi-ME1Ic#V{%;t8Y^YJ2r4G7QAG{!o%y&;@2X*4@0GYyms{{kx|^3>w( z7NCJjNJ`7k${t-8f_m$}ZDGbk<$W_^qoMVRmmIVVjJ^3buOn^BqtaO2HedO$Ip!}| zg&jSKW6u)zh!Txp6Y2^}05>IWskA($S!1kDg}5t=OCBQ@ewY`}hwJl$u8kZiL*Sl~ z3Pz2`k;l5msOR5352}gxut;B?Oh)^>J9RRgzX=cVCe=M~locUpng8j?)tN};N4hMJCu-e752R>*kvfB{ z&n!e{zavu)vu6voUfEBO)p6XzBz^9`zVs_l!+ZWylcy3)i9P!UF)tKo&PpcC3EbhP zM<3A1H|tFI>|P=B%`aP4Cl-mzP?$V8kIe;z-Al$@e7la7S=4#@CL#irWuyy7Koe%) z`7YD_N)U>sQ!UHiaTMiJgb7@GuzDs`!7xRimJwRq4KM2C`cw)#adCyz4R82wY{#eU zlVvk-$guvZ+2_nK!tR9NgorkETK&PpGiO#%`Xn}jXESdXgu$0k>aTP0cbI~O-Lckd zG^BP4jUj@z_V)qeQ1ch^h;2OWgT-MUa~i+CoLL7wUNIf_>) zP{#&d0_v-;3%|Dqw7pfZ4QK^D*)WzCNr#ljLIv8Z2>^H4@-=I1~lEm`W%Rf4aOT*;uOlU2pM1aWZ``W;;3JE zmM1qv`#PY7^3I^Fil}_gyC%gr`bw3CD<=eBSMl$grL0C9yhMOKJ0@!9s<31 z5>Tf)PvQCO2iWfa?G>qc%xf{C6$j?yDd4wo8xB2L6N3jz-?zoN=s|oZG;5pcv55Sr z`f9pLD*tjE);szJ6-kN@4N3v?gH#S{Jfwd2*;dN9^T=fC?peRy2OxOgI5^8~yJH}~ z4wrB^kAPFf;LuB{QEK;<*qt_PXusuNq1fodLcf_8?(UQR@Bpi@pP#|*-2WsOid7~t zh~+d`XQB_vJEmL=?b7%!WGyD;l62|%|D)Aqtj&IH+8H6}YcZazZIPnL`8wSoTHlHa zu=Jyr&qH-VA?^f^Zwu zThQ0D)V2Nj8-?KHAM@(6dQzL{xOOv@nLbV6F&;>%ZrY)PCAjwig||+k)f;Rsw9>I- z1@7KwD8DnAEay(PX)}nef|F_tyidkT2JJ3QOB+vheiXBFF8=n3i*epZl{l?b{;Bbb z{y$F64^f)ks|{>Zy-lf#u398^85zA*%n76(&~ z&z`VIKhWdKna_^qpF^0QDct0{FNa`P%Z8n0e?7;!4~49s__9~ONVG18tHH7Uuo9h* zq1)pTmkC;-XqSAjHL;Ub+b>aFRv z$$qBbLmRliE*4VhI}@80$C)jE_j}}2nwf9K4iJrZxJBbIFw)FVJ{TMVSSt^zgu z-8gV!neLT|NL9<@J;A_ObWH#7OPg3`UiEjR032t^r z3T$AI^31<-DBKZ|rWxjlAoojM`2c8YRvSKVIrT41vT-4Pv7T-uXJRkU__PesyiKac zG;R3ru8(GPb@{6ff;=*SOX4o@-pXXB_&Ro7nviaw9)TG^`8a7ajsy+BxfzhG6ldE_ zR+f&g&v#JRmjTe&tcn0h(_E!8`14@YEZCG(F`oOALt$a_i<=YmTMNtfBQ+!fVY~dU zsH1}VO43`%B@OhYUmT5O-nwked;XhSZbZ7;&vgT`M|mH#Sy13|BS)9(ibDZ9tE{mI zC?2H&p)dubutZaX+qj{!a3k)n_()bNbt+UB2m8`*B?(Roke!+4Xxq(BK_U@<(MVdb z*6TQg;Ljx2Q7p~&!0e?^UOi0X)?4fXi67_xn`74fkv-0-j{yZ#jQ~Z zl`S1I28dHxHsGla3E;`_K4SGI)-%d-#3$`v4`lhOLcuC2R4af#CQmsW=|V5K zq4=Xf{EDv3-w?Wuccp%n7=`TBc8IO<;~EFgMpMIFHtH^)Zo)3ZHp<%ZM7;Lbh0lKd zr&+=s`+u6H|In~an>=jptAlq63Vn2}LiDqJ0S&^GuW_}5;W;rbgr6)%l}5EoQ%ZI5 zJ~PmZdqn!}4gK4KLy48I^LIA38wcr?F;NTdH!}PIIDdM-XmurHmm73NY87WQRnWW$ zHU2^w6p{lZ;63$N^&@~4C*JPPW>eqgIx{`3sSA2g(;#xzTLr-A=@`43Ii5m>LuDop z&~>rko&RLIeKKLV@Sog>Oz_kGh)(pN&ha{;h*u^>DcI|3RgKU!{>X(Vor`hnTOTs{ zs|GP9#9Ko2Y>htsf|{jbhdiuEWWMIxQjus@9e^^PF3Y>VENLozZr0c|Q)-QX{`u+g zWiD_QKF7KDyJDMh=NpDqaDxNFOD2j^z%4;`-|4bNJv=W$w;@vwgRmjCNnuCJJ z_V+fQ^IHGT&n~LIldZy1`(L1qHgk{YiFxFGy#R9Ve!_>JV^k6^5vn0PKHxE3Ee4dv zcbRNysU`J|0$rm0_OQD@YQ<3nXAv4ybYL+gcXWvGGsJ}D75q=7uEAReYcz^Ef!Wpi zGrN5^HFu*l*N%|mm83xKA)vJ}fOcs{nVfQlryzGm%p#uAC&%z7G*uk|%s;mDUP?ao#7aVC9r$BJb^4LvRYGT$|^kdj0 zi4byRGHLY>Fvpp6eo*5$Ohv-#%B-gF^ z`%%#TCrn>OZ-&Ia;9J_4barpQ{^d4hPZVePCL6dKK|!xIWA)<*l%H>GlE0V8bv^d? zdzGfg*@x>GT$U4KTYvt?4iPMWN7JY!5Adwm;3IlNq9v2n)YHB+Y~w&vHd)?y1GcIJ zS_tDR1)y_3wF`RZ$w-kMY0r9-cWk>!sV?hYz_`!2`E6OSJ6Z6T1-`5Q@DugVS-Gb+ zPnGqy)~Vj6}{Y{m4gKrXDPp&GM)4W30^)^&P_cA-mU(3cq6+dG+qwec%3P z%lLQm364xuF+t)HA!?jdtPjRow6Das&OK%c$`XGE+3tTOG1|;LCv}O#BH7wOH6giR zMp@5g)y3V)k?;5}7$%GB;|5=x?oXoQZ)L7Mt^Wq_69hZs=(*bA>&4DSprt>yL4K6I z%OomS*5{AvE% z+kvS#)gOv|ls1;TjB>U-ebg!F5a<%=WMs_XFZ(3oh}ji-Ih_l9_sRKRh>T?eIl zAJC3T4uRM&auiuQ5sXKD8&SgR3Py+4Fo*lf8pi(s4$eMf17`%)w#~{|fT9p3+L~md z7wU-kCq4Stva{bKfcFDKp%x7T<8c3HUC1Q9LxSr@t^#D$Fj;&)6j%lL3_4PD^{_GQ z1S)-?5o%)h3Eba8+QEtCUNUy!V<(j9$G^v{3az8&Q5@w*Cs2_Su-R3K6CaIEYNCC~ zxUph7P)fWa9m17IMBrkbGR?|keWfpTdlvrTUslTg?y*5<*gx-op6k27v(mJB+PV+| zhGh0@8hX20^z`JIWKW&jULJ2w0bXI^8~~(?fd$v@EO4&DK)K6~nqx)!R4C`B^wezR35w!Q=ip*#^{ydW@`^mZQpb!4h1sHt zq+U`oMhSjU8+4sN562Q%bcN3-q#RU#RsH&YlKWw6P)u$hr7y>Y+gDTkk0IxG@)tFK z-ynSz(q&RV`rH6?EcYQI1a6vKt+ey!wD-|=6X*_IH6)c3RiBLOLW%eduk2&&^y!O_n{S%_CRnk%%6Ug?)RRx=TPHiX_MNqrxjVb z2$qzmc`myp&PX%e@%=RIeXjja>NW$4n&hS0`D8?MGOYPq3#vN-Ee^AOjLQsbbz?Z@ zB}B)hytvza+c=n#N5@|OyW;@&#D3NMuT*#HYy_+(v^KN&u^x2Qzv`UoG15S*SobO@ z(vhJvak>V%mGG+D!}WuGYGIzU`SWWGWVGyAtS~*d|4*BzQ8p)vb?sIg5l6c>YvVPpr&3T75kVHB{;CUw)=;F*B7Edo4kR?;n`w zc{;RQZ?QP^ z2fHx0K%EoO#QQ;DuecUttFFhUVE_KaQ-6cb{~!pfT=+p2?-s+@_d_H@MPHG^9o!{p z)QqgnPF;e+8bHtIlx}rPN0e+jIdNv#vudse@yR1l1 zwDlD?Cmz1u^`T{j_TgC3poYo9PAvt3;LzgqcTKc$K9j(+iF2caP=%rjm?Ltl?kf`1@~dq60= z9BGz1ZK*C47V5JLsd#+uw8XSy$LjI-qN`<`!Kb>ar(G9L${2FTUEX_PE`5iI5?cEr z20b1VHy1wRqnGPQVNB>seRM$yoygBfIG91I$ej_OpT#i4ZwX{=C+US4w(6~NlMHu& z%XyB9@WqPPdeUX7;~DpG$?GXzVAcHVo=UT3Ba?FXhqNJW$v5MJ_g&?t!QUouvVdAH zf{ly<6iA5!T-HkfBc3kTC`yLK4{bTU@J`9*xKPp8he3q!bvq;}Cwe)$ayOC7!pVGn z8HcYbZuw>C+09x5TLT;T1`pi4XN}9-7sQ4Sh{LzKAGK>E@19YuKc|lV;0;kVtC-A5 zwv8N{aPt*%+nL;j*8ctc=iY43Tcb!3(Fp8!%I|qw&I_Qs)Vz(|63nK1tG2wD^idE- z<>U%oIg45Xyf}MMhMPNvnCNL!L|b5^X**Ouhnjkix$i%v+}1Q^q5flZE$dLOB8#7! z5|ZYXyJ*rpa^)nvz8p5G1VemiS(#=3bGq#ILwS8Uu%3Ry{qu^9tEpuw8-iD^nckL$ z9pd^$v{l;9xBo{@%*jdB`o+I9a7I5H1V~8ibZNt&wdVYiweR^3n6j|rHBjDs4NMEl zQTN1HDz<%C(mX>+PXV4!3wH+FzxgzcL<<(2`#t*8c=>lW(nK~9lr2>z<_TRmo5I3` zZnko$i!$qz6@#O&ZU2l`D)Tce7qO-1S3+rBx9iMZ;UjSOzaMWIpShDVhjZOs%?G`m zrV1Z)mkeECN~gK;d{|1ANsWa;^{^^M4pt%u!^S}09P?&XfwY)s!kTvOB2V419fJ=pA37O0GeBAozF`Ezr73*brKzgE;fKyvWhr65J zk}2Hmc2Ax;mCtWi6nqeWSX%6o-`+wXSomS8R+1WDbw-p??NY|}ad!Uh zh>VxbJ++;IR$fn*81sGdSIE7ol)Mg=^S3zT?F?g#`ggb^Xl?^8AwH^P0Kp*_`acK` zns#ZsbfH`9SW3wcn|Vg_;W>564#F{Z3B1y->32!LM&U#smfMy7e$st7k@~`hs30KE z^i9l3;+O$Oy4$o>4oX@EWdFYo)Zf+sCbkm+M20?w3TFM^op;u7BiRZ_)*2|4oZ-1NJ!^9cxYbn3kJ=RhzE$*Qn6Gz-{ zPQ5Hs-}s#mY1kM2@yn(GQ~0%UO=-!$Dc5dby@X>W4n7$cha7;f83@{N)X;2KO|o*U zGrHz^VI9+Mnz^Ej1gU?E19{3Zy4mC>c5@?QV*d<;siLwf5})ZvW!!Ay}U|YDhiEwl9wo5*rV;p+5Ln(tLZV!PX5t zQ2P`OB?j1mkHZe|u~+LBZ~tSn`QrAWVcWF&AGv}p`?6N_ zl(Mv}aPe{N$0NXS&1bv>s*Od7X(fodu>j{Nd0T5`s+a$&N%L*=-?4eX260gZ6)bO zYr+8cd5l}+wQFP8=LQn)9WX_s{}?g61E8lZ=HYvehwG{6Pe7$#H*2JGOcXR?+*#Xn z0vt`Mzz7a|*+-zl?QP+*k6?Qg;E5<9Giy}u9#?DHQjB%T>$a2=$xfp{zKce|%G)GX ztVgNxyOK#*dr{TJYn4!f?D`o7F`^B<-&1Ssw&eR{K1{zXmsDajS0yuJ&asQ~LII8^ zxG&n3uWSFyd|aFK-gq(S5UXCxMZff}3vGnNvtW)yqlxM!8Gvf6%LZNmM+_s#A9|H; z0mzkEg9)WL2F&_mmsE3;&`}NwSEcb*{qqW?AoNbt_aFfEdK| z{$F?J0ajo{O~~2th%uPT<9$~Zv~DBYcENkAd9mHM_ERngg9b*-tzsFt~-vH4NZISfms3|VqO!m26b(QD@M2 z-gY+IZo?3c4gKMBl(w)l$?}EL^w>W;;@leh-OVDNFA4q5wo7MJw{xg zbvH^)^zwgpf<{#*89eznR{6X_3mL;N*l2VCdyTxxxRO_(i}9n;Vn{b^SJZQu{m@6t zqj1{BVMziV^&#fki`c=mX#TaYHRy0up@t10LP+-B)?Mg*JAVCSmQa}wLf15X2+YhK zjcshU)9Xs^NLd-T#@N;MO55%~d1Jq;6z>SO8`KXERPf4${r=*0AjVS4^{3_PPKOZn zTWdnTG`(uyZFrt&^9l!@P650xe@^+wjn|9WFi6WDeSVwQOeNYPVgeA5isF@FRs@74 zldD(B2V^!yt*`?@LcO-J=L2gs_n;0ogYTcJ)~EfSGEpZcd&!ZQ_YYuLOtLKRT}6zD z^l!eME*fIc42Bsn8NM_xY#DHAKh!d(!`e34w&(r2N0v3LTx7Z+f0-;U;Pz36;vYE* zV;oCs$B*HQOWKW1@LgUTz2si}B#VX04l=IOf4+BDyR=)mo38}q4@xP|Q_3W!Bf)g{ zIUlLu(R;4dsBZX6R)+|-lQ970Zx=RR>gl-V{TuIzajvdBkFkI2_WnJWRc(qZuj=he$b!a+^&%8EkowCdvIEX=g_=;p$99S3I+eQ{S3x z+yguLF$XvNPKn^n*FR$th6`e1>JIHMIYobxrY!F*B#aa0{(bzZfIVR5rW#>ck!)h)X}x6F-HqOD1mh!Hn)!{0gCcV-+~WQ{!Nl^Vq8TvJg20*Byw=OxOzeKj z$%Mh4FX3~o{FPFsbH8T_#x#2fi-~|d?5`eiBwAbfRUU@ppy3d$T{Q^kb}!+DpD4j& zjIF-Qyv5t3s(VN(nkGheh&D1vRx(^dJ$^6)KP#A9n9!aM`A3S<#+9xeT$x+36*I`M zj>$YjATI4lJ&cS#y1(wM?u9{pz(aT85O^oE)NusWsmW{=z(nA6hS{mYv}nmhYatHM{0WV@p)pqnwMNG@`L}0{Zx<`zT?M!6TpFu=dfLYPvf8g>TiX|2ZDMXRZ>l3X zkdErY7}okaK2ecLQGmoZX(17etM7n>%LP}Oq~c7=s`vYwNi+vR31 z)J$5RTGvCezHzqnu~@hE(o!%<+eXo6*b`%!Mhf({c}CsHDc6T;+u6Fbor(AKEz%Zp z?(WM6pA82U3OR$`eq)^zyCjJ8?|0!�F#a;=^nbJ83vS=<}Uu{o^zk) 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 diff --git a/node_modules/iced-coffee-script/media/rotate2.png b/node_modules/iced-coffee-script/media/rotate2.png deleted file mode 100644 index 369a762735caa43955f78c3a51f12c501589c82f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 90053 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%0Amx=Eh#;k{=BkJ! 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

    a)%jF52mnb(eFpr*0=>L;w!NbWMaG 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%0D&eg#<$F%+H6p$ zc$qb~Cb!o5=J|byhb|G_Xu%-T%M=5u|Grn7g9NSyn4_V*vnpbt>mNOzEgn9ZD9IH` z5Yzh_E8ZS1zQml)xn9^u!Ee`N5%xM8(=C$mjs5FG7ITAZvbor~#MfajySZ~o0^SqX zHO9G#g@O2xsGxUc!MwLo1DQovnt2@w85DJ&cncEn{Xc|Plg2yPC;KJ0PyhLIOryv? z4+v!#uMBS&W>tZ&ISV82aigL8yg3v&-Z5Gg|Govhi8dvCr>v>%mPx;bI!}A=7I5;l zNq6?+l4*LaMlj^S)9oc0_xI`rJCw&B7pcme;pa8TM|ENX`f(LcJDM~>rT~2nW6r#w!PG+u1mzBBTJvd=9O;#2hG*Tw zOPU-CbxTxZWC<&u5^{oj$(>cjf5bV7UA%8b2E51#SyR8XpEYRg4V;pTf_u{R?J=;A z*fdh#C@_#D(md)v3sK^KzXw}*n#(JR-U(zyzG%pR6?9T{w;`qR#ui1N0m&cCS0zZi z#V4e#xsppHCPZNKW4_#Xn{*L$&k-7vV2 zvxApxrTy>FGI5dLFG?$=o8jD2XJ7u%QTpc5;HD9_L0J;JBx6F2XcCC|gNH+DePc6C z@WejSMbJ4~8mk^lriU&!_M%ZspM18^Uc5Ox!NR#)Z&7N_KC@b~zqU=UU@w;xL+ zp@8iwK`aGDlzb*6DtFSsItTOoXg7F#QurRKT=1K3C(UXr8KrTPh1Fl(-1j(;EY}ej z?A02`v?(C!k=N0iV!C@5y0LnvSb2r$f&H0sEHQ?$>4#K@yi4`Uk%GY$>dfdSLy}i1 zOb=_SX6ADU0hE2+A$NJ;v(F-ufTYDy183_#?W+2wDD)v26ZIrtycUaU(@ytI?lN%G zxUaYGug7x5u9ln~;7og$r->S~QGI#q?nx0lZ4vcTe5T)uzE6O&&wS%4sGx4um-)^j z?fuR=1oR_4 zYmAr}#)Ff8j=D-Sx2b)o9&OE@#{vID^2lOIi@UCVAL5Wvy(h+_+n8TGf%Mw@(JpKo zhf@2gHd|VRVmYJA);D6E=w)ryKvbNYg;OtT7H@Bc~ZGBAP5GS9M zD!YIGN2L{-AURbL%LDoQVB&!p9&g5X>hU7u<*RAMiUuW$Dcv3#~RP~TQp~0P)vFZJ_U+5 z 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�UcB1?A1EJaJ)c+*PZGZ96bxa1H^0~F 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 diff --git a/node_modules/iced-coffee-script/package.json b/node_modules/iced-coffee-script/package.json deleted file mode 100644 index d54e155..0000000 --- a/node_modules/iced-coffee-script/package.json +++ /dev/null @@ -1,50 +0,0 @@ -{ - "name": "iced-coffee-script", - "description": "IcedCoffeeScript", - "keywords": [ - "javascript", - "language", - "coffeescript", - "compiler" - ], - "author": { - "name": "Maxwell Krohn" - }, - "version": "1.6.3-g", - "licenses": [ - { - "type": "MIT", - "url": "https://raw.github.com/maxtaco/coffee-script/master/LICENSE" - } - ], - "engines": { - "node": ">=0.8.0" - }, - "directories": { - "lib": "./lib/coffee-script" - }, - "main": "./lib/coffee-script/coffee-script", - "bin": { - "iced": "./bin/coffee", - "icake": "./bin/cake" - }, - "scripts": { - "test": "node ./bin/cake test" - }, - "homepage": "http://maxtaco.github.io/coffee-script", - "bugs": { - "url": "https://github.com/maxtaco/coffee-script/issues" - }, - "repository": { - "type": "git", - "url": "git://github.com/maxtaco/coffee-script.git" - }, - "devDependencies": { - "uglify-js": "~2.2", - "jison": ">=0.2.0" - }, - "readme": " \n \n ICED\n \n _____ __ __\n / ____| / _|/ _|\n .- ----------- -. | | ___ | |_| |_ ___ ___\n ( (ice cubes) ) | | / _ \\| _| _/ _ \\/ _ \\\n |`-..________ ..-'| | |___| (_) | | | || __/ __/\n | | \\_____\\___/|_| |_| \\___|\\___|\n | ;--.\n | (__ \\ _____ _ _\n | | ) ) / ____| (_) | |\n | |/ / | (___ ___ _ __ _ _ __ | |_\n | ( / \\___ \\ / __| '__| | '_ \\| __|\n | |/ ____) | (__| | | | |_) | |_\n | | |_____/ \\___|_| |_| .__/ \\__|\n `-.._________..-' | |\n |_|\n\n\n CoffeeScript is a little language that compiles into JavaScript.\n IcedCoffeeScript is a superset of CoffeeScript that adds two new \n keywords: await and defer.\n\n Install Node.js, and then the CoffeeScript compiler:\n sudo bin/cake install\n\n Or, if you have the Node Package Manager installed:\n npm install -g iced-coffee-script\n (Leave off the -g if you don't wish to install globally.)\n\n Execute a script:\n iced /path/to/script.coffee\n\n Compile a script:\n iced -c /path/to/script.coffee\n\n For documentation, usage, and examples, see:\n http://maxtaco.github.com/coffee-script\n\n For iced-specific technical documentation, see:\n https://github.com/maxtaco/coffee-script/blob/iced/iced.md\n\n To suggest a feature, report a bug, or general discussion:\n https://github.com/maxtaco/coffee-script/issues/\n\n DM or tweet at me with questions: @maxtaco\n\n Or better yet, tweet about how much you love IcedCoffeeScript.\n\n The source repository:\n git://github.com/maxtaco/coffee-script.git\n\n All contributors are listed here:\n http://github.com/maxtaco/coffee-script/contributors\n", - "readmeFilename": "README", - "_id": "iced-coffee-script@1.6.3-g", - "_from": "iced-coffee-script@>1.4.0" -} diff --git a/node_modules/jade/.npmignore b/node_modules/jade/.npmignore deleted file mode 100644 index fdc7b89..0000000 --- a/node_modules/jade/.npmignore +++ /dev/null @@ -1,14 +0,0 @@ -test -support -benchmarks -examples -lib-cov -coverage.html -.gitmodules -.travis.yml -History.md -Makefile -test/ -support/ -benchmarks/ -examples/ diff --git a/node_modules/jade/LICENSE b/node_modules/jade/LICENSE deleted file mode 100644 index 8ad0e0d..0000000 --- a/node_modules/jade/LICENSE +++ /dev/null @@ -1,22 +0,0 @@ -(The MIT License) - -Copyright (c) 2009-2010 TJ Holowaychuk - -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. \ No newline at end of file diff --git a/node_modules/jade/Readme.md b/node_modules/jade/Readme.md deleted file mode 100644 index 9a312cd..0000000 --- a/node_modules/jade/Readme.md +++ /dev/null @@ -1,1300 +0,0 @@ -# Jade - template engine -[![Build Status](https://secure.travis-ci.org/visionmedia/jade.png)](http://travis-ci.org/visionmedia/jade) -[![Dependency Status](https://gemnasium.com/visionmedia/jade.png)](https://gemnasium.com/visionmedia/jade) - - Jade is a high performance template engine heavily influenced by [Haml](http://haml-lang.com) - and implemented with JavaScript for [node](http://nodejs.org). For discussion join the [Google Group](http://groups.google.com/group/jadejs). - -## Test drive - - You can test drive Jade online [here](http://naltatis.github.com/jade-syntax-docs). - -## README Contents - -- [Features](#a1) -- [Implementations](#a2) -- [Installation](#a3) -- [Browser Support](#a4) -- [Public API](#a5) -- [Syntax](#a6) - - [Line Endings](#a6-1) - - [Tags](#a6-2) - - [Tag Text](#a6-3) - - [Comments](#a6-4) - - [Block Comments](#a6-5) - - [Nesting](#a6-6) - - [Block Expansion](#a6-7) - - [Case](#a6-8) - - [Attributes](#a6-9) - - [HTML](#a6-10) - - [Doctypes](#a6-11) -- [Filters](#a7) -- [Code](#a8) -- [Iteration](#a9) -- [Conditionals](#a10) -- [Template inheritance](#a11) -- [Block append / prepend](#a12) -- [Includes](#a13) -- [Mixins](#a14) -- [Generated Output](#a15) -- [Example Makefile](#a16) -- [jade(1)](#a17) -- [Tutorials](#a18) -- [License](#a19) - - -## Features - - - client-side support - - great readability - - flexible indentation - - block-expansion - - mixins - - static includes - - attribute interpolation - - code is escaped by default for security - - contextual error reporting at compile & run time - - executable for compiling jade templates via the command line - - html 5 mode (the default doctype) - - optional memory caching - - combine dynamic and static tag classes - - parse tree manipulation via _filters_ - - template inheritance - - block append / prepend - - supports [Express JS](http://expressjs.com) out of the box - - transparent iteration over objects, arrays, and even non-enumerables via `each` - - block comments - - no tag prefix - - filters - - :stylus must have [stylus](http://github.com/LearnBoost/stylus) installed - - :less must have [less.js](http://github.com/cloudhead/less.js) installed - - :markdown must have [markdown-js](http://github.com/evilstreak/markdown-js), [node-discount](http://github.com/visionmedia/node-discount), or [marked](http://github.com/chjj/marked) installed - - :cdata - - :coffeescript must have [coffee-script](http://jashkenas.github.com/coffee-script/) installed - - [Emacs Mode](https://github.com/brianc/jade-mode) - - [Vim Syntax](https://github.com/digitaltoad/vim-jade) - - [TextMate Bundle](http://github.com/miksago/jade-tmbundle) - - [Coda/SubEtha syntax Mode](https://github.com/aaronmccall/jade.mode) - - [Screencasts](http://tjholowaychuk.com/post/1004255394/jade-screencast-template-engine-for-nodejs) - - [html2jade](https://github.com/donpark/html2jade) converter - - -## Implementations - - - [php](http://github.com/everzet/jade.php) - - [scala](http://scalate.fusesource.org/versions/snapshot/documentation/scaml-reference.html) - - [ruby](http://github.com/stonean/slim) - - [python](https://github.com/SyrusAkbary/pyjade) - - [java](https://github.com/neuland/jade4j) - - -## Installation - -via npm: - -```bash -$ npm install jade -``` - - -## Browser Support - - To compile jade to a single file compatible for client-side use simply execute: - -```bash -$ make jade.js -``` - - Alternatively, if uglifyjs is installed via npm (`npm install uglify-js`) you may execute the following which will create both files. However each release builds these for you. - -```bash -$ make jade.min.js -``` - - By default Jade instruments templates with line number statements such as `__.lineno = 3` for debugging purposes. When used in a browser it's useful to minimize this boiler plate, you can do so by passing the option `{ compileDebug: false }`. The following template - -```jade -p Hello #{name} -``` - - Can then be as small as the following generated function: - -```js -function anonymous(locals, attrs, escape, rethrow) { - var buf = []; - with (locals || {}) { - var interp; - buf.push('\n

    Hello ' + escape((interp = name) == null ? '' : interp) + '\n

    '); - } - return buf.join(""); -} -``` - - Through the use of Jade's `./runtime.js` you may utilize these pre-compiled templates on the client-side _without_ Jade itself, all you need is the associated utility functions (in runtime.js), which are then available as `jade.attrs`, `jade.escape` etc. To enable this you should pass `{ client: true }` to `jade.compile()` to tell Jade to reference the helper functions - via `jade.attrs`, `jade.escape` etc. - -```js -function anonymous(locals, attrs, escape, rethrow) { - var attrs = jade.attrs, escape = jade.escape, rethrow = jade.rethrow; - var buf = []; - with (locals || {}) { - var interp; - buf.push('\n

    Hello ' + escape((interp = name) == null ? '' : interp) + '\n

    '); - } - return buf.join(""); -} -``` - -
    -## Public API - -```js -var jade = require('jade'); - -// Compile a function -var fn = jade.compile('string of jade', options); -fn(locals); -``` - -### Options - - - `self` Use a `self` namespace to hold the locals _(false by default)_ - - `locals` Local variable object - - `filename` Used in exceptions, and required when using includes - - `debug` Outputs tokens and function body generated - - `compiler` Compiler to replace jade's default - - `compileDebug` When `false` no debug instrumentation is compiled - - `pretty` Add pretty-indentation whitespace to output _(false by default)_ - - -## Syntax - - -### Line Endings - -**CRLF** and **CR** are converted to **LF** before parsing. - - -### Tags - -A tag is simply a leading word: - -```jade -html -``` - -for example is converted to `` - -tags can also have ids: - -```jade -div#container -``` - -which would render `
    ` - -how about some classes? - -```jade -div.user-details -``` - -renders `
    ` - -multiple classes? _and_ an id? sure: - -```jade -div#foo.bar.baz -``` - -renders `
    ` - -div div div sure is annoying, how about: - -```jade -#foo -.bar -``` - -which is syntactic sugar for what we have already been doing, and outputs: - -```html -
    -``` - -
    -### Tag Text - -Simply place some content after the tag: - -```jade -p wahoo! -``` - -renders `

    wahoo!

    `. - -well cool, but how about large bodies of text: - -```jade -p - | foo bar baz - | rawr rawr - | super cool - | go jade go -``` - -renders `

    foo bar baz rawr.....

    ` - -interpolation? yup! both types of text can utilize interpolation, -if we passed `{ name: 'tj', email: 'tj@vision-media.ca' }` to the compiled function we can do the following: - -```jade -#user #{name} <#{email}> -``` - -outputs `
    tj <tj@vision-media.ca>
    ` - -Actually want `#{}` for some reason? escape it! - -```jade -p \#{something} -``` - -now we have `

    #{something}

    ` - -We can also utilize the unescaped variant `!{html}`, so the following -will result in a literal script tag: - -```jade -- var html = "" -| !{html} -``` - -Nested tags that also contain text can optionally use a text block: - -```jade -label - | Username: - input(name='user[name]') -``` - -or immediate tag text: - -```jade -label Username: - input(name='user[name]') -``` - -Tags that accept _only_ text such as `script` and `style` do not -need the leading `|` character, for example: - -```jade -html - head - title Example - script - if (foo) { - bar(); - } else { - baz(); - } -``` - -Once again as an alternative, we may use a trailing `.` to indicate a text block, for example: - -```jade -p. - foo asdf - asdf - asdfasdfaf - asdf - asd. -``` - -outputs: - -```html -

    foo asdf -asdf - asdfasdfaf - asdf -asd. -

    -``` - -This however differs from a trailing `.` followed by a space, which although is ignored by the Jade parser, tells Jade that this period is a literal: - -```jade -p . -``` - -outputs: - -```html -

    .

    -``` - -It should be noted that text blocks should be doubled escaped. For example if you desire the following output. - -```html -

    foo\bar

    -``` - -use: - -```jade -p. - foo\\bar -``` - -
    -### Comments - -Single line comments currently look the same as JavaScript comments, -aka `//` and must be placed on their own line: - -```jade -// just some paragraphs -p foo -p bar -``` - -would output - -```html - -

    foo

    -

    bar

    -``` - -Jade also supports unbuffered comments, by simply adding a hyphen: - -```jade -//- will not output within markup -p foo -p bar -``` - -outputting - -```html -

    foo

    -

    bar

    -``` - -
    -### Block Comments - - A block comment is legal as well: - -```jade -body - // - #content - h1 Example -``` - -outputting - -```html - - - -``` - -Jade supports conditional-comments as well, for example: - -```jade -head - //if lt IE 8 - script(src='/ie-sucks.js') -``` - -outputs: - -```html - - - -``` - - -### Nesting - - Jade supports nesting to define the tags in a natural way: - -```jade -ul - li.first - a(href='#') foo - li - a(href='#') bar - li.last - a(href='#') baz -``` - - -### Block Expansion - - Block expansion allows you to create terse single-line nested tags, - the following example is equivalent to the nesting example above. - -```jade -ul - li.first: a(href='#') foo - li: a(href='#') bar - li.last: a(href='#') baz -``` - - -### Case - - The case statement takes the following form: - -```jade -html - body - friends = 10 - case friends - when 0 - p you have no friends - when 1 - p you have a friend - default - p you have #{friends} friends -``` - - Block expansion may also be used: - -```jade -friends = 5 - -html - body - case friends - when 0: p you have no friends - when 1: p you have a friend - default: p you have #{friends} friends -``` - - -### Attributes - -Jade currently supports `(` and `)` as attribute delimiters. - -```jade -a(href='/login', title='View login page') Login -``` - -When a value is `undefined` or `null` the attribute is _not_ added, -so this is fine, it will not compile `something="null"`. - -```jade -div(something=null) -``` - -Boolean attributes are also supported: - -```jade -input(type="checkbox", checked) -``` - -Boolean attributes with code will only output the attribute when `true`: - -```jade -input(type="checkbox", checked=someValue) -``` - -Multiple lines work too: - -```jade -input(type='checkbox', - name='agreement', - checked) -``` - -Multiple lines without the comma work fine: - -```jade -input(type='checkbox' - name='agreement' - checked) -``` - -Funky whitespace? fine: - -```jade -input( - type='checkbox' - name='agreement' - checked) -``` - -Colons work: - -```jade -rss(xmlns:atom="atom") -``` - -Suppose we have the `user` local `{ id: 12, name: 'tobi' }` -and we wish to create an anchor tag with `href` pointing to "/user/12" -we could use regular javascript concatenation: - -```jade -a(href='/user/' + user.id)= user.name -``` - -or we could use jade's interpolation, which I added because everyone -using Ruby or CoffeeScript seems to think this is legal js..: - -```jade -a(href='/user/#{user.id}')= user.name -``` - -The `class` attribute is special-cased when an array is given, -allowing you to pass an array such as `bodyClasses = ['user', 'authenticated']` directly: - -```jade -body(class=bodyClasses) -``` - - -### HTML - - Inline html is fine, we can use the pipe syntax to - write arbitrary text, in this case some html: - -```jade -html - body - |

    Title

    - |

    foo bar baz

    -``` - - Or we can use the trailing `.` to indicate to Jade that we - only want text in this block, allowing us to omit the pipes: - -```jade -html - body. -

    Title

    -

    foo bar baz

    -``` - - Both of these examples yield the same result: - -```html -

    Title

    -

    foo bar baz

    - -``` - - The same rule applies for anywhere you can have text - in jade, raw html is fine: - -```jade -html - body - h1 User #{name} -``` - -
    -### Doctypes - -To add a doctype simply use `!!!`, or `doctype` followed by an optional value: - -```jade -!!! -``` - -or - -```jade -doctype -``` - -Will output the _html 5_ doctype, however: - -```jade -!!! transitional -``` - -Will output the _transitional_ doctype. - -Doctypes are case-insensitive, so the following are equivalent: - -```jade -doctype Basic -doctype basic -``` - -it's also possible to simply pass a doctype literal: - -```jade -doctype html PUBLIC "-//W3C//DTD XHTML Basic 1.1//EN -``` - -yielding: - -```html - -``` - -Below are the doctypes defined by default, which can easily be extended: - -```js -var doctypes = exports.doctypes = { - '5': '', - 'default': '', - 'xml': '', - 'transitional': '', - 'strict': '', - 'frameset': '', - '1.1': '', - 'basic': '', - 'mobile': '' -}; -``` - -To alter the default simply change: - -```js -jade.doctypes.default = 'whatever you want'; -``` - - -## Filters - -Filters are prefixed with `:`, for example `:markdown` and -pass the following block of text to an arbitrary function for processing. View the _features_ -at the top of this document for available filters. - -```jade -body - :markdown - Woah! jade _and_ markdown, very **cool** - we can even link to [stuff](http://google.com) -``` - -Renders: - -```html -

    Woah! jade and markdown, very cool we can even link to stuff

    -``` - - -## Code - -Jade currently supports three classifications of executable code. The first -is prefixed by `-`, and is not buffered: - -```jade -- var foo = 'bar'; -``` - -This can be used for conditionals, or iteration: - -```jade -- for (var key in obj) - p= obj[key] -``` - -Due to Jade's buffering techniques the following is valid as well: - -```jade -- if (foo) - ul - li yay - li foo - li worked -- else - p oh no! didnt work -``` - -Hell, even verbose iteration: - -```jade -- if (items.length) - ul - - items.forEach(function(item){ - li= item - - }) -``` - -Anything you want! - -Next up we have _escaped_ buffered code, which is used to -buffer a return value, which is prefixed by `=`: - -```jade -- var foo = 'bar' -= foo -h1= foo -``` - -Which outputs `bar

    bar

    `. Code buffered by `=` is escaped -by default for security, however to output unescaped return values -you may use `!=`: - -```jade -p!= aVarContainingMoreHTML -``` - - Jade also has designer-friendly variants, making the literal JavaScript - more expressive and declarative. For example the following assignments - are equivalent, and the expression is still regular javascript: - -```jade -- var foo = 'foo ' + 'bar' -foo = 'foo ' + 'bar' -``` - - Likewise Jade has first-class `if`, `else if`, `else`, `until`, `while`, `unless` among others, however you must remember that the expressions are still regular javascript: - -```jade -if foo == 'bar' - ul - li yay - li foo - li worked -else - p oh no! didnt work -``` - -
    -## Iteration - - Along with vanilla JavaScript Jade also supports a subset of - constructs that allow you to create more designer-friendly templates, - one of these constructs is `each`, taking the form: - -```jade -each VAL[, KEY] in OBJ -``` - -An example iterating over an array: - -```jade -- var items = ["one", "two", "three"] -each item in items - li= item -``` - -outputs: - -```html -
  • one
  • -
  • two
  • -
  • three
  • -``` - -iterating an array with index: - -```jade -items = ["one", "two", "three"] -each item, i in items - li #{item}: #{i} -``` - -outputs: - -```html -
  • one: 0
  • -
  • two: 1
  • -
  • three: 2
  • -``` - -iterating an object's keys and values: - -```jade -obj = { foo: 'bar' } -each val, key in obj - li #{key}: #{val} -``` - -would output `
  • foo: bar
  • ` - -Internally Jade converts these statements to regular -JavaScript loops such as `users.forEach(function(user){`, -so lexical scope and nesting applies as it would with regular -JavaScript: - -```jade -each user in users - each role in user.roles - li= role -``` - - You may also use `for` if you prefer: - -```jade -for user in users - for role in user.roles - li= role -``` - -
    -## Conditionals - - Jade conditionals are equivalent to those using the code (`-`) prefix, - however allow you to ditch parenthesis to become more designer friendly, - however keep in mind the expression given is _regular_ JavaScript: - -```jade -for user in users - if user.role == 'admin' - p #{user.name} is an admin - else - p= user.name -``` - - is equivalent to the following using vanilla JavaScript literals: - -```jade -for user in users - - if (user.role == 'admin') - p #{user.name} is an admin - - else - p= user.name -``` - - Jade also provides `unless` which is equivalent to `if (!(expr))`: - -```jade -for user in users - unless user.isAnonymous - p - | Click to view - a(href='/users/' + user.id)= user.name -``` - - -## Template inheritance - - Jade supports template inheritance via the `block` and `extends` keywords. A block is simply a "block" of Jade that may be replaced within a child template, this process is recursive. To activate template inheritance in Express 2.x you must add: `app.set('view options', { layout: false });`. - - Jade blocks can provide default content if desired, however optional as shown below by `block scripts`, `block content`, and `block foot`. - -```jade -html - head - h1 My Site - #{title} - block scripts - script(src='/jquery.js') - body - block content - block foot - #footer - p some footer content -``` - - Now to extend the layout, simply create a new file and use the `extends` directive as shown below, giving the path (with or without the .jade extension). You may now define one or more blocks that will override the parent block content, note that here the `foot` block is _not_ redefined and will output "some footer content". - -```jade -extends layout - -block scripts - script(src='/jquery.js') - script(src='/pets.js') - -block content - h1= title - each pet in pets - include pet -``` - - It's also possible to override a block to provide additional blocks, as shown in the following example where `content` now exposes a `sidebar` and `primary` block for overriding, or the child template could override `content` all together. - -```jade -extends regular-layout - -block content - .sidebar - block sidebar - p nothing - .primary - block primary - p nothing -``` - - -## Block append / prepend - - Jade allows you to _replace_ (default), _prepend_, or _append_ blocks. Suppose for example you have default scripts in a "head" block that you wish to utilize on _every_ page, you might do this: - -```jade -html - head - block head - script(src='/vendor/jquery.js') - script(src='/vendor/caustic.js') - body - block content -``` - - Now suppose you have a page of your application for a JavaScript game, you want some game related scripts as well as these defaults, you can simply `append` the block: - -```jade -extends layout - -block append head - script(src='/vendor/three.js') - script(src='/game.js') -``` - - When using `block append` or `block prepend` the `block` is optional: - -```jade -extends layout - -append head - script(src='/vendor/three.js') - script(src='/game.js') -``` - - -## Includes - - Includes allow you to statically include chunks of Jade, - or other content like css, or html which lives in separate files. The classical example is including a header and footer. Suppose we have the following directory structure: - - ./layout.jade - ./includes/ - ./head.jade - ./foot.jade - -and the following _layout.jade_: - -```jade -html - include includes/head - body - h1 My Site - p Welcome to my super amazing site. - include includes/foot -``` - -both includes _includes/head_ and _includes/foot_ are -read relative to the `filename` option given to _layout.jade_, -which should be an absolute path to this file, however Express does this for you. Include then parses these files, and injects the AST produced to render what you would expect: - -```html - - - My Site - - - -

    My Site

    -

    Welcome to my super lame site.

    - - - -``` - - As mentioned `include` can be used to include other content - such as html or css. By providing an extension Jade will not - assume that the file is Jade source and will include it as - a literal: - -```jade -html - body - include content.html -``` - - Include directives may also accept a block, in which case the - the given block will be appended to the _last_ block defined - in the file. For example if `head.jade` contains: - -```jade -head - script(src='/jquery.js') -``` - - We may append values by providing a block to `include head` - as shown below, adding the two scripts. - -```jade -html - include head - script(src='/foo.js') - script(src='/bar.js') - body - h1 test -``` - - You may also `yield` within an included template, allowing you to explicitly mark where the block given to `include` will be placed. Suppose for example you wish to prepend scripts rather than append, you might do the following: - -```jade -head - yield - script(src='/jquery.js') - script(src='/jquery.ui.js') -``` - - Since included Jade is parsed and literally merges the AST, lexically scoped variables function as if the included Jade was written right in the same file. This means `include` may be used as sort of partial, for example suppose we have `user.jade` which utilizes a `user` variable. - -```jade -h1= user.name -p= user.occupation -``` - -We could then simply `include user` while iterating users, and since the `user` variable is already defined within the loop the included template will have access to it. - -```jade -users = [{ name: 'Tobi', occupation: 'Ferret' }] - -each user in users - .user - include user -``` - -yielding: - -```html -
    -

    Tobi

    -

    Ferret

    -
    -``` - -If we wanted to expose a different variable name as `user` since `user.jade` references that name, we could simply define a new variable as shown here with `user = person`: - -```jade -each person in users - .user - user = person - include user -``` - -
    -## Mixins - - Mixins are converted to regular JavaScript functions in - the compiled template that Jade constructs. Mixins may - take arguments, though not required: - -```jade -mixin list - ul - li foo - li bar - li baz -``` - - Utilizing a mixin without args looks similar, just without a block: - -```jade -h2 Groceries -mixin list -``` - - Mixins may take one or more arguments as well, the arguments - are regular javascripts expressions, so for example the following: - -```jade -mixin pets(pets) - ul.pets - - each pet in pets - li= pet - -mixin profile(user) - .user - h2= user.name - mixin pets(user.pets) -``` - - Would yield something similar to the following html: - -```html -
    -

    tj

    -
      -
    • tobi
    • -
    • loki
    • -
    • jane
    • -
    • manny
    • -
    -
    -``` - -
    -## Generated Output - - Suppose we have the following Jade: - -```jade -- var title = 'yay' -h1.title #{title} -p Just an example -``` - - When the `compileDebug` option is not explicitly `false`, Jade - will compile the function instrumented with `__.lineno = n;`, which - in the event of an exception is passed to `rethrow()` which constructs - a useful message relative to the initial Jade input. - -```js -function anonymous(locals) { - var __ = { lineno: 1, input: "- var title = 'yay'\nh1.title #{title}\np Just an example", filename: "testing/test.js" }; - var rethrow = jade.rethrow; - try { - var attrs = jade.attrs, escape = jade.escape; - var buf = []; - with (locals || {}) { - var interp; - __.lineno = 1; - var title = 'yay' - __.lineno = 2; - buf.push(''); - buf.push('' + escape((interp = title) == null ? '' : interp) + ''); - buf.push('
    '); - __.lineno = 3; - buf.push('

    '); - buf.push('Just an example'); - buf.push('

    '); - } - return buf.join(""); - } catch (err) { - rethrow(err, __.input, __.filename, __.lineno); - } -} -``` - -When the `compileDebug` option _is_ explicitly `false`, this instrumentation -is stripped, which is very helpful for light-weight client-side templates. Combining Jade's options with the `./runtime.js` file in this repo allows you -to toString() compiled templates and avoid running the entire Jade library on -the client, increasing performance, and decreasing the amount of JavaScript -required. - -```js -function anonymous(locals) { - var attrs = jade.attrs, escape = jade.escape; - var buf = []; - with (locals || {}) { - var interp; - var title = 'yay' - buf.push(''); - buf.push('' + escape((interp = title) == null ? '' : interp) + ''); - buf.push(''); - buf.push('

    '); - buf.push('Just an example'); - buf.push('

    '); - } - return buf.join(""); -} -``` - -
    -## Example Makefile - - Below is an example Makefile used to compile _pages/*.jade_ - into _pages/*.html_ files by simply executing `make`. - -_Note:_ If you try to run this snippet and `make` throws a `missing separator` error, you should make sure all indented lines use a tab for indentation instead of spaces. (For whatever reason, GitHub renders this code snippet with 4-space indentation although the actual README file uses tabs in this snippet.) - -```make -JADE = $(shell find . -wholename './pages/*.jade') -HTML = $(JADE:.jade=.html) - -all: $(HTML) - -%.html: %.jade - jade < $< --path $< > $@ - -clean: - rm -f $(HTML) - -.PHONY: clean -``` - -this can be combined with the `watch(1)` command to produce -a watcher-like behaviour: - -```bash -$ watch make -``` - - -## jade(1) - -``` - -Usage: jade [options] [dir|file ...] - -Options: - - -h, --help output usage information - -V, --version output the version number - -o, --obj javascript options object - -O, --out output the compiled html to - -p, --path filename used to resolve includes - -P, --pretty compile pretty html output - -c, --client compile for client-side runtime.js - -D, --no-debug compile without debugging (smaller functions) - -Examples: - - # translate jade the templates dir - $ jade templates - - # create {foo,bar}.html - $ jade {foo,bar}.jade - - # jade over stdio - $ jade < my.jade > my.html - - # jade over stdio - $ echo "h1 Jade!" | jade - - # foo, bar dirs rendering to /tmp - $ jade foo bar --out /tmp - -``` - - -## Tutorials - - - cssdeck interactive [Jade syntax tutorial](http://cssdeck.com/labs/learning-the-jade-templating-engine-syntax) - - cssdeck interactive [Jade logic tutorial](http://cssdeck.com/labs/jade-templating-tutorial-codecast-part-2) - - in [Japanese](http://blog.craftgear.net/4f501e97c1347ec934000001/title/10%E5%88%86%E3%81%A7%E3%82%8F%E3%81%8B%E3%82%8Bjade%E3%83%86%E3%83%B3%E3%83%97%E3%83%AC%E3%83%BC%E3%83%88%E3%82%A8%E3%83%B3%E3%82%B8%E3%83%B3) - - -## License - -(The MIT License) - -Copyright (c) 2009-2010 TJ Holowaychuk <tj@vision-media.ca> - -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/node_modules/jade/Readme_zh-cn.md b/node_modules/jade/Readme_zh-cn.md deleted file mode 100644 index cb90608..0000000 --- a/node_modules/jade/Readme_zh-cn.md +++ /dev/null @@ -1,921 +0,0 @@ -# Jade - 模板引擎 - - Jade 是一个高性能的模板引擎,它深受[Haml](http://haml-lang.com)影响,它是用javascript实现的,并且可以供[node](http://nodejs.org)使用. - - 翻译:[草依山](http://jser.me)   [翻译反馈](http://weibo.com/1826461472/z9jriDdmB#pl_profile_nav)  [Fork me](https://github.com/jserme/jade/) - -## 特性 - - - 客户端支持 - - 代码高可读 - - 灵活的缩进 - - 块展开 - - 混合 - - 静态包含 - - 属性改写 - - 安全,默认代码是转义的 - - 运行时和编译时上下文错误报告 - - 命令行下编译jade模板 - - html 5 模式 (使用 _!!! 5_ 文档类型) - - 在内存中缓存(可选) - - 合并动态和静态标签类 - - 可以通过 _filters_ 修改树 - - 模板继承 - - 原生支持 [Express JS](http://expressjs.com) - - 通过 `each` 枚举对象、数组甚至是不能枚举的对象 - - 块注释 - - 没有前缀的标签 - - AST filters - - 过滤器 - - :sass 必须已经安装[sass.js](http://github.com/visionmedia/sass.js) - - :less 必须已经安装[less.js](http://github.com/cloudhead/less.js) - - :markdown 必须已经安装[markdown-js](http://github.com/evilstreak/markdown-js) 或者[node-discount](http://github.com/visionmedia/node-discount) - - :cdata - - :coffeescript 必须已经安装[coffee-script](http://jashkenas.github.com/coffee-script/) - - [Vim Syntax](https://github.com/digitaltoad/vim-jade) - - [TextMate Bundle](http://github.com/miksago/jade-tmbundle) - - [Screencasts](http://tjholowaychuk.com/post/1004255394/jade-screencast-template-engine-for-nodejs) - - [html2jade](https://github.com/donpark/html2jade) 转换器 - -## 其它实现 - - - [php](http://github.com/everzet/jade.php) - - [scala](http://scalate.fusesource.org/versions/snapshot/documentation/scaml-reference.html) - - [ruby](http://github.com/stonean/slim) - -## 安装 - -通过 npm: - - npm install jade - -## 浏览器支持 - - 把jade编译为一个可供浏览器使用的单文件,只需要简单的执行: - - $ make jade.js - - 如果你已经安装了uglifyjs (`npm install uglify-js`),你可以执行下面的命令它会生成所有的文件。其实每一个正式版本里都帮你做了这事。 - - $ make jade.min.js - - 默认情况下,为了方便调试Jade会把模板组织成带有形如 `__.lineno = 3` 的行号的形式。 - 在浏览器里使用的时候,你可以通过传递一个选项`{ compileDebug: false }`来去掉这个。 - 下面的模板 - - p Hello #{name} - - 会被翻译成下面的函数: - -```js -function anonymous(locals, attrs, escape, rethrow) { - var buf = []; - with (locals || {}) { - var interp; - buf.push('\n

    Hello ' + escape((interp = name) == null ? '' : interp) + '\n

    '); - } - return buf.join(""); -} -``` - - 通过使用Jade的 `./runtime.js`你可以在浏览器使用这些预编译的模板而不需要使用Jade, 你只需要使用runtime.js里的工具函数, 它们会放在`jade.attrs`, `jade.escape` 这些里。 把选项 `{ client: true }` 传递给 `jade.compile()`, Jade 会把这些帮助函数的引用放在`jade.attrs`, `jade.escape`. - -```js -function anonymous(locals, attrs, escape, rethrow) { - var attrs = jade.attrs, escape = jade.escape, rethrow = jade.rethrow; - var buf = []; - with (locals || {}) { - var interp; - buf.push('\n

    Hello ' + escape((interp = name) == null ? '' : interp) + '\n

    '); - } - return buf.join(""); -} -``` - -## 公开API - -```javascript - var jade = require('jade'); - - // Compile a function - var fn = jade.compile('string of jade', options); - fn(locals); -``` - -### 选项 - - - `self` 使用`self` 命名空间来持有本地变量. _默认为false_ - - `locals` 本地变量对象 - - `filename` 异常发生时使用,includes时必需 - - `debug` 输出token和翻译后的函数体 - - `compiler` 替换掉jade默认的编译器 - - `compileDebug` `false`的时候调试的结构不会被输出 - -## 语法 - -### 行结束标志 - -**CRLF** 和 **CR** 会在编译之前被转换为 **LF** - -### 标签 - -标签就是一个简单的单词: - - html - -它会被转换为 `` - -标签也是可以有id的: - - div#container - -它会被转换为 `
    ` - -怎么加类呢? - - div.user-details - -转换为 `
    ` - -多个类? 和id? 也是可以搞定的: - - div#foo.bar.baz - -转换为 `
    ` - -不停的div div div 很讨厌啊 , 可以这样: - - #foo - .bar - -这个算是我们的语法糖,它已经被很好的支持了,上面的会输出: - - `
    ` - -### 标签文本 - -只需要简单的把内容放在标签之后: - - p wahoo! - -它会被渲染为 `

    wahoo!

    `. - -很帅吧,但是大段的文本怎么办呢: - - p - | foo bar baz - | rawr rawr - | super cool - | go jade go - -渲染为 `

    foo bar baz rawr.....

    ` - -怎么和数据结合起来? 所有类型的文本展示都可以和数据结合起来,如果我们把`{ name: 'tj', email: 'tj@vision-media.ca' }` 传给编译函数,下面是模板上的写法: - - #user #{name} <#{email}> - -它会被渲染为 `
    tj <tj@vision-media.ca>
    ` - -当就是要输出`#{}` 的时候怎么办? 转义一下! - - p \#{something} - -它会输出`

    #{something}

    ` - -同样可以使用非转义的变量`!{html}`, 下面的模板将直接输出一个script标签 - - - var html = "" - | !{html} - -内联标签同样可以使用文本块来包含文本: - - label - | Username: - input(name='user[name]') - -或者直接使用标签文本: - - label Username: - input(name='user[name]') - -_只_包含文本的标签,比如`script`, `style`, 和 `textarea` 不需要前缀`|` 字符, 比如: - - html - head - title Example - script - if (foo) { - bar(); - } else { - baz(); - } - -这里还有一种选择,可以使用'.' 来开始一段文本块,比如: - - p. - foo asdf - asdf - asdfasdfaf - asdf - asd. - -会被渲染为: - -

    foo asdf - asdf - asdfasdfaf - asdf - asd - . -

    - -这和带一个空格的 '.' 是不一样的, 带空格的会被Jade的解析器忽略,当作一个普通的文字: - - p . - -渲染为: - -

    .

    - - -需要注意的是广西块需要两次转义。比如想要输出下面的文本: - -

    foo\bar

    - -使用: - - p. - foo\\bar - -### 注释 - -单行注释和JavaScript里是一样的,通过"//"来开始,并且必须单独一行: - - // just some paragraphs - p foo - p bar - -渲染为: - - -

    foo

    -

    bar

    - -Jade 同样支持不输出的注释,加一个短横线就行了: - - //- will not output within markup - p foo - p bar - -渲染为: - -

    foo

    -

    bar

    - -### 块注释 - - 块注释也是支持的: - - body - // - #content - h1 Example - -渲染为: - - - - - -Jade 同样很好的支持了条件注释: - - body - //if IE - a(href='http://www.mozilla.com/en-US/firefox/') Get Firefox - - -渲染为: - - - - - -### 内联 - - Jade 支持以自然的方式定义标签嵌套: - - ul - li.first - a(href='#') foo - li - a(href='#') bar - li.last - a(href='#') baz - -### 块展开 - - 块展开可以帮助你在一行内创建嵌套的标签,下面的例子和上面的是一样的: - - ul - li.first: a(href='#') foo - li: a(href='#') bar - li.last: a(href='#') baz - - -### 属性 - -Jade 现在支持使用'(' 和 ')' 作为属性分隔符 - - a(href='/login', title='View login page') Login - -当一个值是 `undefined` 或者 `null` 属性_不_会被加上, -所以呢,它不会编译出 'something="null"'. - - div(something=null) - -Boolean 属性也是支持的: - - input(type="checkbox", checked) - -使用代码的Boolean 属性只有当属性为`true`时才会输出: - - input(type="checkbox", checked=someValue) - -多行同样也是可用的: - - input(type='checkbox', - name='agreement', - checked) - -多行的时候可以不加逗号: - - input(type='checkbox' - name='agreement' - checked) - -加点空格,格式好看一点?同样支持 - - input( - type='checkbox' - name='agreement' - checked) - -冒号也是支持的: - - rss(xmlns:atom="atom") - -假如我有一个`user` 对象 `{ id: 12, name: 'tobi' }` -我们希望创建一个指向"/user/12"的链接 `href`, 我们可以使用普通的javascript字符串连接,如下: - - a(href='/user/' + user.id)= user.name - -或者我们使用jade的修改方式,这个我想很多使用Ruby或者 CoffeeScript的人会看起来像普通的js..: - - a(href='/user/#{user.id}')= user.name - -`class`属性是一个特殊的属性,你可以直接传递一个数组,比如`bodyClasses = ['user', 'authenticated']` : - - body(class=bodyClasses) - -### HTML - - 内联的html是可以的,我们可以使用管道定义一段文本 : - -``` -html - body - |

    Title

    - |

    foo bar baz

    -``` - - 或者我们可以使用`.` 来告诉Jade我们需要一段文本: - -``` -html - body. -

    Title

    -

    foo bar baz

    -``` - - 上面的两个例子都会渲染成相同的结果: - -``` -

    Title

    -

    foo bar baz

    - -``` - - 这条规则适应于在jade里的任何文本: - -``` -html - body - h1 User #{name} -``` - -### Doctypes - -添加文档类型只需要简单的使用 `!!!`, 或者 `doctype` 跟上下面的可选项: - - !!! - -会渲染出 _transitional_ 文档类型, 或者: - - !!! 5 - -or - - !!! html - -or - - doctype html - -doctypes 是大小写不敏感的, 所以下面两个是一样的: - - doctype Basic - doctype basic - -当然也是可以直接传递一段文档类型的文本: - - doctype html PUBLIC "-//W3C//DTD XHTML Basic 1.1//EN - -渲染后: - - - -会输出 _html 5_ 文档类型. 下面的默认的文档类型,可以很简单的扩展: - -```javascript - var doctypes = exports.doctypes = { - '5': '', - 'xml': '', - 'default': '', - 'transitional': '', - 'strict': '', - 'frameset': '', - '1.1': '', - 'basic': '', - 'mobile': '' - }; -``` - -通过下面的代码可以很简单的改变默认的文档类型: - -```javascript - jade.doctypes.default = 'whatever you want'; -``` - -## 过滤器 - -过滤器前缀 `:`, 比如 `:markdown` 会把下面块里的文本交给专门的函数进行处理。查看顶部 _特性_ 里有哪些可用的过滤器。 - - body - :markdown - Woah! jade _and_ markdown, very **cool** - we can even link to [stuff](http://google.com) - -渲染为: - -

    Woah! jade and markdown, very cool we can even link to stuff

    - -## 代码 - -Jade目前支持三种类型的可执行代码。第一种是前缀`-`, 这是不会被输出的: - - - var foo = 'bar'; - -这可以用在条件语句或者循环中: - - - for (var key in obj) - p= obj[key] - -由于Jade的缓存技术,下面的代码也是可以的: - - - if (foo) - ul - li yay - li foo - li worked - - else - p oh no! didnt work - -哈哈,甚至是很长的循环也是可以的: - - - if (items.length) - ul - - items.forEach(function(item){ - li= item - - }) - -所以你想要的! - -下一步我们要_转义_输出的代码,比如我们返回一个值,只要前缀一个`=`: - - - var foo = 'bar' - = foo - h1= foo - -它会渲染为`bar

    bar

    `. 为了安全起见,使用`=`输出的代码默认是转义的,如果想直接输出不转义的值可以使用`!=`: - - p!= aVarContainingMoreHTML - -Jade 同样是设计师友好的,它可以使javascript更直接更富表现力。比如下面的赋值语句是相等的,同时表达式还是通常的javascript: - - - var foo = 'foo ' + 'bar' - foo = 'foo ' + 'bar' - -Jade会把 `if`, `else if`, `else`, `until`, `while`, `unless`同别的优先对待, 但是你得记住它们还是普通的javascript: - - if foo == 'bar' - ul - li yay - li foo - li worked - else - p oh no! didnt work - -## 循环 - -尽管已经支持JavaScript原生代码,Jade还是支持了一些特殊的标签,它们可以让模板更加易于理解,其中之一就是`each`, 这种形式: - - each VAL[, KEY] in OBJ - -一个遍历数组的例子 : - - - var items = ["one", "two", "three"] - each item in items - li= item - -渲染为: - -
  • one
  • -
  • two
  • -
  • three
  • - -遍历一个数组同时带上索引: - - items = ["one", "two", "three"] - each item, i in items - li #{item}: #{i} - -渲染为: - -
  • one: 0
  • -
  • two: 1
  • -
  • three: 2
  • - -遍历一个数组的键值: - - obj = { foo: 'bar' } - each val, key in obj - li #{key}: #{val} - -将会渲染为:`
  • foo: bar
  • ` - -Jade在内部会把这些语句转换成原生的JavaScript语句,就像使用 `users.forEach(function(user){`, -词法作用域和嵌套会像在普通的JavaScript中一样: - - each user in users - each role in user.roles - li= role - -如果你喜欢,也可以使用`for` : - - for user in users - for role in user.roles - li= role - -## 条件语句 - -Jade 条件语句和使用了(`-`) 前缀的JavaScript语句是一致的,然后它允许你不使用圆括号,这样会看上去对设计师更友好一点, -同时要在心里记住这个表达式渲染出的是_常规_Javascript: - - for user in users - if user.role == 'admin' - p #{user.name} is an admin - else - p= user.name - -和下面的使用了常规JavaScript的代码是相等的: - - for user in users - - if (user.role == 'admin') - p #{user.name} is an admin - - else - p= user.name - -Jade 同时支持`unless`, 这和`if (!(expr))`是等价的: - - for user in users - unless user.isAnonymous - p - | Click to view - a(href='/users/' + user.id)= user.name - -## 模板继承 - - Jade 支持通过 `block` 和 `extends` 关键字来实现模板继承。 一个块就是一个Jade的"block" ,它将在子模板中实现,同时是支持递归的。 - - Jade 块如果没有内容,Jade会添加默认内容,下面的代码默认会输出`block scripts`, `block content`, 和 `block foot`. - -``` -html - head - h1 My Site - #{title} - block scripts - script(src='/jquery.js') - body - block content - block foot - #footer - p some footer content -``` - - 现在我们来继承这个布局,简单创建一个新文件,像下面那样直接使用`extends`,给定路径(可以选择带.jade扩展名或者不带). 你可以定义一个或者更多的块来覆盖父级块内容, 注意到这里的`foot`块_没有_定义,所以它还会输出父级的"some footer content"。 - -``` -extends extend-layout - -block scripts - script(src='/jquery.js') - script(src='/pets.js') - -block content - h1= title - each pet in pets - include pet -``` - - 同样可以在一个子块里添加块,就像下面实现的块`content`里又定义了两个可以被实现的块`sidebar`和`primary`,或者子模板直接实现`content`。 - -``` -extends regular-layout - -block content - .sidebar - block sidebar - p nothing - .primary - block primary - p nothing -``` - -## 包含 - - Includes 允许你静态包含一段Jade, 或者别的存放在单个文件中的东西比如css, html。 非常常见的例子是包含头部和页脚。 假设我们有一个下面目录结构的文件夹: - - ./layout.jade - ./includes/ - ./head.jade - ./tail.jade - -下面是 _layout.jade_ 的内容: - - html - include includes/head - body - h1 My Site - p Welcome to my super amazing site. - include includes/foot - -这两个包含 _includes/head_ 和 _includes/foot_ 都会读取相对于给 _layout.jade_ 参数`filename` 的路径的文件, 这是一个绝对路径,不用担心Express帮你搞定这些了。Include 会解析这些文件,并且插入到已经生成的语法树中,然后渲染为你期待的内容: - -```html - - - My Site - - - -

    My Site

    -

    Welcome to my super lame site.

    - - - -``` - - 前面已经提到,`include` 可以包含比如html或者css这样的内容。给定一个扩展名后,Jade不会把这个文件当作一个Jade源代码,并且会把它当作一个普通文本包含进来: - -``` -html - body - include content.html -``` - - Include 也可以接受块内容,给定的块将会附加到包含文件 _最后_ 的块里。 举个例子,`head.jade` 包含下面的内容: - - - ``` -head - script(src='/jquery.js') -``` - - 我们可以像下面给`include head`添加内容, 这里是添加两个脚本. - -``` -html - include head - script(src='/foo.js') - script(src='/bar.js') - body - h1 test -``` - - -## Mixins - - Mixins在编译的模板里会被Jade转换为普通的JavaScript函数。 Mixins 可以还参数,但不是必需的: - - mixin list - ul - li foo - li bar - li baz - - 使用不带参数的mixin看上去非常简单,在一个块外: - - h2 Groceries - mixin list - - Mixins 也可以带一个或者多个参数,参数就是普通的javascripts表达式,比如下面的例子: - - mixin pets(pets) - ul.pets - - each pet in pets - li= pet - - mixin profile(user) - .user - h2= user.name - mixin pets(user.pets) - - 会输出像下面的html: - -```html -
    -

    tj

    -
      -
    • tobi
    • -
    • loki
    • -
    • jane
    • -
    • manny
    • -
    -
    -``` - -## 产生输出 - - 假设我们有下面的Jade源码: - -``` -- var title = 'yay' -h1.title #{title} -p Just an example -``` - - 当 `compileDebug` 选项不是`false`, Jade 会编译时会把函数里加上 `__.lineno = n;`, 这个参数会在编译出错时传递给`rethrow()`, 而这个函数会在Jade初始输出时给出一个有用的错误信息。 - -```js -function anonymous(locals) { - var __ = { lineno: 1, input: "- var title = 'yay'\nh1.title #{title}\np Just an example", filename: "testing/test.js" }; - var rethrow = jade.rethrow; - try { - var attrs = jade.attrs, escape = jade.escape; - var buf = []; - with (locals || {}) { - var interp; - __.lineno = 1; - var title = 'yay' - __.lineno = 2; - buf.push(''); - buf.push('' + escape((interp = title) == null ? '' : interp) + ''); - buf.push(''); - __.lineno = 3; - buf.push('

    '); - buf.push('Just an example'); - buf.push('

    '); - } - return buf.join(""); - } catch (err) { - rethrow(err, __.input, __.filename, __.lineno); - } -} -``` - -当`compileDebug` 参数是`false`, 这个参数会被去掉,这样对于轻量级的浏览器端模板是非常有用的。结合Jade的参数和当前源码库里的 `./runtime.js` 文件,你可以通过toString()来编译模板而不需要在浏览器端运行整个Jade库,这样可以提高性能,也可以减少载入的JavaScript数量。 - -```js -function anonymous(locals) { - var attrs = jade.attrs, escape = jade.escape; - var buf = []; - with (locals || {}) { - var interp; - var title = 'yay' - buf.push(''); - buf.push('' + escape((interp = title) == null ? '' : interp) + ''); - buf.push(''); - buf.push('

    '); - buf.push('Just an example'); - buf.push('

    '); - } - return buf.join(""); -} -``` - -## Makefile的一个例子 - - 通过执行`make`, 下面的Makefile例子可以把 _pages/*.jade_ 编译为 _pages/*.html_ 。 - -```make -JADE = $(shell find pages/*.jade) -HTML = $(JADE:.jade=.html) - -all: $(HTML) - -%.html: %.jade - jade < $< --path $< > $@ - -clean: - rm -f $(HTML) - -.PHONY: clean -``` - -这个可以和`watch(1)` 命令起来产生像下面的行为: - - $ watch make - -## 命令行的jade(1) - -``` - -使用: jade [options] [dir|file ...] - -选项: - - -h, --help 输出帮助信息 - -v, --version 输出版本号 - -o, --obj javascript选项 - -O, --out 输出编译后的html到 - -p, --path 在处理stdio时,查找包含文件时的查找路径 - -Examples: - - # 编译整个目录 - $ jade templates - - # 生成 {foo,bar}.html - $ jade {foo,bar}.jade - - # 在标准IO下使用jade - $ jade < my.jade > my.html - - # 在标准IO下使用jade, 同时指定用于查找包含的文件 - $ jade < my.jade -p my.jade > my.html - - # 在标准IO下使用jade - $ echo "h1 Jade!" | jade - - # foo, bar 目录渲染到 /tmp - $ jade foo bar --out /tmp - -``` - -## License - -(The MIT License) - -Copyright (c) 2009-2010 TJ Holowaychuk <tj@vision-media.ca> - -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/node_modules/jade/bin/jade b/node_modules/jade/bin/jade deleted file mode 100755 index 8f1c816..0000000 --- a/node_modules/jade/bin/jade +++ /dev/null @@ -1,168 +0,0 @@ -#!/usr/bin/env node - -/** - * Module dependencies. - */ - -var fs = require('fs') - , program = require('commander') - , path = require('path') - , basename = path.basename - , dirname = path.dirname - , resolve = path.resolve - , exists = fs.existsSync || path.existsSync - , join = path.join - , mkdirp = require('mkdirp') - , jade = require('../'); - -// jade options - -var options = {}; - -// options - -program - .version(jade.version) - .usage('[options] [dir|file ...]') - .option('-o, --obj ', 'javascript options object') - .option('-O, --out ', 'output the compiled html to ') - .option('-p, --path ', 'filename used to resolve includes') - .option('-P, --pretty', 'compile pretty html output') - .option('-c, --client', 'compile function for client-side runtime.js') - .option('-D, --no-debug', 'compile without debugging (smaller functions)') - .option('-w, --watch', 'watch files for changes and automatically re-render') - -program.on('--help', function(){ - console.log(' Examples:'); - console.log(''); - console.log(' # translate jade the templates dir'); - console.log(' $ jade templates'); - console.log(''); - console.log(' # create {foo,bar}.html'); - console.log(' $ jade {foo,bar}.jade'); - console.log(''); - console.log(' # jade over stdio'); - console.log(' $ jade < my.jade > my.html'); - console.log(''); - console.log(' # jade over stdio'); - console.log(' $ echo "h1 Jade!" | jade'); - console.log(''); - console.log(' # foo, bar dirs rendering to /tmp'); - console.log(' $ jade foo bar --out /tmp '); - console.log(''); -}); - -program.parse(process.argv); - -// options given, parse them - -if (program.obj) { - if (exists(program.obj)) { - options = JSON.parse(fs.readFileSync(program.obj)); - } else { - options = eval('(' + program.obj + ')'); - } -} - -// --filename - -if (program.path) options.filename = program.path; - -// --no-debug - -options.compileDebug = program.debug; - -// --client - -options.client = program.client; - -// --pretty - -options.pretty = program.pretty; - -// --watch - -options.watch = program.watch; - -// left-over args are file paths - -var files = program.args; - -// compile files - -if (files.length) { - console.log(); - files.forEach(renderFile); - if (options.watch) { - files.forEach(function (file) { - fs.watchFile(file, {interval: 100}, function (curr, prev) { - if (curr.mtime > prev.mtime) renderFile(file); - }); - }); - } - process.on('exit', function () { - console.log(); - }); -// stdio -} else { - stdin(); -} - -/** - * Compile from stdin. - */ - -function stdin() { - var buf = ''; - process.stdin.setEncoding('utf8'); - process.stdin.on('data', function(chunk){ buf += chunk; }); - process.stdin.on('end', function(){ - var fn = jade.compile(buf, options); - var output = options.client - ? fn.toString() - : fn(options); - process.stdout.write(output); - }).resume(); -} - -/** - * Process the given path, compiling the jade files found. - * Always walk the subdirectories. - */ - -function renderFile(path) { - var re = /\.jade$/; - fs.lstat(path, function(err, stat) { - if (err) throw err; - // Found jade file - if (stat.isFile() && re.test(path)) { - fs.readFile(path, 'utf8', function(err, str){ - if (err) throw err; - options.filename = path; - var fn = jade.compile(str, options); - var extname = options.client ? '.js' : '.html'; - path = path.replace(re, extname); - if (program.out) path = join(program.out, basename(path)); - var dir = resolve(dirname(path)); - mkdirp(dir, 0755, function(err){ - if (err) throw err; - var output = options.client - ? fn.toString() - : fn(options); - fs.writeFile(path, output, function(err){ - if (err) throw err; - console.log(' \033[90mrendered \033[36m%s\033[0m', path); - }); - }); - }); - // Found directory - } else if (stat.isDirectory()) { - fs.readdir(path, function(err, files) { - if (err) throw err; - files.map(function(filename) { - return path + '/' + filename; - }).forEach(renderFile); - }); - } - }); -} diff --git a/node_modules/jade/index.js b/node_modules/jade/index.js deleted file mode 100644 index 8ad059f..0000000 --- a/node_modules/jade/index.js +++ /dev/null @@ -1,4 +0,0 @@ - -module.exports = process.env.JADE_COV - ? require('./lib-cov/jade') - : require('./lib/jade'); \ No newline at end of file diff --git a/node_modules/jade/jade.js b/node_modules/jade/jade.js deleted file mode 100644 index ad22ec3..0000000 --- a/node_modules/jade/jade.js +++ /dev/null @@ -1,3654 +0,0 @@ -(function() { - -// CommonJS require() - -function require(p){ - var path = require.resolve(p) - , mod = require.modules[path]; - if (!mod) throw new Error('failed to require "' + p + '"'); - if (!mod.exports) { - mod.exports = {}; - mod.call(mod.exports, mod, mod.exports, require.relative(path)); - } - return mod.exports; - } - -require.modules = {}; - -require.resolve = function (path){ - var orig = path - , reg = path + '.js' - , index = path + '/index.js'; - return require.modules[reg] && reg - || require.modules[index] && index - || orig; - }; - -require.register = function (path, fn){ - require.modules[path] = fn; - }; - -require.relative = function (parent) { - return function(p){ - if ('.' != p.charAt(0)) return require(p); - - var path = parent.split('/') - , segs = p.split('/'); - path.pop(); - - for (var i = 0; i < segs.length; i++) { - var seg = segs[i]; - if ('..' == seg) path.pop(); - else if ('.' != seg) path.push(seg); - } - - return require(path.join('/')); - }; - }; - - -require.register("compiler.js", function(module, exports, require){ - -/*! - * Jade - Compiler - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var nodes = require('./nodes') - , filters = require('./filters') - , doctypes = require('./doctypes') - , selfClosing = require('./self-closing') - , runtime = require('./runtime') - , utils = require('./utils'); - - - if (!Object.keys) { - Object.keys = function(obj){ - var arr = []; - for (var key in obj) { - if (obj.hasOwnProperty(key)) { - arr.push(key); - } - } - return arr; - } - } - - if (!String.prototype.trimLeft) { - String.prototype.trimLeft = function(){ - return this.replace(/^\s+/, ''); - } - } - - - -/** - * Initialize `Compiler` with the given `node`. - * - * @param {Node} node - * @param {Object} options - * @api public - */ - -var Compiler = module.exports = function Compiler(node, options) { - this.options = options = options || {}; - this.node = node; - this.hasCompiledDoctype = false; - this.hasCompiledTag = false; - this.pp = options.pretty || false; - this.debug = false !== options.compileDebug; - this.indents = 0; - this.parentIndents = 0; - if (options.doctype) this.setDoctype(options.doctype); -}; - -/** - * Compiler prototype. - */ - -Compiler.prototype = { - - /** - * Compile parse tree to JavaScript. - * - * @api public - */ - - compile: function(){ - this.buf = ['var interp;']; - if (this.pp) this.buf.push("var __indent = [];"); - this.lastBufferedIdx = -1; - this.visit(this.node); - return this.buf.join('\n'); - }, - - /** - * Sets the default doctype `name`. Sets terse mode to `true` when - * html 5 is used, causing self-closing tags to end with ">" vs "/>", - * and boolean attributes are not mirrored. - * - * @param {string} name - * @api public - */ - - setDoctype: function(name){ - name = (name && name.toLowerCase()) || 'default'; - this.doctype = doctypes[name] || ''; - this.terse = this.doctype.toLowerCase() == ''; - this.xml = 0 == this.doctype.indexOf(' 1 && !escape && block.nodes[0].isText && block.nodes[1].isText) - this.prettyIndent(1, true); - - for (var i = 0; i < len; ++i) { - // Pretty print text - if (pp && i > 0 && !escape && block.nodes[i].isText && block.nodes[i-1].isText) - this.prettyIndent(1, false); - - this.visit(block.nodes[i]); - // Multiple text nodes are separated by newlines - if (block.nodes[i+1] && block.nodes[i].isText && block.nodes[i+1].isText) - this.buffer('\\n'); - } - }, - - /** - * Visit `doctype`. Sets terse mode to `true` when html 5 - * is used, causing self-closing tags to end with ">" vs "/>", - * and boolean attributes are not mirrored. - * - * @param {Doctype} doctype - * @api public - */ - - visitDoctype: function(doctype){ - if (doctype && (doctype.val || !this.doctype)) { - this.setDoctype(doctype.val || 'default'); - } - - if (this.doctype) this.buffer(this.doctype); - this.hasCompiledDoctype = true; - }, - - /** - * Visit `mixin`, generating a function that - * may be called within the template. - * - * @param {Mixin} mixin - * @api public - */ - - visitMixin: function(mixin){ - var name = mixin.name.replace(/-/g, '_') + '_mixin' - , args = mixin.args || '' - , block = mixin.block - , attrs = mixin.attrs - , pp = this.pp; - - if (mixin.call) { - if (pp) this.buf.push("__indent.push('" + Array(this.indents + 1).join(' ') + "');") - if (block || attrs.length) { - - this.buf.push(name + '.call({'); - - if (block) { - this.buf.push('block: function(){'); - - // Render block with no indents, dynamically added when rendered - this.parentIndents++; - var _indents = this.indents; - this.indents = 0; - this.visit(mixin.block); - this.indents = _indents; - this.parentIndents--; - - if (attrs.length) { - this.buf.push('},'); - } else { - this.buf.push('}'); - } - } - - if (attrs.length) { - var val = this.attrs(attrs); - if (val.inherits) { - this.buf.push('attributes: merge({' + val.buf - + '}, attributes), escaped: merge(' + val.escaped + ', escaped, true)'); - } else { - this.buf.push('attributes: {' + val.buf + '}, escaped: ' + val.escaped); - } - } - - if (args) { - this.buf.push('}, ' + args + ');'); - } else { - this.buf.push('});'); - } - - } else { - this.buf.push(name + '(' + args + ');'); - } - if (pp) this.buf.push("__indent.pop();") - } else { - this.buf.push('var ' + name + ' = function(' + args + '){'); - this.buf.push('var block = this.block, attributes = this.attributes || {}, escaped = this.escaped || {};'); - this.parentIndents++; - this.visit(block); - this.parentIndents--; - this.buf.push('};'); - } - }, - - /** - * Visit `tag` buffering tag markup, generating - * attributes, visiting the `tag`'s code and block. - * - * @param {Tag} tag - * @api public - */ - - visitTag: function(tag){ - this.indents++; - var name = tag.name - , pp = this.pp; - - if (tag.buffer) name = "' + (" + name + ") + '"; - - if (!this.hasCompiledTag) { - if (!this.hasCompiledDoctype && 'html' == name) { - this.visitDoctype(); - } - this.hasCompiledTag = true; - } - - // pretty print - if (pp && !tag.isInline()) - this.prettyIndent(0, true); - - if ((~selfClosing.indexOf(name) || tag.selfClosing) && !this.xml) { - this.buffer('<' + name); - this.visitAttributes(tag.attrs); - this.terse - ? this.buffer('>') - : this.buffer('/>'); - } else { - // Optimize attributes buffering - if (tag.attrs.length) { - this.buffer('<' + name); - if (tag.attrs.length) this.visitAttributes(tag.attrs); - this.buffer('>'); - } else { - this.buffer('<' + name + '>'); - } - if (tag.code) this.visitCode(tag.code); - this.escape = 'pre' == tag.name; - this.visit(tag.block); - - // pretty print - if (pp && !tag.isInline() && 'pre' != tag.name && !tag.canInline()) - this.prettyIndent(0, true); - - this.buffer(''); - } - this.indents--; - }, - - /** - * Visit `filter`, throwing when the filter does not exist. - * - * @param {Filter} filter - * @api public - */ - - visitFilter: function(filter){ - var fn = filters[filter.name]; - - // unknown filter - if (!fn) { - if (filter.isASTFilter) { - throw new Error('unknown ast filter "' + filter.name + ':"'); - } else { - throw new Error('unknown filter ":' + filter.name + '"'); - } - } - - if (filter.isASTFilter) { - this.buf.push(fn(filter.block, this, filter.attrs)); - } else { - var text = filter.block.nodes.map(function(node){ return node.val }).join('\n'); - filter.attrs = filter.attrs || {}; - filter.attrs.filename = this.options.filename; - this.buffer(utils.text(fn(text, filter.attrs))); - } - }, - - /** - * Visit `text` node. - * - * @param {Text} text - * @api public - */ - - visitText: function(text){ - text = utils.text(text.val.replace(/\\/g, '_SLASH_')); - if (this.escape) text = escape(text); - text = text.replace(/_SLASH_/g, '\\\\'); - this.buffer(text); - }, - - /** - * Visit a `comment`, only buffering when the buffer flag is set. - * - * @param {Comment} comment - * @api public - */ - - visitComment: function(comment){ - if (!comment.buffer) return; - if (this.pp) this.prettyIndent(1, true); - this.buffer(''); - }, - - /** - * Visit a `BlockComment`. - * - * @param {Comment} comment - * @api public - */ - - visitBlockComment: function(comment){ - if (!comment.buffer) return; - if (0 == comment.val.trim().indexOf('if')) { - this.buffer(''); - } else { - this.buffer(''); - } - }, - - /** - * Visit `code`, respecting buffer / escape flags. - * If the code is followed by a block, wrap it in - * a self-calling function. - * - * @param {Code} code - * @api public - */ - - visitCode: function(code){ - // Wrap code blocks with {}. - // we only wrap unbuffered code blocks ATM - // since they are usually flow control - - // Buffer code - if (code.buffer) { - var val = code.val.trimLeft(); - this.buf.push('var __val__ = ' + val); - val = 'null == __val__ ? "" : __val__'; - if (code.escape) val = 'escape(' + val + ')'; - this.buf.push("buf.push(" + val + ");"); - } else { - this.buf.push(code.val); - } - - // Block support - if (code.block) { - if (!code.buffer) this.buf.push('{'); - this.visit(code.block); - if (!code.buffer) this.buf.push('}'); - } - }, - - /** - * Visit `each` block. - * - * @param {Each} each - * @api public - */ - - visitEach: function(each){ - this.buf.push('' - + '// iterate ' + each.obj + '\n' - + ';(function(){\n' - + ' if (\'number\' == typeof ' + each.obj + '.length) {\n'); - - if (each.alternative) { - this.buf.push(' if (' + each.obj + '.length) {'); - } - - this.buf.push('' - + ' for (var ' + each.key + ' = 0, $$l = ' + each.obj + '.length; ' + each.key + ' < $$l; ' + each.key + '++) {\n' - + ' var ' + each.val + ' = ' + each.obj + '[' + each.key + '];\n'); - - this.visit(each.block); - - this.buf.push(' }\n'); - - if (each.alternative) { - this.buf.push(' } else {'); - this.visit(each.alternative); - this.buf.push(' }'); - } - - this.buf.push('' - + ' } else {\n' - + ' var $$l = 0;\n' - + ' for (var ' + each.key + ' in ' + each.obj + ') {\n' - + ' $$l++;' - + ' if (' + each.obj + '.hasOwnProperty(' + each.key + ')){' - + ' var ' + each.val + ' = ' + each.obj + '[' + each.key + '];\n'); - - this.visit(each.block); - - this.buf.push(' }\n'); - - this.buf.push(' }\n'); - if (each.alternative) { - this.buf.push(' if ($$l === 0) {'); - this.visit(each.alternative); - this.buf.push(' }'); - } - this.buf.push(' }\n}).call(this);\n'); - }, - - /** - * Visit `attrs`. - * - * @param {Array} attrs - * @api public - */ - - visitAttributes: function(attrs){ - var val = this.attrs(attrs); - if (val.inherits) { - this.buf.push("buf.push(attrs(merge({ " + val.buf + - " }, attributes), merge(" + val.escaped + ", escaped, true)));"); - } else if (val.constant) { - eval('var buf={' + val.buf + '};'); - this.buffer(runtime.attrs(buf, JSON.parse(val.escaped)), true); - } else { - this.buf.push("buf.push(attrs({ " + val.buf + " }, " + val.escaped + "));"); - } - }, - - /** - * Compile attributes. - */ - - attrs: function(attrs){ - var buf = [] - , classes = [] - , escaped = {} - , constant = attrs.every(function(attr){ return isConstant(attr.val) }) - , inherits = false; - - if (this.terse) buf.push('terse: true'); - - attrs.forEach(function(attr){ - if (attr.name == 'attributes') return inherits = true; - escaped[attr.name] = attr.escaped; - if (attr.name == 'class') { - classes.push('(' + attr.val + ')'); - } else { - var pair = "'" + attr.name + "':(" + attr.val + ')'; - buf.push(pair); - } - }); - - if (classes.length) { - classes = classes.join(" + ' ' + "); - buf.push("class: " + classes); - } - - return { - buf: buf.join(', ').replace('class:', '"class":'), - escaped: JSON.stringify(escaped), - inherits: inherits, - constant: constant - }; - } -}; - -/** - * Check if expression can be evaluated to a constant - * - * @param {String} expression - * @return {Boolean} - * @api private - */ - -function isConstant(val){ - // Check strings/literals - if (/^ *("([^"\\]*(\\.[^"\\]*)*)"|'([^'\\]*(\\.[^'\\]*)*)'|true|false|null|undefined) *$/i.test(val)) - return true; - - // Check numbers - if (!isNaN(Number(val))) - return true; - - // Check arrays - var matches; - if (matches = /^ *\[(.*)\] *$/.exec(val)) - return matches[1].split(',').every(isConstant); - - return false; -} - -/** - * Escape the given string of `html`. - * - * @param {String} html - * @return {String} - * @api private - */ - -function escape(html){ - return String(html) - .replace(/&(?!\w+;)/g, '&') - .replace(//g, '>') - .replace(/"/g, '"'); -} - -}); // module: compiler.js - -require.register("doctypes.js", function(module, exports, require){ - -/*! - * Jade - doctypes - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -module.exports = { - '5': '' - , 'default': '' - , 'xml': '' - , 'transitional': '' - , 'strict': '' - , 'frameset': '' - , '1.1': '' - , 'basic': '' - , 'mobile': '' -}; -}); // module: doctypes.js - -require.register("filters.js", function(module, exports, require){ - -/*! - * Jade - filters - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -module.exports = { - - /** - * Wrap text with CDATA block. - */ - - cdata: function(str){ - return ''; - }, - - /** - * Transform sass to css, wrapped in style tags. - */ - - sass: function(str){ - str = str.replace(/\\n/g, '\n'); - var sass = require('sass').render(str).replace(/\n/g, '\\n'); - return ''; - }, - - /** - * Transform stylus to css, wrapped in style tags. - */ - - stylus: function(str, options){ - var ret; - str = str.replace(/\\n/g, '\n'); - var stylus = require('stylus'); - stylus(str, options).render(function(err, css){ - if (err) throw err; - ret = css.replace(/\n/g, '\\n'); - }); - return ''; - }, - - /** - * Transform less to css, wrapped in style tags. - */ - - less: function(str){ - var ret; - str = str.replace(/\\n/g, '\n'); - require('less').render(str, function(err, css){ - if (err) throw err; - ret = ''; - }); - return ret; - }, - - /** - * Transform markdown to html. - */ - - markdown: function(str){ - var md; - - // support markdown / discount - try { - md = require('markdown'); - } catch (err){ - try { - md = require('discount'); - } catch (err) { - try { - md = require('markdown-js'); - } catch (err) { - try { - md = require('marked'); - } catch (err) { - throw new - Error('Cannot find markdown library, install markdown, discount, or marked.'); - } - } - } - } - - str = str.replace(/\\n/g, '\n'); - return md.parse(str).replace(/\n/g, '\\n').replace(/'/g,'''); - }, - - /** - * Transform coffeescript to javascript. - */ - - coffeescript: function(str){ - var js = require('coffee-script').compile(str).replace(/\\/g, '\\\\').replace(/\n/g, '\\n'); - return ''; - } -}; - -}); // module: filters.js - -require.register("inline-tags.js", function(module, exports, require){ - -/*! - * Jade - inline tags - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -module.exports = [ - 'a' - , 'abbr' - , 'acronym' - , 'b' - , 'br' - , 'code' - , 'em' - , 'font' - , 'i' - , 'img' - , 'ins' - , 'kbd' - , 'map' - , 'samp' - , 'small' - , 'span' - , 'strong' - , 'sub' - , 'sup' -]; -}); // module: inline-tags.js - -require.register("jade.js", function(module, exports, require){ -/*! - * Jade - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Parser = require('./parser') - , Lexer = require('./lexer') - , Compiler = require('./compiler') - , runtime = require('./runtime') - -/** - * Library version. - */ - -exports.version = '0.27.6'; - -/** - * Expose self closing tags. - */ - -exports.selfClosing = require('./self-closing'); - -/** - * Default supported doctypes. - */ - -exports.doctypes = require('./doctypes'); - -/** - * Text filters. - */ - -exports.filters = require('./filters'); - -/** - * Utilities. - */ - -exports.utils = require('./utils'); - -/** - * Expose `Compiler`. - */ - -exports.Compiler = Compiler; - -/** - * Expose `Parser`. - */ - -exports.Parser = Parser; - -/** - * Expose `Lexer`. - */ - -exports.Lexer = Lexer; - -/** - * Nodes. - */ - -exports.nodes = require('./nodes'); - -/** - * Jade runtime helpers. - */ - -exports.runtime = runtime; - -/** - * Template function cache. - */ - -exports.cache = {}; - -/** - * Parse the given `str` of jade and return a function body. - * - * @param {String} str - * @param {Object} options - * @return {String} - * @api private - */ - -function parse(str, options){ - try { - // Parse - var parser = new Parser(str, options.filename, options); - - // Compile - var compiler = new (options.compiler || Compiler)(parser.parse(), options) - , js = compiler.compile(); - - // Debug compiler - if (options.debug) { - console.error('\nCompiled Function:\n\n\033[90m%s\033[0m', js.replace(/^/gm, ' ')); - } - - return '' - + 'var buf = [];\n' - + (options.self - ? 'var self = locals || {};\n' + js - : 'with (locals || {}) {\n' + js + '\n}\n') - + 'return buf.join("");'; - } catch (err) { - parser = parser.context(); - runtime.rethrow(err, parser.filename, parser.lexer.lineno); - } -} - -/** - * Strip any UTF-8 BOM off of the start of `str`, if it exists. - * - * @param {String} str - * @return {String} - * @api private - */ - -function stripBOM(str){ - return 0xFEFF == str.charCodeAt(0) - ? str.substring(1) - : str; -} - -/** - * Compile a `Function` representation of the given jade `str`. - * - * Options: - * - * - `compileDebug` when `false` debugging code is stripped from the compiled template - * - `client` when `true` the helper functions `escape()` etc will reference `jade.escape()` - * for use with the Jade client-side runtime.js - * - * @param {String} str - * @param {Options} options - * @return {Function} - * @api public - */ - -exports.compile = function(str, options){ - var options = options || {} - , client = options.client - , filename = options.filename - ? JSON.stringify(options.filename) - : 'undefined' - , fn; - - str = stripBOM(String(str)); - - if (options.compileDebug !== false) { - fn = [ - 'var __jade = [{ lineno: 1, filename: ' + filename + ' }];' - , 'try {' - , parse(str, options) - , '} catch (err) {' - , ' rethrow(err, __jade[0].filename, __jade[0].lineno);' - , '}' - ].join('\n'); - } else { - fn = parse(str, options); - } - - if (client) { - fn = 'attrs = attrs || jade.attrs; escape = escape || jade.escape; rethrow = rethrow || jade.rethrow; merge = merge || jade.merge;\n' + fn; - } - - fn = new Function('locals, attrs, escape, rethrow, merge', fn); - - if (client) return fn; - - return function(locals){ - return fn(locals, runtime.attrs, runtime.escape, runtime.rethrow, runtime.merge); - }; -}; - -/** - * Render the given `str` of jade and invoke - * the callback `fn(err, str)`. - * - * Options: - * - * - `cache` enable template caching - * - `filename` filename required for `include` / `extends` and caching - * - * @param {String} str - * @param {Object|Function} options or fn - * @param {Function} fn - * @api public - */ - -exports.render = function(str, options, fn){ - // swap args - if ('function' == typeof options) { - fn = options, options = {}; - } - - // cache requires .filename - if (options.cache && !options.filename) { - return fn(new Error('the "filename" option is required for caching')); - } - - try { - var path = options.filename; - var tmpl = options.cache - ? exports.cache[path] || (exports.cache[path] = exports.compile(str, options)) - : exports.compile(str, options); - fn(null, tmpl(options)); - } catch (err) { - fn(err); - } -}; - -/** - * Render a Jade file at the given `path` and callback `fn(err, str)`. - * - * @param {String} path - * @param {Object|Function} options or callback - * @param {Function} fn - * @api public - */ - -exports.renderFile = function(path, options, fn){ - var key = path + ':string'; - - if ('function' == typeof options) { - fn = options, options = {}; - } - - try { - options.filename = path; - var str = options.cache - ? exports.cache[key] || (exports.cache[key] = fs.readFileSync(path, 'utf8')) - : fs.readFileSync(path, 'utf8'); - exports.render(str, options, fn); - } catch (err) { - fn(err); - } -}; - -/** - * Express support. - */ - -exports.__express = exports.renderFile; - -}); // module: jade.js - -require.register("lexer.js", function(module, exports, require){ -/*! - * Jade - Lexer - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -var utils = require('./utils'); - -/** - * Initialize `Lexer` with the given `str`. - * - * Options: - * - * - `colons` allow colons for attr delimiters - * - * @param {String} str - * @param {Object} options - * @api private - */ - -var Lexer = module.exports = function Lexer(str, options) { - options = options || {}; - this.input = str.replace(/\r\n|\r/g, '\n'); - this.colons = options.colons; - this.deferredTokens = []; - this.lastIndents = 0; - this.lineno = 1; - this.stash = []; - this.indentStack = []; - this.indentRe = null; - this.pipeless = false; -}; - -/** - * Lexer prototype. - */ - -Lexer.prototype = { - - /** - * Construct a token with the given `type` and `val`. - * - * @param {String} type - * @param {String} val - * @return {Object} - * @api private - */ - - tok: function(type, val){ - return { - type: type - , line: this.lineno - , val: val - } - }, - - /** - * Consume the given `len` of input. - * - * @param {Number} len - * @api private - */ - - consume: function(len){ - this.input = this.input.substr(len); - }, - - /** - * Scan for `type` with the given `regexp`. - * - * @param {String} type - * @param {RegExp} regexp - * @return {Object} - * @api private - */ - - scan: function(regexp, type){ - var captures; - if (captures = regexp.exec(this.input)) { - this.consume(captures[0].length); - return this.tok(type, captures[1]); - } - }, - - /** - * Defer the given `tok`. - * - * @param {Object} tok - * @api private - */ - - defer: function(tok){ - this.deferredTokens.push(tok); - }, - - /** - * Lookahead `n` tokens. - * - * @param {Number} n - * @return {Object} - * @api private - */ - - lookahead: function(n){ - var fetch = n - this.stash.length; - while (fetch-- > 0) this.stash.push(this.next()); - return this.stash[--n]; - }, - - /** - * Return the indexOf `start` / `end` delimiters. - * - * @param {String} start - * @param {String} end - * @return {Number} - * @api private - */ - - indexOfDelimiters: function(start, end){ - var str = this.input - , nstart = 0 - , nend = 0 - , pos = 0; - for (var i = 0, len = str.length; i < len; ++i) { - if (start == str.charAt(i)) { - ++nstart; - } else if (end == str.charAt(i)) { - if (++nend == nstart) { - pos = i; - break; - } - } - } - return pos; - }, - - /** - * Stashed token. - */ - - stashed: function() { - return this.stash.length - && this.stash.shift(); - }, - - /** - * Deferred token. - */ - - deferred: function() { - return this.deferredTokens.length - && this.deferredTokens.shift(); - }, - - /** - * end-of-source. - */ - - eos: function() { - if (this.input.length) return; - if (this.indentStack.length) { - this.indentStack.shift(); - return this.tok('outdent'); - } else { - return this.tok('eos'); - } - }, - - /** - * Blank line. - */ - - blank: function() { - var captures; - if (captures = /^\n *\n/.exec(this.input)) { - this.consume(captures[0].length - 1); - - ++this.lineno; - if (this.pipeless) return this.tok('text', ''); - return this.next(); - } - }, - - /** - * Comment. - */ - - comment: function() { - var captures; - if (captures = /^ *\/\/(-)?([^\n]*)/.exec(this.input)) { - this.consume(captures[0].length); - var tok = this.tok('comment', captures[2]); - tok.buffer = '-' != captures[1]; - return tok; - } - }, - - /** - * Interpolated tag. - */ - - interpolation: function() { - var captures; - if (captures = /^#\{(.*?)\}/.exec(this.input)) { - this.consume(captures[0].length); - return this.tok('interpolation', captures[1]); - } - }, - - /** - * Tag. - */ - - tag: function() { - var captures; - if (captures = /^(\w[-:\w]*)(\/?)/.exec(this.input)) { - this.consume(captures[0].length); - var tok, name = captures[1]; - if (':' == name[name.length - 1]) { - name = name.slice(0, -1); - tok = this.tok('tag', name); - this.defer(this.tok(':')); - while (' ' == this.input[0]) this.input = this.input.substr(1); - } else { - tok = this.tok('tag', name); - } - tok.selfClosing = !! captures[2]; - return tok; - } - }, - - /** - * Filter. - */ - - filter: function() { - return this.scan(/^:(\w+)/, 'filter'); - }, - - /** - * Doctype. - */ - - doctype: function() { - return this.scan(/^(?:!!!|doctype) *([^\n]+)?/, 'doctype'); - }, - - /** - * Id. - */ - - id: function() { - return this.scan(/^#([\w-]+)/, 'id'); - }, - - /** - * Class. - */ - - className: function() { - return this.scan(/^\.([\w-]+)/, 'class'); - }, - - /** - * Text. - */ - - text: function() { - return this.scan(/^(?:\| ?| ?)?([^\n]+)/, 'text'); - }, - - /** - * Extends. - */ - - "extends": function() { - return this.scan(/^extends? +([^\n]+)/, 'extends'); - }, - - /** - * Block prepend. - */ - - prepend: function() { - var captures; - if (captures = /^prepend +([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - var mode = 'prepend' - , name = captures[1] - , tok = this.tok('block', name); - tok.mode = mode; - return tok; - } - }, - - /** - * Block append. - */ - - append: function() { - var captures; - if (captures = /^append +([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - var mode = 'append' - , name = captures[1] - , tok = this.tok('block', name); - tok.mode = mode; - return tok; - } - }, - - /** - * Block. - */ - - block: function() { - var captures; - if (captures = /^block\b *(?:(prepend|append) +)?([^\n]*)/.exec(this.input)) { - this.consume(captures[0].length); - var mode = captures[1] || 'replace' - , name = captures[2] - , tok = this.tok('block', name); - - tok.mode = mode; - return tok; - } - }, - - /** - * Yield. - */ - - yield: function() { - return this.scan(/^yield */, 'yield'); - }, - - /** - * Include. - */ - - include: function() { - return this.scan(/^include +([^\n]+)/, 'include'); - }, - - /** - * Case. - */ - - "case": function() { - return this.scan(/^case +([^\n]+)/, 'case'); - }, - - /** - * When. - */ - - when: function() { - return this.scan(/^when +([^:\n]+)/, 'when'); - }, - - /** - * Default. - */ - - "default": function() { - return this.scan(/^default */, 'default'); - }, - - /** - * Assignment. - */ - - assignment: function() { - var captures; - if (captures = /^(\w+) += *([^;\n]+)( *;? *)/.exec(this.input)) { - this.consume(captures[0].length); - var name = captures[1] - , val = captures[2]; - return this.tok('code', 'var ' + name + ' = (' + val + ');'); - } - }, - - /** - * Call mixin. - */ - - call: function(){ - var captures; - if (captures = /^\+([-\w]+)/.exec(this.input)) { - this.consume(captures[0].length); - var tok = this.tok('call', captures[1]); - - // Check for args (not attributes) - if (captures = /^ *\((.*?)\)/.exec(this.input)) { - if (!/^ *[-\w]+ *=/.test(captures[1])) { - this.consume(captures[0].length); - tok.args = captures[1]; - } - } - - return tok; - } - }, - - /** - * Mixin. - */ - - mixin: function(){ - var captures; - if (captures = /^mixin +([-\w]+)(?: *\((.*)\))?/.exec(this.input)) { - this.consume(captures[0].length); - var tok = this.tok('mixin', captures[1]); - tok.args = captures[2]; - return tok; - } - }, - - /** - * Conditional. - */ - - conditional: function() { - var captures; - if (captures = /^(if|unless|else if|else)\b([^\n]*)/.exec(this.input)) { - this.consume(captures[0].length); - var type = captures[1] - , js = captures[2]; - - switch (type) { - case 'if': js = 'if (' + js + ')'; break; - case 'unless': js = 'if (!(' + js + '))'; break; - case 'else if': js = 'else if (' + js + ')'; break; - case 'else': js = 'else'; break; - } - - return this.tok('code', js); - } - }, - - /** - * While. - */ - - "while": function() { - var captures; - if (captures = /^while +([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - return this.tok('code', 'while (' + captures[1] + ')'); - } - }, - - /** - * Each. - */ - - each: function() { - var captures; - if (captures = /^(?:- *)?(?:each|for) +(\w+)(?: *, *(\w+))? * in *([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - var tok = this.tok('each', captures[1]); - tok.key = captures[2] || '$index'; - tok.code = captures[3]; - return tok; - } - }, - - /** - * Code. - */ - - code: function() { - var captures; - if (captures = /^(!?=|-)([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - var flags = captures[1]; - captures[1] = captures[2]; - var tok = this.tok('code', captures[1]); - tok.escape = flags.charAt(0) === '='; - tok.buffer = flags.charAt(0) === '=' || flags.charAt(1) === '='; - return tok; - } - }, - - /** - * Attributes. - */ - - attrs: function() { - if ('(' == this.input.charAt(0)) { - var index = this.indexOfDelimiters('(', ')') - , str = this.input.substr(1, index-1) - , tok = this.tok('attrs') - , len = str.length - , colons = this.colons - , states = ['key'] - , escapedAttr - , key = '' - , val = '' - , quote - , c - , p; - - function state(){ - return states[states.length - 1]; - } - - function interpolate(attr) { - return attr.replace(/(\\)?#\{([^}]+)\}/g, function(_, escape, expr){ - return escape - ? _ - : quote + " + (" + expr + ") + " + quote; - }); - } - - this.consume(index + 1); - tok.attrs = {}; - tok.escaped = {}; - - function parse(c) { - var real = c; - // TODO: remove when people fix ":" - if (colons && ':' == c) c = '='; - switch (c) { - case ',': - case '\n': - switch (state()) { - case 'expr': - case 'array': - case 'string': - case 'object': - val += c; - break; - default: - states.push('key'); - val = val.trim(); - key = key.trim(); - if ('' == key) return; - key = key.replace(/^['"]|['"]$/g, '').replace('!', ''); - tok.escaped[key] = escapedAttr; - tok.attrs[key] = '' == val - ? true - : interpolate(val); - key = val = ''; - } - break; - case '=': - switch (state()) { - case 'key char': - key += real; - break; - case 'val': - case 'expr': - case 'array': - case 'string': - case 'object': - val += real; - break; - default: - escapedAttr = '!' != p; - states.push('val'); - } - break; - case '(': - if ('val' == state() - || 'expr' == state()) states.push('expr'); - val += c; - break; - case ')': - if ('expr' == state() - || 'val' == state()) states.pop(); - val += c; - break; - case '{': - if ('val' == state()) states.push('object'); - val += c; - break; - case '}': - if ('object' == state()) states.pop(); - val += c; - break; - case '[': - if ('val' == state()) states.push('array'); - val += c; - break; - case ']': - if ('array' == state()) states.pop(); - val += c; - break; - case '"': - case "'": - switch (state()) { - case 'key': - states.push('key char'); - break; - case 'key char': - states.pop(); - break; - case 'string': - if (c == quote) states.pop(); - val += c; - break; - default: - states.push('string'); - val += c; - quote = c; - } - break; - case '': - break; - default: - switch (state()) { - case 'key': - case 'key char': - key += c; - break; - default: - val += c; - } - } - p = c; - } - - for (var i = 0; i < len; ++i) { - parse(str.charAt(i)); - } - - parse(','); - - if ('/' == this.input.charAt(0)) { - this.consume(1); - tok.selfClosing = true; - } - - return tok; - } - }, - - /** - * Indent | Outdent | Newline. - */ - - indent: function() { - var captures, re; - - // established regexp - if (this.indentRe) { - captures = this.indentRe.exec(this.input); - // determine regexp - } else { - // tabs - re = /^\n(\t*) */; - captures = re.exec(this.input); - - // spaces - if (captures && !captures[1].length) { - re = /^\n( *)/; - captures = re.exec(this.input); - } - - // established - if (captures && captures[1].length) this.indentRe = re; - } - - if (captures) { - var tok - , indents = captures[1].length; - - ++this.lineno; - this.consume(indents + 1); - - if (' ' == this.input[0] || '\t' == this.input[0]) { - throw new Error('Invalid indentation, you can use tabs or spaces but not both'); - } - - // blank line - if ('\n' == this.input[0]) return this.tok('newline'); - - // outdent - if (this.indentStack.length && indents < this.indentStack[0]) { - while (this.indentStack.length && this.indentStack[0] > indents) { - this.stash.push(this.tok('outdent')); - this.indentStack.shift(); - } - tok = this.stash.pop(); - // indent - } else if (indents && indents != this.indentStack[0]) { - this.indentStack.unshift(indents); - tok = this.tok('indent', indents); - // newline - } else { - tok = this.tok('newline'); - } - - return tok; - } - }, - - /** - * Pipe-less text consumed only when - * pipeless is true; - */ - - pipelessText: function() { - if (this.pipeless) { - if ('\n' == this.input[0]) return; - var i = this.input.indexOf('\n'); - if (-1 == i) i = this.input.length; - var str = this.input.substr(0, i); - this.consume(str.length); - return this.tok('text', str); - } - }, - - /** - * ':' - */ - - colon: function() { - return this.scan(/^: */, ':'); - }, - - /** - * Return the next token object, or those - * previously stashed by lookahead. - * - * @return {Object} - * @api private - */ - - advance: function(){ - return this.stashed() - || this.next(); - }, - - /** - * Return the next token object. - * - * @return {Object} - * @api private - */ - - next: function() { - return this.deferred() - || this.blank() - || this.eos() - || this.pipelessText() - || this.yield() - || this.doctype() - || this.interpolation() - || this["case"]() - || this.when() - || this["default"]() - || this["extends"]() - || this.append() - || this.prepend() - || this.block() - || this.include() - || this.mixin() - || this.call() - || this.conditional() - || this.each() - || this["while"]() - || this.assignment() - || this.tag() - || this.filter() - || this.code() - || this.id() - || this.className() - || this.attrs() - || this.indent() - || this.comment() - || this.colon() - || this.text(); - } -}; - -}); // module: lexer.js - -require.register("nodes/attrs.js", function(module, exports, require){ - -/*! - * Jade - nodes - Attrs - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'), - Block = require('./block'); - -/** - * Initialize a `Attrs` node. - * - * @api public - */ - -var Attrs = module.exports = function Attrs() { - this.attrs = []; -}; - -/** - * Inherit from `Node`. - */ - -Attrs.prototype = new Node; -Attrs.prototype.constructor = Attrs; - - -/** - * Set attribute `name` to `val`, keep in mind these become - * part of a raw js object literal, so to quote a value you must - * '"quote me"', otherwise or example 'user.name' is literal JavaScript. - * - * @param {String} name - * @param {String} val - * @param {Boolean} escaped - * @return {Tag} for chaining - * @api public - */ - -Attrs.prototype.setAttribute = function(name, val, escaped){ - this.attrs.push({ name: name, val: val, escaped: escaped }); - return this; -}; - -/** - * Remove attribute `name` when present. - * - * @param {String} name - * @api public - */ - -Attrs.prototype.removeAttribute = function(name){ - for (var i = 0, len = this.attrs.length; i < len; ++i) { - if (this.attrs[i] && this.attrs[i].name == name) { - delete this.attrs[i]; - } - } -}; - -/** - * Get attribute value by `name`. - * - * @param {String} name - * @return {String} - * @api public - */ - -Attrs.prototype.getAttribute = function(name){ - for (var i = 0, len = this.attrs.length; i < len; ++i) { - if (this.attrs[i] && this.attrs[i].name == name) { - return this.attrs[i].val; - } - } -}; - -}); // module: nodes/attrs.js - -require.register("nodes/block-comment.js", function(module, exports, require){ - -/*! - * Jade - nodes - BlockComment - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `BlockComment` with the given `block`. - * - * @param {String} val - * @param {Block} block - * @param {Boolean} buffer - * @api public - */ - -var BlockComment = module.exports = function BlockComment(val, block, buffer) { - this.block = block; - this.val = val; - this.buffer = buffer; -}; - -/** - * Inherit from `Node`. - */ - -BlockComment.prototype = new Node; -BlockComment.prototype.constructor = BlockComment; - -}); // module: nodes/block-comment.js - -require.register("nodes/block.js", function(module, exports, require){ - -/*! - * Jade - nodes - Block - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a new `Block` with an optional `node`. - * - * @param {Node} node - * @api public - */ - -var Block = module.exports = function Block(node){ - this.nodes = []; - if (node) this.push(node); -}; - -/** - * Inherit from `Node`. - */ - -Block.prototype = new Node; -Block.prototype.constructor = Block; - - -/** - * Block flag. - */ - -Block.prototype.isBlock = true; - -/** - * Replace the nodes in `other` with the nodes - * in `this` block. - * - * @param {Block} other - * @api private - */ - -Block.prototype.replace = function(other){ - other.nodes = this.nodes; -}; - -/** - * Pust the given `node`. - * - * @param {Node} node - * @return {Number} - * @api public - */ - -Block.prototype.push = function(node){ - return this.nodes.push(node); -}; - -/** - * Check if this block is empty. - * - * @return {Boolean} - * @api public - */ - -Block.prototype.isEmpty = function(){ - return 0 == this.nodes.length; -}; - -/** - * Unshift the given `node`. - * - * @param {Node} node - * @return {Number} - * @api public - */ - -Block.prototype.unshift = function(node){ - return this.nodes.unshift(node); -}; - -/** - * Return the "last" block, or the first `yield` node. - * - * @return {Block} - * @api private - */ - -Block.prototype.includeBlock = function(){ - var ret = this - , node; - - for (var i = 0, len = this.nodes.length; i < len; ++i) { - node = this.nodes[i]; - if (node.yield) return node; - else if (node.textOnly) continue; - else if (node.includeBlock) ret = node.includeBlock(); - else if (node.block && !node.block.isEmpty()) ret = node.block.includeBlock(); - if (ret.yield) return ret; - } - - return ret; -}; - -/** - * Return a clone of this block. - * - * @return {Block} - * @api private - */ - -Block.prototype.clone = function(){ - var clone = new Block; - for (var i = 0, len = this.nodes.length; i < len; ++i) { - clone.push(this.nodes[i].clone()); - } - return clone; -}; - - -}); // module: nodes/block.js - -require.register("nodes/case.js", function(module, exports, require){ - -/*! - * Jade - nodes - Case - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a new `Case` with `expr`. - * - * @param {String} expr - * @api public - */ - -var Case = exports = module.exports = function Case(expr, block){ - this.expr = expr; - this.block = block; -}; - -/** - * Inherit from `Node`. - */ - -Case.prototype = new Node; -Case.prototype.constructor = Case; - - -var When = exports.When = function When(expr, block){ - this.expr = expr; - this.block = block; - this.debug = false; -}; - -/** - * Inherit from `Node`. - */ - -When.prototype = new Node; -When.prototype.constructor = When; - - - -}); // module: nodes/case.js - -require.register("nodes/code.js", function(module, exports, require){ - -/*! - * Jade - nodes - Code - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Code` node with the given code `val`. - * Code may also be optionally buffered and escaped. - * - * @param {String} val - * @param {Boolean} buffer - * @param {Boolean} escape - * @api public - */ - -var Code = module.exports = function Code(val, buffer, escape) { - this.val = val; - this.buffer = buffer; - this.escape = escape; - if (val.match(/^ *else/)) this.debug = false; -}; - -/** - * Inherit from `Node`. - */ - -Code.prototype = new Node; -Code.prototype.constructor = Code; - -}); // module: nodes/code.js - -require.register("nodes/comment.js", function(module, exports, require){ - -/*! - * Jade - nodes - Comment - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Comment` with the given `val`, optionally `buffer`, - * otherwise the comment may render in the output. - * - * @param {String} val - * @param {Boolean} buffer - * @api public - */ - -var Comment = module.exports = function Comment(val, buffer) { - this.val = val; - this.buffer = buffer; -}; - -/** - * Inherit from `Node`. - */ - -Comment.prototype = new Node; -Comment.prototype.constructor = Comment; - -}); // module: nodes/comment.js - -require.register("nodes/doctype.js", function(module, exports, require){ - -/*! - * Jade - nodes - Doctype - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Doctype` with the given `val`. - * - * @param {String} val - * @api public - */ - -var Doctype = module.exports = function Doctype(val) { - this.val = val; -}; - -/** - * Inherit from `Node`. - */ - -Doctype.prototype = new Node; -Doctype.prototype.constructor = Doctype; - -}); // module: nodes/doctype.js - -require.register("nodes/each.js", function(module, exports, require){ - -/*! - * Jade - nodes - Each - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize an `Each` node, representing iteration - * - * @param {String} obj - * @param {String} val - * @param {String} key - * @param {Block} block - * @api public - */ - -var Each = module.exports = function Each(obj, val, key, block) { - this.obj = obj; - this.val = val; - this.key = key; - this.block = block; -}; - -/** - * Inherit from `Node`. - */ - -Each.prototype = new Node; -Each.prototype.constructor = Each; - -}); // module: nodes/each.js - -require.register("nodes/filter.js", function(module, exports, require){ - -/*! - * Jade - nodes - Filter - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node') - , Block = require('./block'); - -/** - * Initialize a `Filter` node with the given - * filter `name` and `block`. - * - * @param {String} name - * @param {Block|Node} block - * @api public - */ - -var Filter = module.exports = function Filter(name, block, attrs) { - this.name = name; - this.block = block; - this.attrs = attrs; - this.isASTFilter = !block.nodes.every(function(node){ return node.isText }); -}; - -/** - * Inherit from `Node`. - */ - -Filter.prototype = new Node; -Filter.prototype.constructor = Filter; - -}); // module: nodes/filter.js - -require.register("nodes/index.js", function(module, exports, require){ - -/*! - * Jade - nodes - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -exports.Node = require('./node'); -exports.Tag = require('./tag'); -exports.Code = require('./code'); -exports.Each = require('./each'); -exports.Case = require('./case'); -exports.Text = require('./text'); -exports.Block = require('./block'); -exports.Mixin = require('./mixin'); -exports.Filter = require('./filter'); -exports.Comment = require('./comment'); -exports.Literal = require('./literal'); -exports.BlockComment = require('./block-comment'); -exports.Doctype = require('./doctype'); - -}); // module: nodes/index.js - -require.register("nodes/literal.js", function(module, exports, require){ - -/*! - * Jade - nodes - Literal - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Literal` node with the given `str. - * - * @param {String} str - * @api public - */ - -var Literal = module.exports = function Literal(str) { - this.str = str - .replace(/\\/g, "\\\\") - .replace(/\n|\r\n/g, "\\n") - .replace(/'/g, "\\'"); -}; - -/** - * Inherit from `Node`. - */ - -Literal.prototype = new Node; -Literal.prototype.constructor = Literal; - - -}); // module: nodes/literal.js - -require.register("nodes/mixin.js", function(module, exports, require){ - -/*! - * Jade - nodes - Mixin - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Attrs = require('./attrs'); - -/** - * Initialize a new `Mixin` with `name` and `block`. - * - * @param {String} name - * @param {String} args - * @param {Block} block - * @api public - */ - -var Mixin = module.exports = function Mixin(name, args, block, call){ - this.name = name; - this.args = args; - this.block = block; - this.attrs = []; - this.call = call; -}; - -/** - * Inherit from `Attrs`. - */ - -Mixin.prototype = new Attrs; -Mixin.prototype.constructor = Mixin; - - - -}); // module: nodes/mixin.js - -require.register("nodes/node.js", function(module, exports, require){ - -/*! - * Jade - nodes - Node - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Initialize a `Node`. - * - * @api public - */ - -var Node = module.exports = function Node(){}; - -/** - * Clone this node (return itself) - * - * @return {Node} - * @api private - */ - -Node.prototype.clone = function(){ - return this; -}; - -}); // module: nodes/node.js - -require.register("nodes/tag.js", function(module, exports, require){ - -/*! - * Jade - nodes - Tag - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Attrs = require('./attrs'), - Block = require('./block'), - inlineTags = require('../inline-tags'); - -/** - * Initialize a `Tag` node with the given tag `name` and optional `block`. - * - * @param {String} name - * @param {Block} block - * @api public - */ - -var Tag = module.exports = function Tag(name, block) { - this.name = name; - this.attrs = []; - this.block = block || new Block; -}; - -/** - * Inherit from `Attrs`. - */ - -Tag.prototype = new Attrs; -Tag.prototype.constructor = Tag; - - -/** - * Clone this tag. - * - * @return {Tag} - * @api private - */ - -Tag.prototype.clone = function(){ - var clone = new Tag(this.name, this.block.clone()); - clone.line = this.line; - clone.attrs = this.attrs; - clone.textOnly = this.textOnly; - return clone; -}; - -/** - * Check if this tag is an inline tag. - * - * @return {Boolean} - * @api private - */ - -Tag.prototype.isInline = function(){ - return ~inlineTags.indexOf(this.name); -}; - -/** - * Check if this tag's contents can be inlined. Used for pretty printing. - * - * @return {Boolean} - * @api private - */ - -Tag.prototype.canInline = function(){ - var nodes = this.block.nodes; - - function isInline(node){ - // Recurse if the node is a block - if (node.isBlock) return node.nodes.every(isInline); - return node.isText || (node.isInline && node.isInline()); - } - - // Empty tag - if (!nodes.length) return true; - - // Text-only or inline-only tag - if (1 == nodes.length) return isInline(nodes[0]); - - // Multi-line inline-only tag - if (this.block.nodes.every(isInline)) { - for (var i = 1, len = nodes.length; i < len; ++i) { - if (nodes[i-1].isText && nodes[i].isText) - return false; - } - return true; - } - - // Mixed tag - return false; -}; -}); // module: nodes/tag.js - -require.register("nodes/text.js", function(module, exports, require){ - -/*! - * Jade - nodes - Text - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Text` node with optional `line`. - * - * @param {String} line - * @api public - */ - -var Text = module.exports = function Text(line) { - this.val = ''; - if ('string' == typeof line) this.val = line; -}; - -/** - * Inherit from `Node`. - */ - -Text.prototype = new Node; -Text.prototype.constructor = Text; - - -/** - * Flag as text. - */ - -Text.prototype.isText = true; -}); // module: nodes/text.js - -require.register("parser.js", function(module, exports, require){ - -/*! - * Jade - Parser - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Lexer = require('./lexer') - , nodes = require('./nodes') - , utils = require('./utils'); - -/** - * Initialize `Parser` with the given input `str` and `filename`. - * - * @param {String} str - * @param {String} filename - * @param {Object} options - * @api public - */ - -var Parser = exports = module.exports = function Parser(str, filename, options){ - this.input = str; - this.lexer = new Lexer(str, options); - this.filename = filename; - this.blocks = {}; - this.mixins = {}; - this.options = options; - this.contexts = [this]; -}; - -/** - * Tags that may not contain tags. - */ - -var textOnly = exports.textOnly = ['script', 'style']; - -/** - * Parser prototype. - */ - -Parser.prototype = { - - /** - * Push `parser` onto the context stack, - * or pop and return a `Parser`. - */ - - context: function(parser){ - if (parser) { - this.contexts.push(parser); - } else { - return this.contexts.pop(); - } - }, - - /** - * Return the next token object. - * - * @return {Object} - * @api private - */ - - advance: function(){ - return this.lexer.advance(); - }, - - /** - * Skip `n` tokens. - * - * @param {Number} n - * @api private - */ - - skip: function(n){ - while (n--) this.advance(); - }, - - /** - * Single token lookahead. - * - * @return {Object} - * @api private - */ - - peek: function() { - return this.lookahead(1); - }, - - /** - * Return lexer lineno. - * - * @return {Number} - * @api private - */ - - line: function() { - return this.lexer.lineno; - }, - - /** - * `n` token lookahead. - * - * @param {Number} n - * @return {Object} - * @api private - */ - - lookahead: function(n){ - return this.lexer.lookahead(n); - }, - - /** - * Parse input returning a string of js for evaluation. - * - * @return {String} - * @api public - */ - - parse: function(){ - var block = new nodes.Block, parser; - block.line = this.line(); - - while ('eos' != this.peek().type) { - if ('newline' == this.peek().type) { - this.advance(); - } else { - block.push(this.parseExpr()); - } - } - - if (parser = this.extending) { - this.context(parser); - var ast = parser.parse(); - this.context(); - // hoist mixins - for (var name in this.mixins) - ast.unshift(this.mixins[name]); - return ast; - } - - return block; - }, - - /** - * Expect the given type, or throw an exception. - * - * @param {String} type - * @api private - */ - - expect: function(type){ - if (this.peek().type === type) { - return this.advance(); - } else { - throw new Error('expected "' + type + '", but got "' + this.peek().type + '"'); - } - }, - - /** - * Accept the given `type`. - * - * @param {String} type - * @api private - */ - - accept: function(type){ - if (this.peek().type === type) { - return this.advance(); - } - }, - - /** - * tag - * | doctype - * | mixin - * | include - * | filter - * | comment - * | text - * | each - * | code - * | yield - * | id - * | class - * | interpolation - */ - - parseExpr: function(){ - switch (this.peek().type) { - case 'tag': - return this.parseTag(); - case 'mixin': - return this.parseMixin(); - case 'block': - return this.parseBlock(); - case 'case': - return this.parseCase(); - case 'when': - return this.parseWhen(); - case 'default': - return this.parseDefault(); - case 'extends': - return this.parseExtends(); - case 'include': - return this.parseInclude(); - case 'doctype': - return this.parseDoctype(); - case 'filter': - return this.parseFilter(); - case 'comment': - return this.parseComment(); - case 'text': - return this.parseText(); - case 'each': - return this.parseEach(); - case 'code': - return this.parseCode(); - case 'call': - return this.parseCall(); - case 'interpolation': - return this.parseInterpolation(); - case 'yield': - this.advance(); - var block = new nodes.Block; - block.yield = true; - return block; - case 'id': - case 'class': - var tok = this.advance(); - this.lexer.defer(this.lexer.tok('tag', 'div')); - this.lexer.defer(tok); - return this.parseExpr(); - default: - throw new Error('unexpected token "' + this.peek().type + '"'); - } - }, - - /** - * Text - */ - - parseText: function(){ - var tok = this.expect('text') - , node = new nodes.Text(tok.val); - node.line = this.line(); - return node; - }, - - /** - * ':' expr - * | block - */ - - parseBlockExpansion: function(){ - if (':' == this.peek().type) { - this.advance(); - return new nodes.Block(this.parseExpr()); - } else { - return this.block(); - } - }, - - /** - * case - */ - - parseCase: function(){ - var val = this.expect('case').val - , node = new nodes.Case(val); - node.line = this.line(); - node.block = this.block(); - return node; - }, - - /** - * when - */ - - parseWhen: function(){ - var val = this.expect('when').val - return new nodes.Case.When(val, this.parseBlockExpansion()); - }, - - /** - * default - */ - - parseDefault: function(){ - this.expect('default'); - return new nodes.Case.When('default', this.parseBlockExpansion()); - }, - - /** - * code - */ - - parseCode: function(){ - var tok = this.expect('code') - , node = new nodes.Code(tok.val, tok.buffer, tok.escape) - , block - , i = 1; - node.line = this.line(); - while (this.lookahead(i) && 'newline' == this.lookahead(i).type) ++i; - block = 'indent' == this.lookahead(i).type; - if (block) { - this.skip(i-1); - node.block = this.block(); - } - return node; - }, - - /** - * comment - */ - - parseComment: function(){ - var tok = this.expect('comment') - , node; - - if ('indent' == this.peek().type) { - node = new nodes.BlockComment(tok.val, this.block(), tok.buffer); - } else { - node = new nodes.Comment(tok.val, tok.buffer); - } - - node.line = this.line(); - return node; - }, - - /** - * doctype - */ - - parseDoctype: function(){ - var tok = this.expect('doctype') - , node = new nodes.Doctype(tok.val); - node.line = this.line(); - return node; - }, - - /** - * filter attrs? text-block - */ - - parseFilter: function(){ - var block - , tok = this.expect('filter') - , attrs = this.accept('attrs'); - - this.lexer.pipeless = true; - block = this.parseTextBlock(); - this.lexer.pipeless = false; - - var node = new nodes.Filter(tok.val, block, attrs && attrs.attrs); - node.line = this.line(); - return node; - }, - - /** - * tag ':' attrs? block - */ - - parseASTFilter: function(){ - var block - , tok = this.expect('tag') - , attrs = this.accept('attrs'); - - this.expect(':'); - block = this.block(); - - var node = new nodes.Filter(tok.val, block, attrs && attrs.attrs); - node.line = this.line(); - return node; - }, - - /** - * each block - */ - - parseEach: function(){ - var tok = this.expect('each') - , node = new nodes.Each(tok.code, tok.val, tok.key); - node.line = this.line(); - node.block = this.block(); - if (this.peek().type == 'code' && this.peek().val == 'else') { - this.advance(); - node.alternative = this.block(); - } - return node; - }, - - /** - * 'extends' name - */ - - parseExtends: function(){ - var path = require('path') - , fs = require('fs') - , dirname = path.dirname - , basename = path.basename - , join = path.join; - - if (!this.filename) - throw new Error('the "filename" option is required to extend templates'); - - var path = this.expect('extends').val.trim() - , dir = dirname(this.filename); - - var path = join(dir, path + '.jade') - , str = fs.readFileSync(path, 'utf8') - , parser = new Parser(str, path, this.options); - - parser.blocks = this.blocks; - parser.contexts = this.contexts; - this.extending = parser; - - // TODO: null node - return new nodes.Literal(''); - }, - - /** - * 'block' name block - */ - - parseBlock: function(){ - var block = this.expect('block') - , mode = block.mode - , name = block.val.trim(); - - block = 'indent' == this.peek().type - ? this.block() - : new nodes.Block(new nodes.Literal('')); - - var prev = this.blocks[name]; - - if (prev) { - switch (prev.mode) { - case 'append': - block.nodes = block.nodes.concat(prev.nodes); - prev = block; - break; - case 'prepend': - block.nodes = prev.nodes.concat(block.nodes); - prev = block; - break; - } - } - - block.mode = mode; - return this.blocks[name] = prev || block; - }, - - /** - * include block? - */ - - parseInclude: function(){ - var path = require('path') - , fs = require('fs') - , dirname = path.dirname - , basename = path.basename - , join = path.join; - - var path = this.expect('include').val.trim() - , dir = dirname(this.filename); - - if (!this.filename) - throw new Error('the "filename" option is required to use includes'); - - // no extension - if (!~basename(path).indexOf('.')) { - path += '.jade'; - } - - // non-jade - if ('.jade' != path.substr(-5)) { - var path = join(dir, path) - , str = fs.readFileSync(path, 'utf8'); - return new nodes.Literal(str); - } - - var path = join(dir, path) - , str = fs.readFileSync(path, 'utf8') - , parser = new Parser(str, path, this.options); - parser.blocks = utils.merge({}, this.blocks); - parser.mixins = this.mixins; - - this.context(parser); - var ast = parser.parse(); - this.context(); - ast.filename = path; - - if ('indent' == this.peek().type) { - ast.includeBlock().push(this.block()); - } - - return ast; - }, - - /** - * call ident block - */ - - parseCall: function(){ - var tok = this.expect('call') - , name = tok.val - , args = tok.args - , mixin = new nodes.Mixin(name, args, new nodes.Block, true); - - this.tag(mixin); - if (mixin.block.isEmpty()) mixin.block = null; - return mixin; - }, - - /** - * mixin block - */ - - parseMixin: function(){ - var tok = this.expect('mixin') - , name = tok.val - , args = tok.args - , mixin; - - // definition - if ('indent' == this.peek().type) { - mixin = new nodes.Mixin(name, args, this.block(), false); - this.mixins[name] = mixin; - return mixin; - // call - } else { - return new nodes.Mixin(name, args, null, true); - } - }, - - /** - * indent (text | newline)* outdent - */ - - parseTextBlock: function(){ - var block = new nodes.Block; - block.line = this.line(); - var spaces = this.expect('indent').val; - if (null == this._spaces) this._spaces = spaces; - var indent = Array(spaces - this._spaces + 1).join(' '); - while ('outdent' != this.peek().type) { - switch (this.peek().type) { - case 'newline': - this.advance(); - break; - case 'indent': - this.parseTextBlock().nodes.forEach(function(node){ - block.push(node); - }); - break; - default: - var text = new nodes.Text(indent + this.advance().val); - text.line = this.line(); - block.push(text); - } - } - - if (spaces == this._spaces) this._spaces = null; - this.expect('outdent'); - return block; - }, - - /** - * indent expr* outdent - */ - - block: function(){ - var block = new nodes.Block; - block.line = this.line(); - this.expect('indent'); - while ('outdent' != this.peek().type) { - if ('newline' == this.peek().type) { - this.advance(); - } else { - block.push(this.parseExpr()); - } - } - this.expect('outdent'); - return block; - }, - - /** - * interpolation (attrs | class | id)* (text | code | ':')? newline* block? - */ - - parseInterpolation: function(){ - var tok = this.advance(); - var tag = new nodes.Tag(tok.val); - tag.buffer = true; - return this.tag(tag); - }, - - /** - * tag (attrs | class | id)* (text | code | ':')? newline* block? - */ - - parseTag: function(){ - // ast-filter look-ahead - var i = 2; - if ('attrs' == this.lookahead(i).type) ++i; - if (':' == this.lookahead(i).type) { - if ('indent' == this.lookahead(++i).type) { - return this.parseASTFilter(); - } - } - - var tok = this.advance() - , tag = new nodes.Tag(tok.val); - - tag.selfClosing = tok.selfClosing; - - return this.tag(tag); - }, - - /** - * Parse tag. - */ - - tag: function(tag){ - var dot; - - tag.line = this.line(); - - // (attrs | class | id)* - out: - while (true) { - switch (this.peek().type) { - case 'id': - case 'class': - var tok = this.advance(); - tag.setAttribute(tok.type, "'" + tok.val + "'"); - continue; - case 'attrs': - var tok = this.advance() - , obj = tok.attrs - , escaped = tok.escaped - , names = Object.keys(obj); - - if (tok.selfClosing) tag.selfClosing = true; - - for (var i = 0, len = names.length; i < len; ++i) { - var name = names[i] - , val = obj[name]; - tag.setAttribute(name, val, escaped[name]); - } - continue; - default: - break out; - } - } - - // check immediate '.' - if ('.' == this.peek().val) { - dot = tag.textOnly = true; - this.advance(); - } - - // (text | code | ':')? - switch (this.peek().type) { - case 'text': - tag.block.push(this.parseText()); - break; - case 'code': - tag.code = this.parseCode(); - break; - case ':': - this.advance(); - tag.block = new nodes.Block; - tag.block.push(this.parseExpr()); - break; - } - - // newline* - while ('newline' == this.peek().type) this.advance(); - - tag.textOnly = tag.textOnly || ~textOnly.indexOf(tag.name); - - // script special-case - if ('script' == tag.name) { - var type = tag.getAttribute('type'); - if (!dot && type && 'text/javascript' != type.replace(/^['"]|['"]$/g, '')) { - tag.textOnly = false; - } - } - - // block? - if ('indent' == this.peek().type) { - if (tag.textOnly) { - this.lexer.pipeless = true; - tag.block = this.parseTextBlock(); - this.lexer.pipeless = false; - } else { - var block = this.block(); - if (tag.block) { - for (var i = 0, len = block.nodes.length; i < len; ++i) { - tag.block.push(block.nodes[i]); - } - } else { - tag.block = block; - } - } - } - - return tag; - } -}; - -}); // module: parser.js - -require.register("runtime.js", function(module, exports, require){ - -/*! - * Jade - runtime - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Lame Array.isArray() polyfill for now. - */ - -if (!Array.isArray) { - Array.isArray = function(arr){ - return '[object Array]' == Object.prototype.toString.call(arr); - }; -} - -/** - * Lame Object.keys() polyfill for now. - */ - -if (!Object.keys) { - Object.keys = function(obj){ - var arr = []; - for (var key in obj) { - if (obj.hasOwnProperty(key)) { - arr.push(key); - } - } - return arr; - } -} - -/** - * Merge two attribute objects giving precedence - * to values in object `b`. Classes are special-cased - * allowing for arrays and merging/joining appropriately - * resulting in a string. - * - * @param {Object} a - * @param {Object} b - * @return {Object} a - * @api private - */ - -exports.merge = function merge(a, b) { - var ac = a['class']; - var bc = b['class']; - - if (ac || bc) { - ac = ac || []; - bc = bc || []; - if (!Array.isArray(ac)) ac = [ac]; - if (!Array.isArray(bc)) bc = [bc]; - ac = ac.filter(nulls); - bc = bc.filter(nulls); - a['class'] = ac.concat(bc).join(' '); - } - - for (var key in b) { - if (key != 'class') { - a[key] = b[key]; - } - } - - return a; -}; - -/** - * Filter null `val`s. - * - * @param {Mixed} val - * @return {Mixed} - * @api private - */ - -function nulls(val) { - return val != null; -} - -/** - * Render the given attributes object. - * - * @param {Object} obj - * @param {Object} escaped - * @return {String} - * @api private - */ - -exports.attrs = function attrs(obj, escaped){ - var buf = [] - , terse = obj.terse; - - delete obj.terse; - var keys = Object.keys(obj) - , len = keys.length; - - if (len) { - buf.push(''); - for (var i = 0; i < len; ++i) { - var key = keys[i] - , val = obj[key]; - - if ('boolean' == typeof val || null == val) { - if (val) { - terse - ? buf.push(key) - : buf.push(key + '="' + key + '"'); - } - } else if (0 == key.indexOf('data') && 'string' != typeof val) { - buf.push(key + "='" + JSON.stringify(val) + "'"); - } else if ('class' == key && Array.isArray(val)) { - buf.push(key + '="' + exports.escape(val.join(' ')) + '"'); - } else if (escaped && escaped[key]) { - buf.push(key + '="' + exports.escape(val) + '"'); - } else { - buf.push(key + '="' + val + '"'); - } - } - } - - return buf.join(' '); -}; - -/** - * Escape the given string of `html`. - * - * @param {String} html - * @return {String} - * @api private - */ - -exports.escape = function escape(html){ - return String(html) - .replace(/&(?!(\w+|\#\d+);)/g, '&') - .replace(//g, '>') - .replace(/"/g, '"'); -}; - -/** - * Re-throw the given `err` in context to the - * the jade in `filename` at the given `lineno`. - * - * @param {Error} err - * @param {String} filename - * @param {String} lineno - * @api private - */ - -exports.rethrow = function rethrow(err, filename, lineno){ - if (!filename) throw err; - - var context = 3 - , str = require('fs').readFileSync(filename, 'utf8') - , lines = str.split('\n') - , start = Math.max(lineno - context, 0) - , end = Math.min(lines.length, lineno + context); - - // Error context - var context = lines.slice(start, end).map(function(line, i){ - var curr = i + start + 1; - return (curr == lineno ? ' > ' : ' ') - + curr - + '| ' - + line; - }).join('\n'); - - // Alter exception message - err.path = filename; - err.message = (filename || 'Jade') + ':' + lineno - + '\n' + context + '\n\n' + err.message; - throw err; -}; - -}); // module: runtime.js - -require.register("self-closing.js", function(module, exports, require){ - -/*! - * Jade - self closing tags - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -module.exports = [ - 'meta' - , 'img' - , 'link' - , 'input' - , 'source' - , 'area' - , 'base' - , 'col' - , 'br' - , 'hr' -]; -}); // module: self-closing.js - -require.register("utils.js", function(module, exports, require){ - -/*! - * Jade - utils - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Convert interpolation in the given string to JavaScript. - * - * @param {String} str - * @return {String} - * @api private - */ - -var interpolate = exports.interpolate = function(str){ - return str.replace(/(_SLASH_)?([#!]){(.*?)}/g, function(str, escape, flag, code){ - code = code - .replace(/\\'/g, "'") - .replace(/_SLASH_/g, '\\'); - - return escape - ? str.slice(7) - : "' + " - + ('!' == flag ? '' : 'escape') - + "((interp = " + code - + ") == null ? '' : interp) + '"; - }); -}; - -/** - * Escape single quotes in `str`. - * - * @param {String} str - * @return {String} - * @api private - */ - -var escape = exports.escape = function(str) { - return str.replace(/'/g, "\\'"); -}; - -/** - * Interpolate, and escape the given `str`. - * - * @param {String} str - * @return {String} - * @api private - */ - -exports.text = function(str){ - return interpolate(escape(str)); -}; - -/** - * Merge `b` into `a`. - * - * @param {Object} a - * @param {Object} b - * @return {Object} - * @api public - */ - -exports.merge = function(a, b) { - for (var key in b) a[key] = b[key]; - return a; -}; - - -}); // module: utils.js - -window.jade = require("jade"); -})(); diff --git a/node_modules/jade/jade.md b/node_modules/jade/jade.md deleted file mode 100644 index 051dc03..0000000 --- a/node_modules/jade/jade.md +++ /dev/null @@ -1,510 +0,0 @@ - -# Jade - - The jade template engine for node.js - -## Synopsis - - jade [-h|--help] [-v|--version] [-o|--obj STR] - [-O|--out DIR] [-p|--path PATH] [-P|--pretty] - [-c|--client] [-D|--no-debug] - -## Examples - - translate jade the templates dir - - $ jade templates - - create {foo,bar}.html - - $ jade {foo,bar}.jade - - jade over stdio - - $ jade < my.jade > my.html - - jade over s - - $ echo "h1 Jade!" | jade - - foo, bar dirs rendering to /tmp - - $ jade foo bar --out /tmp - - compile client-side templates without debugging - instrumentation, making the output javascript - very light-weight. This requires runtime.js - in your projects. - - $ jade --client --no-debug < my.jade - -## Tags - - Tags are simply nested via whitespace, closing - tags defined for you. These indents are called "blocks". - - ul - li - a Foo - li - a Bar - - You may have several tags in one "block": - - ul - li - a Foo - a Bar - a Baz - -## Self-closing Tags - - Some tags are flagged as self-closing by default, such - as `meta`, `link`, and so on. To explicitly self-close - a tag simply append the `/` character: - - foo/ - foo(bar='baz')/ - - Would yield: - - - - -## Attributes - - Tag attributes look similar to HTML, however - the values are regular JavaScript, here are - some examples: - - a(href='google.com') Google - a(class='button', href='google.com') Google - - As mentioned the attribute values are just JavaScript, - this means ternary operations and other JavaScript expressions - work just fine: - - body(class=user.authenticated ? 'authenticated' : 'anonymous') - a(href=user.website || 'http://google.com') - - Multiple lines work too: - - input(type='checkbox', - name='agreement', - checked) - - Multiple lines without the comma work fine: - - input(type='checkbox' - name='agreement' - checked) - - Funky whitespace? fine: - - input( - type='checkbox' - name='agreement' - checked) - -## Boolean attributes - - Boolean attributes are mirrored by Jade, and accept - bools, aka _true_ or _false_. When no value is specified - _true_ is assumed. For example: - - input(type="checkbox", checked) - // => "" - - For example if the checkbox was for an agreement, perhaps `user.agreed` - was _true_ the following would also output 'checked="checked"': - - input(type="checkbox", checked=user.agreed) - -## Class attributes - - The _class_ attribute accepts an array of classes, - this can be handy when generated from a javascript - function etc: - - classes = ['foo', 'bar', 'baz'] - a(class=classes) - // => "" - -## Class literal - - Classes may be defined using a ".CLASSNAME" syntax: - - .button - // => "
    " - - Or chained: - - .large.button - // => "
    " - - The previous defaulted to divs, however you - may also specify the tag type: - - h1.title My Title - // => "

    My Title

    " - -## Id literal - - Much like the class literal there's an id literal: - - #user-1 - // => "
    " - - Again we may specify the tag as well: - - ul#menu - li: a(href='/home') Home - li: a(href='/store') Store - li: a(href='/contact') Contact - - Finally all of these may be used in any combination, - the following are all valid tags: - - a.button#contact(style: 'color: red') Contact - a.button(style: 'color: red')#contact Contact - a(style: 'color: red').button#contact Contact - -## Block expansion - - Jade supports the concept of "block expansion", in which - using a trailing ":" after a tag will inject a block: - - ul - li: a Foo - li: a Bar - li: a Baz - -## Text - - Arbitrary text may follow tags: - - p Welcome to my site - - yields: - -

    Welcome to my site

    - -## Pipe text - - Another form of text is "pipe" text. Pipes act - as the text margin for large bodies of text. - - p - | This is a large - | body of text for - | this tag. - | - | Nothing too - | exciting. - - yields: - -

    This is a large - body of text for - this tag. - - Nothing too - exciting. -

    - - Using pipes we can also specify regular Jade tags - within the text: - - p - | Click to visit - a(href='http://google.com') Google - | if you want. - -## Text only tags - - As an alternative to pipe text you may add - a trailing "." to indicate that the block - contains nothing but plain-text, no tags: - - p. - This is a large - body of text for - this tag. - - Nothing too - exciting. - - Some tags are text-only by default, for example - _script_, _textarea_, and _style_ tags do not - contain nested HTML so Jade implies the trailing ".": - - script - if (foo) { - bar(); - } - - style - body { - padding: 50px; - font: 14px Helvetica; - } - -## Template script tags - - Sometimes it's useful to define HTML in script - tags using Jade, typically for client-side templates. - - To do this simply give the _script_ tag an arbitrary - _type_ attribute such as _text/x-template_: - - script(type='text/template') - h1 Look! - p Jade still works in here! - -## Interpolation - - Both plain-text and piped-text support interpolation, - which comes in two forms, escapes and non-escaped. The - following will output the _user.name_ in the paragraph - but HTML within it will be escaped to prevent XSS attacks: - - p Welcome #{user.name} - - The following syntax is identical however it will _not_ escape - HTML, and should only be used with strings that you trust: - - p Welcome !{user.name} - -## Inline HTML - - Sometimes constructing small inline snippets of HTML - in Jade can be annoying, luckily we can add plain - HTML as well: - - p Welcome #{user.name} - -## Code - - To buffer output with Jade simply use _=_ at the beginning - of a line or after a tag. This method escapes any HTML - present in the string. - - p= user.description - - To buffer output unescaped use the _!=_ variant, but again - be careful of XSS. - - p!= user.description - - The final way to mess with JavaScript code in Jade is the unbuffered - _-_, which can be used for conditionals, defining variables etc: - - - var user = { description: 'foo bar baz' } - #user - - if (user.description) { - h2 Description - p.description= user.description - - } - - When compiled blocks are wrapped in anonymous functions, so the - following is also valid, without braces: - - - var user = { description: 'foo bar baz' } - #user - - if (user.description) - h2 Description - p.description= user.description - - If you really want you could even use `.forEach()` and others: - - - users.forEach(function(user){ - .user - h2= user.name - p User #{user.name} is #{user.age} years old - - }) - - Taking this further Jade provides some syntax for conditionals, - iteration, switch statements etc. Let's look at those next! - -## Assignment - - Jade's first-class assignment is simple, simply use the _=_ - operator and Jade will _var_ it for you. The following are equivalent: - - - var user = { name: 'tobi' } - user = { name: 'tobi' } - -## Conditionals - - Jade's first-class conditional syntax allows for optional - parenthesis, and you may now omit the leading _-_ otherwise - it's identical, still just regular javascript: - - user = { description: 'foo bar baz' } - #user - if user.description - h2 Description - p.description= user.description - - Jade provides the negated version, _unless_ as well, the following - are equivalent: - - - if (!(user.isAnonymous)) - p You're logged in as #{user.name} - - unless user.isAnonymous - p You're logged in as #{user.name} - -## Iteration - - JavaScript's _for_ loops don't look very declarative, so Jade - also provides its own _for_ loop construct, aliased as _each_: - - for user in users - .user - h2= user.name - p user #{user.name} is #{user.age} year old - - As mentioned _each_ is identical: - - each user in users - .user - h2= user.name - - If necessary the index is available as well: - - for user, i in users - .user(class='user-#{i}') - h2= user.name - - Remember, it's just JavaScript: - - ul#letters - for letter in ['a', 'b', 'c'] - li= letter - -## Mixins - - Mixins provide a way to define jade "functions" which "mix in" - their contents when called. This is useful for abstracting - out large fragments of Jade. - - The simplest possible mixin which accepts no arguments might - look like this: - - mixin hello - p Hello - - You use a mixin by placing `+` before the name: - - +hello - - For something a little more dynamic, mixins can take - arguments, the mixin itself is converted to a javascript - function internally: - - mixin hello(user) - p Hello #{user} - - +hello('Tobi') - - Yields: - -

    Hello Tobi

    - - Mixins may optionally take blocks, when a block is passed - its contents becomes the implicit `block` argument. For - example here is a mixin passed a block, and also invoked - without passing a block: - - mixin article(title) - .article - .article-wrapper - h1= title - if block - block - else - p No content provided - - +article('Hello world') - - +article('Hello world') - p This is my - p Amazing article - - yields: - -
    -
    -

    Hello world

    -

    No content provided

    -
    -
    - -
    -
    -

    Hello world

    -

    This is my

    -

    Amazing article

    -
    -
    - - Mixins can even take attributes, just like a tag. When - attributes are passed they become the implicit `attributes` - argument. Individual attributes can be accessed just like - normal object properties: - - mixin centered - .centered(class=attributes.class) - block - - +centered.bold Hello world - - +centered.red - p This is my - p Amazing article - - yields: - -
    Hello world
    -
    -

    This is my

    -

    Amazing article

    -
    - - If you use `attributes` directly, *all* passed attributes - get used: - - mixin link - a.menu(attributes) - block - - +link.highlight(href='#top') Top - +link#sec1.plain(href='#section1') Section 1 - +link#sec2.plain(href='#section2') Section 2 - - yields: - - Top - Section 1 - Section 2 - - If you pass arguments, they must directly follow the mixin: - - mixin list(arr) - if block - .title - block - ul(attributes) - each item in arr - li= item - - +list(['foo', 'bar', 'baz'])(id='myList', class='bold') - - yields: - -
      -
    • foo
    • -
    • bar
    • -
    • baz
    • -
    diff --git a/node_modules/jade/jade.min.js b/node_modules/jade/jade.min.js deleted file mode 100644 index 93f3b3d..0000000 --- a/node_modules/jade/jade.min.js +++ /dev/null @@ -1,2 +0,0 @@ -(function(){function require(p){var path=require.resolve(p),mod=require.modules[path];if(!mod)throw new Error('failed to require "'+p+'"');return mod.exports||(mod.exports={},mod.call(mod.exports,mod,mod.exports,require.relative(path))),mod.exports}require.modules={},require.resolve=function(path){var orig=path,reg=path+".js",index=path+"/index.js";return require.modules[reg]&®||require.modules[index]&&index||orig},require.register=function(path,fn){require.modules[path]=fn},require.relative=function(parent){return function(p){if("."!=p.charAt(0))return require(p);var path=parent.split("/"),segs=p.split("/");path.pop();for(var i=0;i/g,">").replace(/"/g,""")}var nodes=require("./nodes"),filters=require("./filters"),doctypes=require("./doctypes"),selfClosing=require("./self-closing"),runtime=require("./runtime"),utils=require("./utils");Object.keys||(Object.keys=function(obj){var arr=[];for(var key in obj)obj.hasOwnProperty(key)&&arr.push(key);return arr}),String.prototype.trimLeft||(String.prototype.trimLeft=function(){return this.replace(/^\s+/,"")});var Compiler=module.exports=function Compiler(node,options){this.options=options=options||{},this.node=node,this.hasCompiledDoctype=!1,this.hasCompiledTag=!1,this.pp=options.pretty||!1,this.debug=!1!==options.compileDebug,this.indents=0,this.parentIndents=0,options.doctype&&this.setDoctype(options.doctype)};Compiler.prototype={compile:function(){return this.buf=["var interp;"],this.pp&&this.buf.push("var __indent = [];"),this.lastBufferedIdx=-1,this.visit(this.node),this.buf.join("\n")},setDoctype:function(name){name=name&&name.toLowerCase()||"default",this.doctype=doctypes[name]||"",this.terse=this.doctype.toLowerCase()=="",this.xml=0==this.doctype.indexOf("1&&!escape&&block.nodes[0].isText&&block.nodes[1].isText&&this.prettyIndent(1,!0);for(var i=0;i0&&!escape&&block.nodes[i].isText&&block.nodes[i-1].isText&&this.prettyIndent(1,!1),this.visit(block.nodes[i]),block.nodes[i+1]&&block.nodes[i].isText&&block.nodes[i+1].isText&&this.buffer("\\n")},visitDoctype:function(doctype){doctype&&(doctype.val||!this.doctype)&&this.setDoctype(doctype.val||"default"),this.doctype&&this.buffer(this.doctype),this.hasCompiledDoctype=!0},visitMixin:function(mixin){var name=mixin.name.replace(/-/g,"_")+"_mixin",args=mixin.args||"",block=mixin.block,attrs=mixin.attrs,pp=this.pp;if(mixin.call){pp&&this.buf.push("__indent.push('"+Array(this.indents+1).join(" ")+"');");if(block||attrs.length){this.buf.push(name+".call({");if(block){this.buf.push("block: function(){"),this.parentIndents++;var _indents=this.indents;this.indents=0,this.visit(mixin.block),this.indents=_indents,this.parentIndents--,attrs.length?this.buf.push("},"):this.buf.push("}")}if(attrs.length){var val=this.attrs(attrs);val.inherits?this.buf.push("attributes: merge({"+val.buf+"}, attributes), escaped: merge("+val.escaped+", escaped, true)"):this.buf.push("attributes: {"+val.buf+"}, escaped: "+val.escaped)}args?this.buf.push("}, "+args+");"):this.buf.push("});")}else this.buf.push(name+"("+args+");");pp&&this.buf.push("__indent.pop();")}else this.buf.push("var "+name+" = function("+args+"){"),this.buf.push("var block = this.block, attributes = this.attributes || {}, escaped = this.escaped || {};"),this.parentIndents++,this.visit(block),this.parentIndents--,this.buf.push("};")},visitTag:function(tag){this.indents++;var name=tag.name,pp=this.pp;tag.buffer&&(name="' + ("+name+") + '"),this.hasCompiledTag||(!this.hasCompiledDoctype&&"html"==name&&this.visitDoctype(),this.hasCompiledTag=!0),pp&&!tag.isInline()&&this.prettyIndent(0,!0),(~selfClosing.indexOf(name)||tag.selfClosing)&&!this.xml?(this.buffer("<"+name),this.visitAttributes(tag.attrs),this.terse?this.buffer(">"):this.buffer("/>")):(tag.attrs.length?(this.buffer("<"+name),tag.attrs.length&&this.visitAttributes(tag.attrs),this.buffer(">")):this.buffer("<"+name+">"),tag.code&&this.visitCode(tag.code),this.escape="pre"==tag.name,this.visit(tag.block),pp&&!tag.isInline()&&"pre"!=tag.name&&!tag.canInline()&&this.prettyIndent(0,!0),this.buffer("")),this.indents--},visitFilter:function(filter){var fn=filters[filter.name];if(!fn)throw filter.isASTFilter?new Error('unknown ast filter "'+filter.name+':"'):new Error('unknown filter ":'+filter.name+'"');if(filter.isASTFilter)this.buf.push(fn(filter.block,this,filter.attrs));else{var text=filter.block.nodes.map(function(node){return node.val}).join("\n");filter.attrs=filter.attrs||{},filter.attrs.filename=this.options.filename,this.buffer(utils.text(fn(text,filter.attrs)))}},visitText:function(text){text=utils.text(text.val.replace(/\\/g,"_SLASH_")),this.escape&&(text=escape(text)),text=text.replace(/_SLASH_/g,"\\\\"),this.buffer(text)},visitComment:function(comment){if(!comment.buffer)return;this.pp&&this.prettyIndent(1,!0),this.buffer("")},visitBlockComment:function(comment){if(!comment.buffer)return;0==comment.val.trim().indexOf("if")?(this.buffer("")):(this.buffer(""))},visitCode:function(code){if(code.buffer){var val=code.val.trimLeft();this.buf.push("var __val__ = "+val),val='null == __val__ ? "" : __val__',code.escape&&(val="escape("+val+")"),this.buf.push("buf.push("+val+");")}else this.buf.push(code.val);code.block&&(code.buffer||this.buf.push("{"),this.visit(code.block),code.buffer||this.buf.push("}"))},visitEach:function(each){this.buf.push("// iterate "+each.obj+"\n"+";(function(){\n"+" if ('number' == typeof "+each.obj+".length) {\n"),each.alternative&&this.buf.push(" if ("+each.obj+".length) {"),this.buf.push(" for (var "+each.key+" = 0, $$l = "+each.obj+".length; "+each.key+" < $$l; "+each.key+"++) {\n"+" var "+each.val+" = "+each.obj+"["+each.key+"];\n"),this.visit(each.block),this.buf.push(" }\n"),each.alternative&&(this.buf.push(" } else {"),this.visit(each.alternative),this.buf.push(" }")),this.buf.push(" } else {\n var $$l = 0;\n for (var "+each.key+" in "+each.obj+") {\n"+" $$l++;"+" if ("+each.obj+".hasOwnProperty("+each.key+")){"+" var "+each.val+" = "+each.obj+"["+each.key+"];\n"),this.visit(each.block),this.buf.push(" }\n"),this.buf.push(" }\n"),each.alternative&&(this.buf.push(" if ($$l === 0) {"),this.visit(each.alternative),this.buf.push(" }")),this.buf.push(" }\n}).call(this);\n")},visitAttributes:function(attrs){var val=this.attrs(attrs);val.inherits?this.buf.push("buf.push(attrs(merge({ "+val.buf+" }, attributes), merge("+val.escaped+", escaped, true)));"):val.constant?(eval("var buf={"+val.buf+"};"),this.buffer(runtime.attrs(buf,JSON.parse(val.escaped)),!0)):this.buf.push("buf.push(attrs({ "+val.buf+" }, "+val.escaped+"));")},attrs:function(attrs){var buf=[],classes=[],escaped={},constant=attrs.every(function(attr){return isConstant(attr.val)}),inherits=!1;return this.terse&&buf.push("terse: true"),attrs.forEach(function(attr){if(attr.name=="attributes")return inherits=!0;escaped[attr.name]=attr.escaped;if(attr.name=="class")classes.push("("+attr.val+")");else{var pair="'"+attr.name+"':("+attr.val+")";buf.push(pair)}}),classes.length&&(classes=classes.join(" + ' ' + "),buf.push("class: "+classes)),{buf:buf.join(", ").replace("class:",'"class":'),escaped:JSON.stringify(escaped),inherits:inherits,constant:constant}}}}),require.register("doctypes.js",function(module,exports,require){module.exports={5:"","default":"",xml:'',transitional:'',strict:'',frameset:'',1.1:'',basic:'',mobile:''}}),require.register("filters.js",function(module,exports,require){module.exports={cdata:function(str){return""},sass:function(str){str=str.replace(/\\n/g,"\n");var sass=require("sass").render(str).replace(/\n/g,"\\n");return'"},stylus:function(str,options){var ret;str=str.replace(/\\n/g,"\n");var stylus=require("stylus");return stylus(str,options).render(function(err,css){if(err)throw err;ret=css.replace(/\n/g,"\\n")}),'"},less:function(str){var ret;return str=str.replace(/\\n/g,"\n"),require("less").render(str,function(err,css){if(err)throw err;ret='"}),ret},markdown:function(str){var md;try{md=require("markdown")}catch(err){try{md=require("discount")}catch(err){try{md=require("markdown-js")}catch(err){try{md=require("marked")}catch(err){throw new Error("Cannot find markdown library, install markdown, discount, or marked.")}}}}return str=str.replace(/\\n/g,"\n"),md.parse(str).replace(/\n/g,"\\n").replace(/'/g,"'")},coffeescript:function(str){var js=require("coffee-script").compile(str).replace(/\\/g,"\\\\").replace(/\n/g,"\\n");return'"}}}),require.register("inline-tags.js",function(module,exports,require){module.exports=["a","abbr","acronym","b","br","code","em","font","i","img","ins","kbd","map","samp","small","span","strong","sub","sup"]}),require.register("jade.js",function(module,exports,require){function parse(str,options){try{var parser=new Parser(str,options.filename,options),compiler=new(options.compiler||Compiler)(parser.parse(),options),js=compiler.compile();return options.debug&&console.error("\nCompiled Function:\n\n%s",js.replace(/^/gm," ")),"var buf = [];\n"+(options.self?"var self = locals || {};\n"+js:"with (locals || {}) {\n"+js+"\n}\n")+'return buf.join("");'}catch(err){parser=parser.context(),runtime.rethrow(err,parser.filename,parser.lexer.lineno)}}function stripBOM(str){return 65279==str.charCodeAt(0)?str.substring(1):str}var Parser=require("./parser"),Lexer=require("./lexer"),Compiler=require("./compiler"),runtime=require("./runtime");exports.version="0.27.6",exports.selfClosing=require("./self-closing"),exports.doctypes=require("./doctypes"),exports.filters=require("./filters"),exports.utils=require("./utils"),exports.Compiler=Compiler,exports.Parser=Parser,exports.Lexer=Lexer,exports.nodes=require("./nodes"),exports.runtime=runtime,exports.cache={},exports.compile=function(str,options){var options=options||{},client=options.client,filename=options.filename?JSON.stringify(options.filename):"undefined",fn;return str=stripBOM(String(str)),options.compileDebug!==!1?fn=["var __jade = [{ lineno: 1, filename: "+filename+" }];","try {",parse(str,options),"} catch (err) {"," rethrow(err, __jade[0].filename, __jade[0].lineno);","}"].join("\n"):fn=parse(str,options),client&&(fn="attrs = attrs || jade.attrs; escape = escape || jade.escape; rethrow = rethrow || jade.rethrow; merge = merge || jade.merge;\n"+fn),fn=new Function("locals, attrs, escape, rethrow, merge",fn),client?fn:function(locals){return fn(locals,runtime.attrs,runtime.escape,runtime.rethrow,runtime.merge)}},exports.render=function(str,options,fn){"function"==typeof options&&(fn=options,options={});if(options.cache&&!options.filename)return fn(new Error('the "filename" option is required for caching'));try{var path=options.filename,tmpl=options.cache?exports.cache[path]||(exports.cache[path]=exports.compile(str,options)):exports.compile(str,options);fn(null,tmpl(options))}catch(err){fn(err)}},exports.renderFile=function(path,options,fn){var key=path+":string";"function"==typeof options&&(fn=options,options={});try{options.filename=path;var str=options.cache?exports.cache[key]||(exports.cache[key]=fs.readFileSync(path,"utf8")):fs.readFileSync(path,"utf8");exports.render(str,options,fn)}catch(err){fn(err)}},exports.__express=exports.renderFile}),require.register("lexer.js",function(module,exports,require){var utils=require("./utils"),Lexer=module.exports=function Lexer(str,options){options=options||{},this.input=str.replace(/\r\n|\r/g,"\n"),this.colons=options.colons,this.deferredTokens=[],this.lastIndents=0,this.lineno=1,this.stash=[],this.indentStack=[],this.indentRe=null,this.pipeless=!1};Lexer.prototype={tok:function(type,val){return{type:type,line:this.lineno,val:val}},consume:function(len){this.input=this.input.substr(len)},scan:function(regexp,type){var captures;if(captures=regexp.exec(this.input))return this.consume(captures[0].length),this.tok(type,captures[1])},defer:function(tok){this.deferredTokens.push(tok)},lookahead:function(n){var fetch=n-this.stash.length;while(fetch-->0)this.stash.push(this.next());return this.stash[--n]},indexOfDelimiters:function(start,end){var str=this.input,nstart=0,nend=0,pos=0;for(var i=0,len=str.length;iindents)this.stash.push(this.tok("outdent")),this.indentStack.shift();tok=this.stash.pop()}else indents&&indents!=this.indentStack[0]?(this.indentStack.unshift(indents),tok=this.tok("indent",indents)):tok=this.tok("newline");return tok}},pipelessText:function(){if(this.pipeless){if("\n"==this.input[0])return;var i=this.input.indexOf("\n");-1==i&&(i=this.input.length);var str=this.input.substr(0,i);return this.consume(str.length),this.tok("text",str)}},colon:function(){return this.scan(/^: */,":")},advance:function(){return this.stashed()||this.next()},next:function(){return this.deferred()||this.blank()||this.eos()||this.pipelessText()||this.yield()||this.doctype()||this.interpolation()||this["case"]()||this.when()||this["default"]()||this["extends"]()||this.append()||this.prepend()||this.block()||this.include()||this.mixin()||this.call()||this.conditional()||this.each()||this["while"]()||this.assignment()||this.tag()||this.filter()||this.code()||this.id()||this.className()||this.attrs()||this.indent()||this.comment()||this.colon()||this.text()}}}),require.register("nodes/attrs.js",function(module,exports,require){var Node=require("./node"),Block=require("./block"),Attrs=module.exports=function Attrs(){this.attrs=[]};Attrs.prototype=new Node,Attrs.prototype.constructor=Attrs,Attrs.prototype.setAttribute=function(name,val,escaped){return this.attrs.push({name:name,val:val,escaped:escaped}),this},Attrs.prototype.removeAttribute=function(name){for(var i=0,len=this.attrs.length;i/g,">").replace(/"/g,""")},exports.rethrow=function rethrow(err,filename,lineno){if(!filename)throw err;var context=3,str=require("fs").readFileSync(filename,"utf8"),lines=str.split("\n"),start=Math.max(lineno-context,0),end=Math.min(lines.length,lineno+context),context=lines.slice(start,end).map(function(line,i){var curr=i+start+1;return(curr==lineno?" > ":" ")+curr+"| "+line}).join("\n");throw err.path=filename,err.message=(filename||"Jade")+":"+lineno+"\n"+context+"\n\n"+err.message,err}}),require.register("self-closing.js",function(module,exports,require){module.exports=["meta","img","link","input","source","area","base","col","br","hr"]}),require.register("utils.js",function(module,exports,require){var interpolate=exports.interpolate=function(str){return str.replace(/(_SLASH_)?([#!]){(.*?)}/g,function(str,escape,flag,code){return code=code.replace(/\\'/g,"'").replace(/_SLASH_/g,"\\"),escape?str.slice(7):"' + "+("!"==flag?"":"escape")+"((interp = "+code+") == null ? '' : interp) + '"})},escape=exports.escape=function(str){return str.replace(/'/g,"\\'")};exports.text=function(str){return interpolate(escape(str))},exports.merge=function(a,b){for(var key in b)a[key]=b[key];return a}}),window.jade=require("jade")})(); \ No newline at end of file diff --git a/node_modules/jade/lib/compiler.js b/node_modules/jade/lib/compiler.js deleted file mode 100644 index bb56142..0000000 --- a/node_modules/jade/lib/compiler.js +++ /dev/null @@ -1,655 +0,0 @@ - -/*! - * Jade - Compiler - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var nodes = require('./nodes') - , filters = require('./filters') - , doctypes = require('./doctypes') - , selfClosing = require('./self-closing') - , runtime = require('./runtime') - , utils = require('./utils'); - -// if browser -// -// if (!Object.keys) { -// Object.keys = function(obj){ -// var arr = []; -// for (var key in obj) { -// if (obj.hasOwnProperty(key)) { -// arr.push(key); -// } -// } -// return arr; -// } -// } -// -// if (!String.prototype.trimLeft) { -// String.prototype.trimLeft = function(){ -// return this.replace(/^\s+/, ''); -// } -// } -// -// end - - -/** - * Initialize `Compiler` with the given `node`. - * - * @param {Node} node - * @param {Object} options - * @api public - */ - -var Compiler = module.exports = function Compiler(node, options) { - this.options = options = options || {}; - this.node = node; - this.hasCompiledDoctype = false; - this.hasCompiledTag = false; - this.pp = options.pretty || false; - this.debug = false !== options.compileDebug; - this.indents = 0; - this.parentIndents = 0; - if (options.doctype) this.setDoctype(options.doctype); -}; - -/** - * Compiler prototype. - */ - -Compiler.prototype = { - - /** - * Compile parse tree to JavaScript. - * - * @api public - */ - - compile: function(){ - this.buf = ['var interp;']; - if (this.pp) this.buf.push("var __indent = [];"); - this.lastBufferedIdx = -1; - this.visit(this.node); - return this.buf.join('\n'); - }, - - /** - * Sets the default doctype `name`. Sets terse mode to `true` when - * html 5 is used, causing self-closing tags to end with ">" vs "/>", - * and boolean attributes are not mirrored. - * - * @param {string} name - * @api public - */ - - setDoctype: function(name){ - name = (name && name.toLowerCase()) || 'default'; - this.doctype = doctypes[name] || ''; - this.terse = this.doctype.toLowerCase() == ''; - this.xml = 0 == this.doctype.indexOf(' 1 && !escape && block.nodes[0].isText && block.nodes[1].isText) - this.prettyIndent(1, true); - - for (var i = 0; i < len; ++i) { - // Pretty print text - if (pp && i > 0 && !escape && block.nodes[i].isText && block.nodes[i-1].isText) - this.prettyIndent(1, false); - - this.visit(block.nodes[i]); - // Multiple text nodes are separated by newlines - if (block.nodes[i+1] && block.nodes[i].isText && block.nodes[i+1].isText) - this.buffer('\\n'); - } - }, - - /** - * Visit `doctype`. Sets terse mode to `true` when html 5 - * is used, causing self-closing tags to end with ">" vs "/>", - * and boolean attributes are not mirrored. - * - * @param {Doctype} doctype - * @api public - */ - - visitDoctype: function(doctype){ - if (doctype && (doctype.val || !this.doctype)) { - this.setDoctype(doctype.val || 'default'); - } - - if (this.doctype) this.buffer(this.doctype); - this.hasCompiledDoctype = true; - }, - - /** - * Visit `mixin`, generating a function that - * may be called within the template. - * - * @param {Mixin} mixin - * @api public - */ - - visitMixin: function(mixin){ - var name = mixin.name.replace(/-/g, '_') + '_mixin' - , args = mixin.args || '' - , block = mixin.block - , attrs = mixin.attrs - , pp = this.pp; - - if (mixin.call) { - if (pp) this.buf.push("__indent.push('" + Array(this.indents + 1).join(' ') + "');") - if (block || attrs.length) { - - this.buf.push(name + '.call({'); - - if (block) { - this.buf.push('block: function(){'); - - // Render block with no indents, dynamically added when rendered - this.parentIndents++; - var _indents = this.indents; - this.indents = 0; - this.visit(mixin.block); - this.indents = _indents; - this.parentIndents--; - - if (attrs.length) { - this.buf.push('},'); - } else { - this.buf.push('}'); - } - } - - if (attrs.length) { - var val = this.attrs(attrs); - if (val.inherits) { - this.buf.push('attributes: merge({' + val.buf - + '}, attributes), escaped: merge(' + val.escaped + ', escaped, true)'); - } else { - this.buf.push('attributes: {' + val.buf + '}, escaped: ' + val.escaped); - } - } - - if (args) { - this.buf.push('}, ' + args + ');'); - } else { - this.buf.push('});'); - } - - } else { - this.buf.push(name + '(' + args + ');'); - } - if (pp) this.buf.push("__indent.pop();") - } else { - this.buf.push('var ' + name + ' = function(' + args + '){'); - this.buf.push('var block = this.block, attributes = this.attributes || {}, escaped = this.escaped || {};'); - this.parentIndents++; - this.visit(block); - this.parentIndents--; - this.buf.push('};'); - } - }, - - /** - * Visit `tag` buffering tag markup, generating - * attributes, visiting the `tag`'s code and block. - * - * @param {Tag} tag - * @api public - */ - - visitTag: function(tag){ - this.indents++; - var name = tag.name - , pp = this.pp; - - if (tag.buffer) name = "' + (" + name + ") + '"; - - if (!this.hasCompiledTag) { - if (!this.hasCompiledDoctype && 'html' == name) { - this.visitDoctype(); - } - this.hasCompiledTag = true; - } - - // pretty print - if (pp && !tag.isInline()) - this.prettyIndent(0, true); - - if ((~selfClosing.indexOf(name) || tag.selfClosing) && !this.xml) { - this.buffer('<' + name); - this.visitAttributes(tag.attrs); - this.terse - ? this.buffer('>') - : this.buffer('/>'); - } else { - // Optimize attributes buffering - if (tag.attrs.length) { - this.buffer('<' + name); - if (tag.attrs.length) this.visitAttributes(tag.attrs); - this.buffer('>'); - } else { - this.buffer('<' + name + '>'); - } - if (tag.code) this.visitCode(tag.code); - this.escape = 'pre' == tag.name; - this.visit(tag.block); - - // pretty print - if (pp && !tag.isInline() && 'pre' != tag.name && !tag.canInline()) - this.prettyIndent(0, true); - - this.buffer(''); - } - this.indents--; - }, - - /** - * Visit `filter`, throwing when the filter does not exist. - * - * @param {Filter} filter - * @api public - */ - - visitFilter: function(filter){ - var fn = filters[filter.name]; - - // unknown filter - if (!fn) throw new Error('unknown filter ":' + filter.name + '"'); - - var text = filter.block.nodes.map( - function(node){ return node.val; } - ).join('\n'); - filter.attrs = filter.attrs || {}; - filter.attrs.filename = this.options.filename; - this.buffer(utils.text(fn(text, filter.attrs))); - }, - - /** - * Visit `text` node. - * - * @param {Text} text - * @api public - */ - - visitText: function(text){ - text = utils.text(text.val.replace(/\\/g, '_SLASH_')); - if (this.escape) text = escape(text); - text = text.replace(/_SLASH_/g, '\\\\'); - this.buffer(text); - }, - - /** - * Visit a `comment`, only buffering when the buffer flag is set. - * - * @param {Comment} comment - * @api public - */ - - visitComment: function(comment){ - if (!comment.buffer) return; - if (this.pp) this.prettyIndent(1, true); - this.buffer(''); - }, - - /** - * Visit a `BlockComment`. - * - * @param {Comment} comment - * @api public - */ - - visitBlockComment: function(comment){ - if (!comment.buffer) return; - if (0 == comment.val.trim().indexOf('if')) { - this.buffer(''); - } else { - this.buffer(''); - } - }, - - /** - * Visit `code`, respecting buffer / escape flags. - * If the code is followed by a block, wrap it in - * a self-calling function. - * - * @param {Code} code - * @api public - */ - - visitCode: function(code){ - // Wrap code blocks with {}. - // we only wrap unbuffered code blocks ATM - // since they are usually flow control - - // Buffer code - if (code.buffer) { - var val = code.val.trimLeft(); - this.buf.push('var __val__ = ' + val); - val = 'null == __val__ ? "" : __val__'; - if (code.escape) val = 'escape(' + val + ')'; - this.buf.push("buf.push(" + val + ");"); - } else { - this.buf.push(code.val); - } - - // Block support - if (code.block) { - if (!code.buffer) this.buf.push('{'); - this.visit(code.block); - if (!code.buffer) this.buf.push('}'); - } - }, - - /** - * Visit `each` block. - * - * @param {Each} each - * @api public - */ - - visitEach: function(each){ - this.buf.push('' - + '// iterate ' + each.obj + '\n' - + ';(function(){\n' - + ' if (\'number\' == typeof ' + each.obj + '.length) {\n'); - - if (each.alternative) { - this.buf.push(' if (' + each.obj + '.length) {'); - } - - this.buf.push('' - + ' for (var ' + each.key + ' = 0, $$l = ' + each.obj + '.length; ' + each.key + ' < $$l; ' + each.key + '++) {\n' - + ' var ' + each.val + ' = ' + each.obj + '[' + each.key + '];\n'); - - this.visit(each.block); - - this.buf.push(' }\n'); - - if (each.alternative) { - this.buf.push(' } else {'); - this.visit(each.alternative); - this.buf.push(' }'); - } - - this.buf.push('' - + ' } else {\n' - + ' var $$l = 0;\n' - + ' for (var ' + each.key + ' in ' + each.obj + ') {\n' - + ' $$l++;' - // if browser - // + ' if (' + each.obj + '.hasOwnProperty(' + each.key + ')){' - // end - + ' var ' + each.val + ' = ' + each.obj + '[' + each.key + '];\n'); - - this.visit(each.block); - - // if browser - // this.buf.push(' }\n'); - // end - - this.buf.push(' }\n'); - if (each.alternative) { - this.buf.push(' if ($$l === 0) {'); - this.visit(each.alternative); - this.buf.push(' }'); - } - this.buf.push(' }\n}).call(this);\n'); - }, - - /** - * Visit `attrs`. - * - * @param {Array} attrs - * @api public - */ - - visitAttributes: function(attrs){ - var val = this.attrs(attrs); - if (val.inherits) { - this.buf.push("buf.push(attrs(merge({ " + val.buf + - " }, attributes), merge(" + val.escaped + ", escaped, true)));"); - } else if (val.constant) { - eval('var buf={' + val.buf + '};'); - this.buffer(runtime.attrs(buf, JSON.parse(val.escaped)), true); - } else { - this.buf.push("buf.push(attrs({ " + val.buf + " }, " + val.escaped + "));"); - } - }, - - /** - * Compile attributes. - */ - - attrs: function(attrs){ - var buf = [] - , classes = [] - , escaped = {} - , constant = attrs.every(function(attr){ return isConstant(attr.val) }) - , inherits = false; - - if (this.terse) buf.push('terse: true'); - - attrs.forEach(function(attr){ - if (attr.name == 'attributes') return inherits = true; - escaped[attr.name] = attr.escaped; - if (attr.name == 'class') { - classes.push('(' + attr.val + ')'); - } else { - var pair = "'" + attr.name + "':(" + attr.val + ')'; - buf.push(pair); - } - }); - - if (classes.length) { - classes = classes.join(" + ' ' + "); - buf.push('"class": ' + classes); - } - - return { - buf: buf.join(', '), - escaped: JSON.stringify(escaped), - inherits: inherits, - constant: constant - }; - } -}; - -/** - * Check if expression can be evaluated to a constant - * - * @param {String} expression - * @return {Boolean} - * @api private - */ - -function isConstant(val){ - // Check strings/literals - if (/^ *("([^"\\]*(\\.[^"\\]*)*)"|'([^'\\]*(\\.[^'\\]*)*)'|true|false|null|undefined) *$/i.test(val)) - return true; - - // Check numbers - if (!isNaN(Number(val))) - return true; - - // Check arrays - var matches; - if (matches = /^ *\[(.*)\] *$/.exec(val)) - return matches[1].split(',').every(isConstant); - - return false; -} - -/** - * Escape the given string of `html`. - * - * @param {String} html - * @return {String} - * @api private - */ - -function escape(html){ - return String(html) - .replace(/&(?!\w+;)/g, '&') - .replace(//g, '>') - .replace(/"/g, '"'); -} diff --git a/node_modules/jade/lib/doctypes.js b/node_modules/jade/lib/doctypes.js deleted file mode 100644 index e87ca1e..0000000 --- a/node_modules/jade/lib/doctypes.js +++ /dev/null @@ -1,18 +0,0 @@ - -/*! - * Jade - doctypes - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -module.exports = { - '5': '' - , 'default': '' - , 'xml': '' - , 'transitional': '' - , 'strict': '' - , 'frameset': '' - , '1.1': '' - , 'basic': '' - , 'mobile': '' -}; \ No newline at end of file diff --git a/node_modules/jade/lib/filters.js b/node_modules/jade/lib/filters.js deleted file mode 100644 index d633559..0000000 --- a/node_modules/jade/lib/filters.js +++ /dev/null @@ -1,105 +0,0 @@ - -/*! - * Jade - filters - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Wrap text with CDATA block. - */ - -exports.cdata = function(str){ - return ''; -}; - -/** - * Wrap text in script tags. - */ - -exports.js = function(str){ - return ''; -}; - -/** - * Wrap text in style tags. - */ - -exports.css = function(str){ - return ''; -}; - -/** - * Transform stylus to css, wrapped in style tags. - */ - -exports.stylus = function(str, options){ - var ret; - str = str.replace(/\\n/g, '\n'); - var stylus = require('stylus'); - stylus(str, options).render(function(err, css){ - if (err) throw err; - ret = css.replace(/\n/g, '\\n'); - }); - return ''; -}; - -/** - * Transform less to css, wrapped in style tags. - */ - -exports.less = function(str){ - var ret; - str = str.replace(/\\n/g, '\n'); - require('less').render(str, function(err, css){ - if (err) throw err; - ret = ''; - }); - return ret; -}; - -/** - * Transform markdown to html. - */ - -exports.markdown = function(str){ - var md; - - // support markdown / discount - try { - md = require('markdown'); - } catch (err){ - try { - md = require('discount'); - } catch (err) { - try { - md = require('markdown-js'); - } catch (err) { - try { - md = require('marked'); - } catch (err) { - throw new - Error('Cannot find markdown library, install markdown, discount, or marked.'); - } - } - } - } - - str = str.replace(/\\n/g, '\n'); - return md.parse(str).replace(/\n/g, '\\n').replace(/'/g,'''); -}; - -/** - * Transform coffeescript to javascript. - */ - -exports.coffeescript = function(str){ - var js = require('coffee-script').compile(str).replace(/\\/g, '\\\\').replace(/\n/g, '\\n'); - return ''; -}; - -// aliases - -exports.md = exports.markdown; -exports.styl = exports.stylus; -exports.coffee = exports.coffeescript; diff --git a/node_modules/jade/lib/inline-tags.js b/node_modules/jade/lib/inline-tags.js deleted file mode 100644 index 491de0b..0000000 --- a/node_modules/jade/lib/inline-tags.js +++ /dev/null @@ -1,28 +0,0 @@ - -/*! - * Jade - inline tags - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -module.exports = [ - 'a' - , 'abbr' - , 'acronym' - , 'b' - , 'br' - , 'code' - , 'em' - , 'font' - , 'i' - , 'img' - , 'ins' - , 'kbd' - , 'map' - , 'samp' - , 'small' - , 'span' - , 'strong' - , 'sub' - , 'sup' -]; \ No newline at end of file diff --git a/node_modules/jade/lib/jade.js b/node_modules/jade/lib/jade.js deleted file mode 100644 index d8330cc..0000000 --- a/node_modules/jade/lib/jade.js +++ /dev/null @@ -1,253 +0,0 @@ -/*! - * Jade - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Parser = require('./parser') - , Lexer = require('./lexer') - , Compiler = require('./compiler') - , runtime = require('./runtime') -// if node - , fs = require('fs'); -// end - -/** - * Library version. - */ - -exports.version = '0.28.2'; - -/** - * Expose self closing tags. - */ - -exports.selfClosing = require('./self-closing'); - -/** - * Default supported doctypes. - */ - -exports.doctypes = require('./doctypes'); - -/** - * Text filters. - */ - -exports.filters = require('./filters'); - -/** - * Utilities. - */ - -exports.utils = require('./utils'); - -/** - * Expose `Compiler`. - */ - -exports.Compiler = Compiler; - -/** - * Expose `Parser`. - */ - -exports.Parser = Parser; - -/** - * Expose `Lexer`. - */ - -exports.Lexer = Lexer; - -/** - * Nodes. - */ - -exports.nodes = require('./nodes'); - -/** - * Jade runtime helpers. - */ - -exports.runtime = runtime; - -/** - * Template function cache. - */ - -exports.cache = {}; - -/** - * Parse the given `str` of jade and return a function body. - * - * @param {String} str - * @param {Object} options - * @return {String} - * @api private - */ - -function parse(str, options){ - try { - // Parse - var parser = new Parser(str, options.filename, options); - - // Compile - var compiler = new (options.compiler || Compiler)(parser.parse(), options) - , js = compiler.compile(); - - // Debug compiler - if (options.debug) { - console.error('\nCompiled Function:\n\n\033[90m%s\033[0m', js.replace(/^/gm, ' ')); - } - - return '' - + 'var buf = [];\n' - + (options.self - ? 'var self = locals || {};\n' + js - : 'with (locals || {}) {\n' + js + '\n}\n') - + 'return buf.join("");'; - } catch (err) { - parser = parser.context(); - runtime.rethrow(err, parser.filename, parser.lexer.lineno); - } -} - -/** - * Strip any UTF-8 BOM off of the start of `str`, if it exists. - * - * @param {String} str - * @return {String} - * @api private - */ - -function stripBOM(str){ - return 0xFEFF == str.charCodeAt(0) - ? str.substring(1) - : str; -} - -/** - * Compile a `Function` representation of the given jade `str`. - * - * Options: - * - * - `compileDebug` when `false` debugging code is stripped from the compiled template - * - `client` when `true` the helper functions `escape()` etc will reference `jade.escape()` - * for use with the Jade client-side runtime.js - * - * @param {String} str - * @param {Options} options - * @return {Function} - * @api public - */ - -exports.compile = function(str, options){ - var options = options || {} - , client = options.client - , filename = options.filename - ? JSON.stringify(options.filename) - : 'undefined' - , fn; - - str = stripBOM(String(str)); - - if (options.compileDebug !== false) { - fn = [ - 'var __jade = [{ lineno: 1, filename: ' + filename + ' }];' - , 'try {' - , parse(str, options) - , '} catch (err) {' - , ' rethrow(err, __jade[0].filename, __jade[0].lineno);' - , '}' - ].join('\n'); - } else { - fn = parse(str, options); - } - - if (client) { - fn = 'attrs = attrs || jade.attrs; escape = escape || jade.escape; rethrow = rethrow || jade.rethrow; merge = merge || jade.merge;\n' + fn; - } - - fn = new Function('locals, attrs, escape, rethrow, merge', fn); - - if (client) return fn; - - return function(locals){ - return fn(locals, runtime.attrs, runtime.escape, runtime.rethrow, runtime.merge); - }; -}; - -/** - * Render the given `str` of jade and invoke - * the callback `fn(err, str)`. - * - * Options: - * - * - `cache` enable template caching - * - `filename` filename required for `include` / `extends` and caching - * - * @param {String} str - * @param {Object|Function} options or fn - * @param {Function} fn - * @api public - */ - -exports.render = function(str, options, fn){ - // swap args - if ('function' == typeof options) { - fn = options, options = {}; - } - - // cache requires .filename - if (options.cache && !options.filename) { - return fn(new Error('the "filename" option is required for caching')); - } - - try { - var path = options.filename; - var tmpl = options.cache - ? exports.cache[path] || (exports.cache[path] = exports.compile(str, options)) - : exports.compile(str, options); - fn(null, tmpl(options)); - } catch (err) { - fn(err); - } -}; - -/** - * Render a Jade file at the given `path` and callback `fn(err, str)`. - * - * @param {String} path - * @param {Object|Function} options or callback - * @param {Function} fn - * @api public - */ - -exports.renderFile = function(path, options, fn){ - var key = path + ':string'; - - if ('function' == typeof options) { - fn = options, options = {}; - } - - try { - options.filename = path; - var str = options.cache - ? exports.cache[key] || (exports.cache[key] = fs.readFileSync(path, 'utf8')) - : fs.readFileSync(path, 'utf8'); - exports.render(str, options, fn); - } catch (err) { - fn(err); - } -}; - -/** - * Express support. - */ - -exports.__express = exports.renderFile; diff --git a/node_modules/jade/lib/lexer.js b/node_modules/jade/lib/lexer.js deleted file mode 100644 index 1e0bfff..0000000 --- a/node_modules/jade/lib/lexer.js +++ /dev/null @@ -1,775 +0,0 @@ -/*! - * Jade - Lexer - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -var utils = require('./utils'); - -/** - * Initialize `Lexer` with the given `str`. - * - * Options: - * - * - `colons` allow colons for attr delimiters - * - * @param {String} str - * @param {Object} options - * @api private - */ - -var Lexer = module.exports = function Lexer(str, options) { - options = options || {}; - this.input = str.replace(/\r\n|\r/g, '\n'); - this.colons = options.colons; - this.deferredTokens = []; - this.lastIndents = 0; - this.lineno = 1; - this.stash = []; - this.indentStack = []; - this.indentRe = null; - this.pipeless = false; -}; - -/** - * Lexer prototype. - */ - -Lexer.prototype = { - - /** - * Construct a token with the given `type` and `val`. - * - * @param {String} type - * @param {String} val - * @return {Object} - * @api private - */ - - tok: function(type, val){ - return { - type: type - , line: this.lineno - , val: val - } - }, - - /** - * Consume the given `len` of input. - * - * @param {Number} len - * @api private - */ - - consume: function(len){ - this.input = this.input.substr(len); - }, - - /** - * Scan for `type` with the given `regexp`. - * - * @param {String} type - * @param {RegExp} regexp - * @return {Object} - * @api private - */ - - scan: function(regexp, type){ - var captures; - if (captures = regexp.exec(this.input)) { - this.consume(captures[0].length); - return this.tok(type, captures[1]); - } - }, - - /** - * Defer the given `tok`. - * - * @param {Object} tok - * @api private - */ - - defer: function(tok){ - this.deferredTokens.push(tok); - }, - - /** - * Lookahead `n` tokens. - * - * @param {Number} n - * @return {Object} - * @api private - */ - - lookahead: function(n){ - var fetch = n - this.stash.length; - while (fetch-- > 0) this.stash.push(this.next()); - return this.stash[--n]; - }, - - /** - * Return the indexOf `start` / `end` delimiters. - * - * @param {String} start - * @param {String} end - * @return {Number} - * @api private - */ - - indexOfDelimiters: function(start, end){ - var str = this.input - , nstart = 0 - , nend = 0 - , pos = 0; - for (var i = 0, len = str.length; i < len; ++i) { - if (start == str.charAt(i)) { - ++nstart; - } else if (end == str.charAt(i)) { - if (++nend == nstart) { - pos = i; - break; - } - } - } - return pos; - }, - - /** - * Stashed token. - */ - - stashed: function() { - return this.stash.length - && this.stash.shift(); - }, - - /** - * Deferred token. - */ - - deferred: function() { - return this.deferredTokens.length - && this.deferredTokens.shift(); - }, - - /** - * end-of-source. - */ - - eos: function() { - if (this.input.length) return; - if (this.indentStack.length) { - this.indentStack.shift(); - return this.tok('outdent'); - } else { - return this.tok('eos'); - } - }, - - /** - * Blank line. - */ - - blank: function() { - var captures; - if (captures = /^\n *\n/.exec(this.input)) { - this.consume(captures[0].length - 1); - ++this.lineno; - if (this.pipeless) return this.tok('text', ''); - return this.next(); - } - }, - - /** - * Comment. - */ - - comment: function() { - var captures; - if (captures = /^ *\/\/(-)?([^\n]*)/.exec(this.input)) { - this.consume(captures[0].length); - var tok = this.tok('comment', captures[2]); - tok.buffer = '-' != captures[1]; - return tok; - } - }, - - /** - * Interpolated tag. - */ - - interpolation: function() { - var captures; - if (captures = /^#\{(.*?)\}/.exec(this.input)) { - this.consume(captures[0].length); - return this.tok('interpolation', captures[1]); - } - }, - - /** - * Tag. - */ - - tag: function() { - var captures; - if (captures = /^(\w[-:\w]*)(\/?)/.exec(this.input)) { - this.consume(captures[0].length); - var tok, name = captures[1]; - if (':' == name[name.length - 1]) { - name = name.slice(0, -1); - tok = this.tok('tag', name); - this.defer(this.tok(':')); - while (' ' == this.input[0]) this.input = this.input.substr(1); - } else { - tok = this.tok('tag', name); - } - tok.selfClosing = !! captures[2]; - return tok; - } - }, - - /** - * Filter. - */ - - filter: function() { - return this.scan(/^:(\w+)/, 'filter'); - }, - - /** - * Doctype. - */ - - doctype: function() { - return this.scan(/^(?:!!!|doctype) *([^\n]+)?/, 'doctype'); - }, - - /** - * Id. - */ - - id: function() { - return this.scan(/^#([\w-]+)/, 'id'); - }, - - /** - * Class. - */ - - className: function() { - return this.scan(/^\.([\w-]+)/, 'class'); - }, - - /** - * Text. - */ - - text: function() { - return this.scan(/^(?:\| ?| ?)?([^\n]+)/, 'text'); - }, - - /** - * Extends. - */ - - "extends": function() { - return this.scan(/^extends? +([^\n]+)/, 'extends'); - }, - - /** - * Block prepend. - */ - - prepend: function() { - var captures; - if (captures = /^prepend +([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - var mode = 'prepend' - , name = captures[1] - , tok = this.tok('block', name); - tok.mode = mode; - return tok; - } - }, - - /** - * Block append. - */ - - append: function() { - var captures; - if (captures = /^append +([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - var mode = 'append' - , name = captures[1] - , tok = this.tok('block', name); - tok.mode = mode; - return tok; - } - }, - - /** - * Block. - */ - - block: function() { - var captures; - if (captures = /^block\b *(?:(prepend|append) +)?([^\n]*)/.exec(this.input)) { - this.consume(captures[0].length); - var mode = captures[1] || 'replace' - , name = captures[2] - , tok = this.tok('block', name); - - tok.mode = mode; - return tok; - } - }, - - /** - * Yield. - */ - - yield: function() { - return this.scan(/^yield */, 'yield'); - }, - - /** - * Include. - */ - - include: function() { - return this.scan(/^include +([^\n]+)/, 'include'); - }, - - /** - * Case. - */ - - "case": function() { - return this.scan(/^case +([^\n]+)/, 'case'); - }, - - /** - * When. - */ - - when: function() { - return this.scan(/^when +([^:\n]+)/, 'when'); - }, - - /** - * Default. - */ - - "default": function() { - return this.scan(/^default */, 'default'); - }, - - /** - * Assignment. - */ - - assignment: function() { - var captures; - if (captures = /^(\w+) += *([^;\n]+)( *;? *)/.exec(this.input)) { - this.consume(captures[0].length); - var name = captures[1] - , val = captures[2]; - return this.tok('code', 'var ' + name + ' = (' + val + ');'); - } - }, - - /** - * Call mixin. - */ - - call: function(){ - var captures; - if (captures = /^\+([-\w]+)/.exec(this.input)) { - this.consume(captures[0].length); - var tok = this.tok('call', captures[1]); - - // Check for args (not attributes) - if (captures = /^ *\((.*?)\)/.exec(this.input)) { - if (!/^ *[-\w]+ *=/.test(captures[1])) { - this.consume(captures[0].length); - tok.args = captures[1]; - } - } - - return tok; - } - }, - - /** - * Mixin. - */ - - mixin: function(){ - var captures; - if (captures = /^mixin +([-\w]+)(?: *\((.*)\))?/.exec(this.input)) { - this.consume(captures[0].length); - var tok = this.tok('mixin', captures[1]); - tok.args = captures[2]; - return tok; - } - }, - - /** - * Conditional. - */ - - conditional: function() { - var captures; - if (captures = /^(if|unless|else if|else)\b([^\n]*)/.exec(this.input)) { - this.consume(captures[0].length); - var type = captures[1] - , js = captures[2]; - - switch (type) { - case 'if': js = 'if (' + js + ')'; break; - case 'unless': js = 'if (!(' + js + '))'; break; - case 'else if': js = 'else if (' + js + ')'; break; - case 'else': js = 'else'; break; - } - - return this.tok('code', js); - } - }, - - /** - * While. - */ - - "while": function() { - var captures; - if (captures = /^while +([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - return this.tok('code', 'while (' + captures[1] + ')'); - } - }, - - /** - * Each. - */ - - each: function() { - var captures; - if (captures = /^(?:- *)?(?:each|for) +(\w+)(?: *, *(\w+))? * in *([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - var tok = this.tok('each', captures[1]); - tok.key = captures[2] || '$index'; - tok.code = captures[3]; - return tok; - } - }, - - /** - * Code. - */ - - code: function() { - var captures; - if (captures = /^(!?=|-)([^\n]+)/.exec(this.input)) { - this.consume(captures[0].length); - var flags = captures[1]; - captures[1] = captures[2]; - var tok = this.tok('code', captures[1]); - tok.escape = flags.charAt(0) === '='; - tok.buffer = flags.charAt(0) === '=' || flags.charAt(1) === '='; - return tok; - } - }, - - /** - * Attributes. - */ - - attrs: function() { - if ('(' == this.input.charAt(0)) { - var index = this.indexOfDelimiters('(', ')') - , str = this.input.substr(1, index-1) - , tok = this.tok('attrs') - , len = str.length - , colons = this.colons - , states = ['key'] - , escapedAttr - , key = '' - , val = '' - , quote - , c - , p; - - function state(){ - return states[states.length - 1]; - } - - function interpolate(attr) { - return attr.replace(/(\\)?#\{([^}]+)\}/g, function(_, escape, expr){ - return escape - ? _ - : quote + " + (" + expr + ") + " + quote; - }); - } - - this.consume(index + 1); - tok.attrs = {}; - tok.escaped = {}; - - function parse(c) { - var real = c; - // TODO: remove when people fix ":" - if (colons && ':' == c) c = '='; - switch (c) { - case ',': - case '\n': - switch (state()) { - case 'expr': - case 'array': - case 'string': - case 'object': - val += c; - break; - default: - states.push('key'); - val = val.trim(); - key = key.trim(); - if ('' == key) return; - key = key.replace(/^['"]|['"]$/g, '').replace('!', ''); - tok.escaped[key] = escapedAttr; - tok.attrs[key] = '' == val - ? true - : interpolate(val); - key = val = ''; - } - break; - case '=': - switch (state()) { - case 'key char': - key += real; - break; - case 'val': - case 'expr': - case 'array': - case 'string': - case 'object': - val += real; - break; - default: - escapedAttr = '!' != p; - states.push('val'); - } - break; - case '(': - if ('val' == state() - || 'expr' == state()) states.push('expr'); - val += c; - break; - case ')': - if ('expr' == state() - || 'val' == state()) states.pop(); - val += c; - break; - case '{': - if ('val' == state()) states.push('object'); - val += c; - break; - case '}': - if ('object' == state()) states.pop(); - val += c; - break; - case '[': - if ('val' == state()) states.push('array'); - val += c; - break; - case ']': - if ('array' == state()) states.pop(); - val += c; - break; - case '"': - case "'": - switch (state()) { - case 'key': - states.push('key char'); - break; - case 'key char': - states.pop(); - break; - case 'string': - if (c == quote) states.pop(); - val += c; - break; - default: - states.push('string'); - val += c; - quote = c; - } - break; - case '': - break; - default: - switch (state()) { - case 'key': - case 'key char': - key += c; - break; - default: - val += c; - } - } - p = c; - } - - for (var i = 0; i < len; ++i) { - parse(str.charAt(i)); - } - - parse(','); - - if ('/' == this.input.charAt(0)) { - this.consume(1); - tok.selfClosing = true; - } - - return tok; - } - }, - - /** - * Indent | Outdent | Newline. - */ - - indent: function() { - var captures, re; - - // established regexp - if (this.indentRe) { - captures = this.indentRe.exec(this.input); - // determine regexp - } else { - // tabs - re = /^\n(\t*) */; - captures = re.exec(this.input); - - // spaces - if (captures && !captures[1].length) { - re = /^\n( *)/; - captures = re.exec(this.input); - } - - // established - if (captures && captures[1].length) this.indentRe = re; - } - - if (captures) { - var tok - , indents = captures[1].length; - - ++this.lineno; - this.consume(indents + 1); - - if (' ' == this.input[0] || '\t' == this.input[0]) { - throw new Error('Invalid indentation, you can use tabs or spaces but not both'); - } - - // blank line - if ('\n' == this.input[0]) return this.tok('newline'); - - // outdent - if (this.indentStack.length && indents < this.indentStack[0]) { - while (this.indentStack.length && this.indentStack[0] > indents) { - this.stash.push(this.tok('outdent')); - this.indentStack.shift(); - } - tok = this.stash.pop(); - // indent - } else if (indents && indents != this.indentStack[0]) { - this.indentStack.unshift(indents); - tok = this.tok('indent', indents); - // newline - } else { - tok = this.tok('newline'); - } - - return tok; - } - }, - - /** - * Pipe-less text consumed only when - * pipeless is true; - */ - - pipelessText: function() { - if (this.pipeless) { - if ('\n' == this.input[0]) return; - var i = this.input.indexOf('\n'); - if (-1 == i) i = this.input.length; - var str = this.input.substr(0, i); - this.consume(str.length); - return this.tok('text', str); - } - }, - - /** - * ':' - */ - - colon: function() { - return this.scan(/^: */, ':'); - }, - - /** - * Return the next token object, or those - * previously stashed by lookahead. - * - * @return {Object} - * @api private - */ - - advance: function(){ - return this.stashed() - || this.next(); - }, - - /** - * Return the next token object. - * - * @return {Object} - * @api private - */ - - next: function() { - return this.deferred() - || this.blank() - || this.eos() - || this.pipelessText() - || this.yield() - || this.doctype() - || this.interpolation() - || this["case"]() - || this.when() - || this["default"]() - || this["extends"]() - || this.append() - || this.prepend() - || this.block() - || this.include() - || this.mixin() - || this.call() - || this.conditional() - || this.each() - || this["while"]() - || this.assignment() - || this.tag() - || this.filter() - || this.code() - || this.id() - || this.className() - || this.attrs() - || this.indent() - || this.comment() - || this.colon() - || this.text(); - } -}; diff --git a/node_modules/jade/lib/nodes/attrs.js b/node_modules/jade/lib/nodes/attrs.js deleted file mode 100644 index 5de9b59..0000000 --- a/node_modules/jade/lib/nodes/attrs.js +++ /dev/null @@ -1,77 +0,0 @@ - -/*! - * Jade - nodes - Attrs - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'), - Block = require('./block'); - -/** - * Initialize a `Attrs` node. - * - * @api public - */ - -var Attrs = module.exports = function Attrs() { - this.attrs = []; -}; - -/** - * Inherit from `Node`. - */ - -Attrs.prototype.__proto__ = Node.prototype; - -/** - * Set attribute `name` to `val`, keep in mind these become - * part of a raw js object literal, so to quote a value you must - * '"quote me"', otherwise or example 'user.name' is literal JavaScript. - * - * @param {String} name - * @param {String} val - * @param {Boolean} escaped - * @return {Tag} for chaining - * @api public - */ - -Attrs.prototype.setAttribute = function(name, val, escaped){ - this.attrs.push({ name: name, val: val, escaped: escaped }); - return this; -}; - -/** - * Remove attribute `name` when present. - * - * @param {String} name - * @api public - */ - -Attrs.prototype.removeAttribute = function(name){ - for (var i = 0, len = this.attrs.length; i < len; ++i) { - if (this.attrs[i] && this.attrs[i].name == name) { - delete this.attrs[i]; - } - } -}; - -/** - * Get attribute value by `name`. - * - * @param {String} name - * @return {String} - * @api public - */ - -Attrs.prototype.getAttribute = function(name){ - for (var i = 0, len = this.attrs.length; i < len; ++i) { - if (this.attrs[i] && this.attrs[i].name == name) { - return this.attrs[i].val; - } - } -}; diff --git a/node_modules/jade/lib/nodes/block-comment.js b/node_modules/jade/lib/nodes/block-comment.js deleted file mode 100644 index 4f41e4a..0000000 --- a/node_modules/jade/lib/nodes/block-comment.js +++ /dev/null @@ -1,33 +0,0 @@ - -/*! - * Jade - nodes - BlockComment - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `BlockComment` with the given `block`. - * - * @param {String} val - * @param {Block} block - * @param {Boolean} buffer - * @api public - */ - -var BlockComment = module.exports = function BlockComment(val, block, buffer) { - this.block = block; - this.val = val; - this.buffer = buffer; -}; - -/** - * Inherit from `Node`. - */ - -BlockComment.prototype.__proto__ = Node.prototype; \ No newline at end of file diff --git a/node_modules/jade/lib/nodes/block.js b/node_modules/jade/lib/nodes/block.js deleted file mode 100644 index 6bb18c9..0000000 --- a/node_modules/jade/lib/nodes/block.js +++ /dev/null @@ -1,122 +0,0 @@ - -/*! - * Jade - nodes - Block - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a new `Block` with an optional `node`. - * - * @param {Node} node - * @api public - */ - -var Block = module.exports = function Block(node){ - this.nodes = []; - if (node) this.push(node); -}; - -/** - * Inherit from `Node`. - */ - -Block.prototype.__proto__ = Node.prototype; - -/** - * Block flag. - */ - -Block.prototype.isBlock = true; - -/** - * Replace the nodes in `other` with the nodes - * in `this` block. - * - * @param {Block} other - * @api private - */ - -Block.prototype.replace = function(other){ - other.nodes = this.nodes; -}; - -/** - * Pust the given `node`. - * - * @param {Node} node - * @return {Number} - * @api public - */ - -Block.prototype.push = function(node){ - return this.nodes.push(node); -}; - -/** - * Check if this block is empty. - * - * @return {Boolean} - * @api public - */ - -Block.prototype.isEmpty = function(){ - return 0 == this.nodes.length; -}; - -/** - * Unshift the given `node`. - * - * @param {Node} node - * @return {Number} - * @api public - */ - -Block.prototype.unshift = function(node){ - return this.nodes.unshift(node); -}; - -/** - * Return the "last" block, or the first `yield` node. - * - * @return {Block} - * @api private - */ - -Block.prototype.includeBlock = function(){ - var ret = this - , node; - - for (var i = 0, len = this.nodes.length; i < len; ++i) { - node = this.nodes[i]; - if (node.yield) return node; - else if (node.textOnly) continue; - else if (node.includeBlock) ret = node.includeBlock(); - else if (node.block && !node.block.isEmpty()) ret = node.block.includeBlock(); - if (ret.yield) return ret; - } - - return ret; -}; - -/** - * Return a clone of this block. - * - * @return {Block} - * @api private - */ - -Block.prototype.clone = function(){ - var clone = new Block; - for (var i = 0, len = this.nodes.length; i < len; ++i) { - clone.push(this.nodes[i].clone()); - } - return clone; -}; - diff --git a/node_modules/jade/lib/nodes/case.js b/node_modules/jade/lib/nodes/case.js deleted file mode 100644 index 08ff033..0000000 --- a/node_modules/jade/lib/nodes/case.js +++ /dev/null @@ -1,43 +0,0 @@ - -/*! - * Jade - nodes - Case - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a new `Case` with `expr`. - * - * @param {String} expr - * @api public - */ - -var Case = exports = module.exports = function Case(expr, block){ - this.expr = expr; - this.block = block; -}; - -/** - * Inherit from `Node`. - */ - -Case.prototype.__proto__ = Node.prototype; - -var When = exports.When = function When(expr, block){ - this.expr = expr; - this.block = block; - this.debug = false; -}; - -/** - * Inherit from `Node`. - */ - -When.prototype.__proto__ = Node.prototype; - diff --git a/node_modules/jade/lib/nodes/code.js b/node_modules/jade/lib/nodes/code.js deleted file mode 100644 index babc675..0000000 --- a/node_modules/jade/lib/nodes/code.js +++ /dev/null @@ -1,35 +0,0 @@ - -/*! - * Jade - nodes - Code - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Code` node with the given code `val`. - * Code may also be optionally buffered and escaped. - * - * @param {String} val - * @param {Boolean} buffer - * @param {Boolean} escape - * @api public - */ - -var Code = module.exports = function Code(val, buffer, escape) { - this.val = val; - this.buffer = buffer; - this.escape = escape; - if (val.match(/^ *else/)) this.debug = false; -}; - -/** - * Inherit from `Node`. - */ - -Code.prototype.__proto__ = Node.prototype; \ No newline at end of file diff --git a/node_modules/jade/lib/nodes/comment.js b/node_modules/jade/lib/nodes/comment.js deleted file mode 100644 index 2e1469e..0000000 --- a/node_modules/jade/lib/nodes/comment.js +++ /dev/null @@ -1,32 +0,0 @@ - -/*! - * Jade - nodes - Comment - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Comment` with the given `val`, optionally `buffer`, - * otherwise the comment may render in the output. - * - * @param {String} val - * @param {Boolean} buffer - * @api public - */ - -var Comment = module.exports = function Comment(val, buffer) { - this.val = val; - this.buffer = buffer; -}; - -/** - * Inherit from `Node`. - */ - -Comment.prototype.__proto__ = Node.prototype; \ No newline at end of file diff --git a/node_modules/jade/lib/nodes/doctype.js b/node_modules/jade/lib/nodes/doctype.js deleted file mode 100644 index b8f33e5..0000000 --- a/node_modules/jade/lib/nodes/doctype.js +++ /dev/null @@ -1,29 +0,0 @@ - -/*! - * Jade - nodes - Doctype - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Doctype` with the given `val`. - * - * @param {String} val - * @api public - */ - -var Doctype = module.exports = function Doctype(val) { - this.val = val; -}; - -/** - * Inherit from `Node`. - */ - -Doctype.prototype.__proto__ = Node.prototype; \ No newline at end of file diff --git a/node_modules/jade/lib/nodes/each.js b/node_modules/jade/lib/nodes/each.js deleted file mode 100644 index f54101f..0000000 --- a/node_modules/jade/lib/nodes/each.js +++ /dev/null @@ -1,35 +0,0 @@ - -/*! - * Jade - nodes - Each - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize an `Each` node, representing iteration - * - * @param {String} obj - * @param {String} val - * @param {String} key - * @param {Block} block - * @api public - */ - -var Each = module.exports = function Each(obj, val, key, block) { - this.obj = obj; - this.val = val; - this.key = key; - this.block = block; -}; - -/** - * Inherit from `Node`. - */ - -Each.prototype.__proto__ = Node.prototype; \ No newline at end of file diff --git a/node_modules/jade/lib/nodes/filter.js b/node_modules/jade/lib/nodes/filter.js deleted file mode 100644 index 0d7ff6e..0000000 --- a/node_modules/jade/lib/nodes/filter.js +++ /dev/null @@ -1,34 +0,0 @@ - -/*! - * Jade - nodes - Filter - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node') - , Block = require('./block'); - -/** - * Initialize a `Filter` node with the given - * filter `name` and `block`. - * - * @param {String} name - * @param {Block|Node} block - * @api public - */ - -var Filter = module.exports = function Filter(name, block, attrs) { - this.name = name; - this.block = block; - this.attrs = attrs; -}; - -/** - * Inherit from `Node`. - */ - -Filter.prototype.__proto__ = Node.prototype; \ No newline at end of file diff --git a/node_modules/jade/lib/nodes/index.js b/node_modules/jade/lib/nodes/index.js deleted file mode 100644 index 386ad2f..0000000 --- a/node_modules/jade/lib/nodes/index.js +++ /dev/null @@ -1,20 +0,0 @@ - -/*! - * Jade - nodes - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -exports.Node = require('./node'); -exports.Tag = require('./tag'); -exports.Code = require('./code'); -exports.Each = require('./each'); -exports.Case = require('./case'); -exports.Text = require('./text'); -exports.Block = require('./block'); -exports.Mixin = require('./mixin'); -exports.Filter = require('./filter'); -exports.Comment = require('./comment'); -exports.Literal = require('./literal'); -exports.BlockComment = require('./block-comment'); -exports.Doctype = require('./doctype'); diff --git a/node_modules/jade/lib/nodes/literal.js b/node_modules/jade/lib/nodes/literal.js deleted file mode 100644 index fde586b..0000000 --- a/node_modules/jade/lib/nodes/literal.js +++ /dev/null @@ -1,32 +0,0 @@ - -/*! - * Jade - nodes - Literal - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Literal` node with the given `str. - * - * @param {String} str - * @api public - */ - -var Literal = module.exports = function Literal(str) { - this.str = str - .replace(/\\/g, "\\\\") - .replace(/\n|\r\n/g, "\\n") - .replace(/'/g, "\\'"); -}; - -/** - * Inherit from `Node`. - */ - -Literal.prototype.__proto__ = Node.prototype; diff --git a/node_modules/jade/lib/nodes/mixin.js b/node_modules/jade/lib/nodes/mixin.js deleted file mode 100644 index 8407bc7..0000000 --- a/node_modules/jade/lib/nodes/mixin.js +++ /dev/null @@ -1,36 +0,0 @@ - -/*! - * Jade - nodes - Mixin - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Attrs = require('./attrs'); - -/** - * Initialize a new `Mixin` with `name` and `block`. - * - * @param {String} name - * @param {String} args - * @param {Block} block - * @api public - */ - -var Mixin = module.exports = function Mixin(name, args, block, call){ - this.name = name; - this.args = args; - this.block = block; - this.attrs = []; - this.call = call; -}; - -/** - * Inherit from `Attrs`. - */ - -Mixin.prototype.__proto__ = Attrs.prototype; - diff --git a/node_modules/jade/lib/nodes/node.js b/node_modules/jade/lib/nodes/node.js deleted file mode 100644 index e98f042..0000000 --- a/node_modules/jade/lib/nodes/node.js +++ /dev/null @@ -1,25 +0,0 @@ - -/*! - * Jade - nodes - Node - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Initialize a `Node`. - * - * @api public - */ - -var Node = module.exports = function Node(){}; - -/** - * Clone this node (return itself) - * - * @return {Node} - * @api private - */ - -Node.prototype.clone = function(){ - return this; -}; diff --git a/node_modules/jade/lib/nodes/tag.js b/node_modules/jade/lib/nodes/tag.js deleted file mode 100644 index 4b6728a..0000000 --- a/node_modules/jade/lib/nodes/tag.js +++ /dev/null @@ -1,95 +0,0 @@ - -/*! - * Jade - nodes - Tag - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Attrs = require('./attrs'), - Block = require('./block'), - inlineTags = require('../inline-tags'); - -/** - * Initialize a `Tag` node with the given tag `name` and optional `block`. - * - * @param {String} name - * @param {Block} block - * @api public - */ - -var Tag = module.exports = function Tag(name, block) { - this.name = name; - this.attrs = []; - this.block = block || new Block; -}; - -/** - * Inherit from `Attrs`. - */ - -Tag.prototype.__proto__ = Attrs.prototype; - -/** - * Clone this tag. - * - * @return {Tag} - * @api private - */ - -Tag.prototype.clone = function(){ - var clone = new Tag(this.name, this.block.clone()); - clone.line = this.line; - clone.attrs = this.attrs; - clone.textOnly = this.textOnly; - return clone; -}; - -/** - * Check if this tag is an inline tag. - * - * @return {Boolean} - * @api private - */ - -Tag.prototype.isInline = function(){ - return ~inlineTags.indexOf(this.name); -}; - -/** - * Check if this tag's contents can be inlined. Used for pretty printing. - * - * @return {Boolean} - * @api private - */ - -Tag.prototype.canInline = function(){ - var nodes = this.block.nodes; - - function isInline(node){ - // Recurse if the node is a block - if (node.isBlock) return node.nodes.every(isInline); - return node.isText || (node.isInline && node.isInline()); - } - - // Empty tag - if (!nodes.length) return true; - - // Text-only or inline-only tag - if (1 == nodes.length) return isInline(nodes[0]); - - // Multi-line inline-only tag - if (this.block.nodes.every(isInline)) { - for (var i = 1, len = nodes.length; i < len; ++i) { - if (nodes[i-1].isText && nodes[i].isText) - return false; - } - return true; - } - - // Mixed tag - return false; -}; \ No newline at end of file diff --git a/node_modules/jade/lib/nodes/text.js b/node_modules/jade/lib/nodes/text.js deleted file mode 100644 index 3b5dd55..0000000 --- a/node_modules/jade/lib/nodes/text.js +++ /dev/null @@ -1,36 +0,0 @@ - -/*! - * Jade - nodes - Text - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Node = require('./node'); - -/** - * Initialize a `Text` node with optional `line`. - * - * @param {String} line - * @api public - */ - -var Text = module.exports = function Text(line) { - this.val = ''; - if ('string' == typeof line) this.val = line; -}; - -/** - * Inherit from `Node`. - */ - -Text.prototype.__proto__ = Node.prototype; - -/** - * Flag as text. - */ - -Text.prototype.isText = true; \ No newline at end of file diff --git a/node_modules/jade/lib/parser.js b/node_modules/jade/lib/parser.js deleted file mode 100644 index f8c929f..0000000 --- a/node_modules/jade/lib/parser.js +++ /dev/null @@ -1,699 +0,0 @@ -/*! - * Jade - Parser - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var Lexer = require('./lexer') - , nodes = require('./nodes') - , utils = require('./utils') - , filters = require('./filters') - , path = require('path') - , extname = path.extname; - -/** - * Initialize `Parser` with the given input `str` and `filename`. - * - * @param {String} str - * @param {String} filename - * @param {Object} options - * @api public - */ - -var Parser = exports = module.exports = function Parser(str, filename, options){ - this.input = str; - this.lexer = new Lexer(str, options); - this.filename = filename; - this.blocks = {}; - this.mixins = {}; - this.options = options; - this.contexts = [this]; -}; - -/** - * Tags that may not contain tags. - */ - -var textOnly = exports.textOnly = ['script', 'style']; - -/** - * Parser prototype. - */ - -Parser.prototype = { - - /** - * Push `parser` onto the context stack, - * or pop and return a `Parser`. - */ - - context: function(parser){ - if (parser) { - this.contexts.push(parser); - } else { - return this.contexts.pop(); - } - }, - - /** - * Return the next token object. - * - * @return {Object} - * @api private - */ - - advance: function(){ - return this.lexer.advance(); - }, - - /** - * Skip `n` tokens. - * - * @param {Number} n - * @api private - */ - - skip: function(n){ - while (n--) this.advance(); - }, - - /** - * Single token lookahead. - * - * @return {Object} - * @api private - */ - - peek: function() { - return this.lookahead(1); - }, - - /** - * Return lexer lineno. - * - * @return {Number} - * @api private - */ - - line: function() { - return this.lexer.lineno; - }, - - /** - * `n` token lookahead. - * - * @param {Number} n - * @return {Object} - * @api private - */ - - lookahead: function(n){ - return this.lexer.lookahead(n); - }, - - /** - * Parse input returning a string of js for evaluation. - * - * @return {String} - * @api public - */ - - parse: function(){ - var block = new nodes.Block, parser; - block.line = this.line(); - - while ('eos' != this.peek().type) { - if ('newline' == this.peek().type) { - this.advance(); - } else { - block.push(this.parseExpr()); - } - } - - if (parser = this.extending) { - this.context(parser); - var ast = parser.parse(); - this.context(); - // hoist mixins - for (var name in this.mixins) - ast.unshift(this.mixins[name]); - return ast; - } - - return block; - }, - - /** - * Expect the given type, or throw an exception. - * - * @param {String} type - * @api private - */ - - expect: function(type){ - if (this.peek().type === type) { - return this.advance(); - } else { - throw new Error('expected "' + type + '", but got "' + this.peek().type + '"'); - } - }, - - /** - * Accept the given `type`. - * - * @param {String} type - * @api private - */ - - accept: function(type){ - if (this.peek().type === type) { - return this.advance(); - } - }, - - /** - * tag - * | doctype - * | mixin - * | include - * | filter - * | comment - * | text - * | each - * | code - * | yield - * | id - * | class - * | interpolation - */ - - parseExpr: function(){ - switch (this.peek().type) { - case 'tag': - return this.parseTag(); - case 'mixin': - return this.parseMixin(); - case 'block': - return this.parseBlock(); - case 'case': - return this.parseCase(); - case 'when': - return this.parseWhen(); - case 'default': - return this.parseDefault(); - case 'extends': - return this.parseExtends(); - case 'include': - return this.parseInclude(); - case 'doctype': - return this.parseDoctype(); - case 'filter': - return this.parseFilter(); - case 'comment': - return this.parseComment(); - case 'text': - return this.parseText(); - case 'each': - return this.parseEach(); - case 'code': - return this.parseCode(); - case 'call': - return this.parseCall(); - case 'interpolation': - return this.parseInterpolation(); - case 'yield': - this.advance(); - var block = new nodes.Block; - block.yield = true; - return block; - case 'id': - case 'class': - var tok = this.advance(); - this.lexer.defer(this.lexer.tok('tag', 'div')); - this.lexer.defer(tok); - return this.parseExpr(); - default: - throw new Error('unexpected token "' + this.peek().type + '"'); - } - }, - - /** - * Text - */ - - parseText: function(){ - var tok = this.expect('text'); - var node = new nodes.Text(tok.val); - node.line = this.line(); - return node; - }, - - /** - * ':' expr - * | block - */ - - parseBlockExpansion: function(){ - if (':' == this.peek().type) { - this.advance(); - return new nodes.Block(this.parseExpr()); - } else { - return this.block(); - } - }, - - /** - * case - */ - - parseCase: function(){ - var val = this.expect('case').val; - var node = new nodes.Case(val); - node.line = this.line(); - node.block = this.block(); - return node; - }, - - /** - * when - */ - - parseWhen: function(){ - var val = this.expect('when').val - return new nodes.Case.When(val, this.parseBlockExpansion()); - }, - - /** - * default - */ - - parseDefault: function(){ - this.expect('default'); - return new nodes.Case.When('default', this.parseBlockExpansion()); - }, - - /** - * code - */ - - parseCode: function(){ - var tok = this.expect('code'); - var node = new nodes.Code(tok.val, tok.buffer, tok.escape); - var block; - var i = 1; - node.line = this.line(); - while (this.lookahead(i) && 'newline' == this.lookahead(i).type) ++i; - block = 'indent' == this.lookahead(i).type; - if (block) { - this.skip(i-1); - node.block = this.block(); - } - return node; - }, - - /** - * comment - */ - - parseComment: function(){ - var tok = this.expect('comment'); - var node; - - if ('indent' == this.peek().type) { - node = new nodes.BlockComment(tok.val, this.block(), tok.buffer); - } else { - node = new nodes.Comment(tok.val, tok.buffer); - } - - node.line = this.line(); - return node; - }, - - /** - * doctype - */ - - parseDoctype: function(){ - var tok = this.expect('doctype'); - var node = new nodes.Doctype(tok.val); - node.line = this.line(); - return node; - }, - - /** - * filter attrs? text-block - */ - - parseFilter: function(){ - var tok = this.expect('filter'); - var attrs = this.accept('attrs'); - var block; - - this.lexer.pipeless = true; - block = this.parseTextBlock(); - this.lexer.pipeless = false; - - var node = new nodes.Filter(tok.val, block, attrs && attrs.attrs); - node.line = this.line(); - return node; - }, - - /** - * each block - */ - - parseEach: function(){ - var tok = this.expect('each'); - var node = new nodes.Each(tok.code, tok.val, tok.key); - node.line = this.line(); - node.block = this.block(); - if (this.peek().type == 'code' && this.peek().val == 'else') { - this.advance(); - node.alternative = this.block(); - } - return node; - }, - - /** - * 'extends' name - */ - - parseExtends: function(){ - var path = require('path'); - var fs = require('fs'); - var dirname = path.dirname; - var basename = path.basename; - var join = path.join; - - if (!this.filename) - throw new Error('the "filename" option is required to extend templates'); - - path = this.expect('extends').val.trim(); - var dir = dirname(this.filename); - - path = join(dir, path + '.jade'); - var str = fs.readFileSync(path, 'utf8'); - var parser = new Parser(str, path, this.options); - - parser.blocks = this.blocks; - parser.contexts = this.contexts; - this.extending = parser; - - // TODO: null node - return new nodes.Literal(''); - }, - - /** - * 'block' name block - */ - - parseBlock: function(){ - var block = this.expect('block'); - var mode = block.mode; - var name = block.val.trim(); - - block = 'indent' == this.peek().type - ? this.block() - : new nodes.Block(new nodes.Literal('')); - - var prev = this.blocks[name]; - - if (prev) { - switch (prev.mode) { - case 'append': - block.nodes = block.nodes.concat(prev.nodes); - prev = block; - break; - case 'prepend': - block.nodes = prev.nodes.concat(block.nodes); - prev = block; - break; - } - } - - block.mode = mode; - return this.blocks[name] = prev || block; - }, - - /** - * include block? - */ - - parseInclude: function(){ - var path = require('path'); - var fs = require('fs'); - var dirname = path.dirname; - var basename = path.basename; - var join = path.join; - var str; - - path = this.expect('include').val.trim(); - var dir = dirname(this.filename); - - if (!this.filename) - throw new Error('the "filename" option is required to use includes'); - - // no extension - if (!~basename(path).indexOf('.')) { - path += '.jade'; - } - - // non-jade - if ('.jade' != path.substr(-5)) { - path = join(dir, path); - str = fs.readFileSync(path, 'utf8').replace(/\r/g, ''); - var ext = extname(path).slice(1); - var filter = filters[ext]; - if (filter) str = filter(str, { filename: path }).replace(/\\n/g, '\n'); - return new nodes.Literal(str); - } - - path = join(dir, path); - str = fs.readFileSync(path, 'utf8'); - var parser = new Parser(str, path, this.options); - parser.blocks = utils.merge({}, this.blocks); - parser.mixins = this.mixins; - - this.context(parser); - var ast = parser.parse(); - this.context(); - ast.filename = path; - - if ('indent' == this.peek().type) { - ast.includeBlock().push(this.block()); - } - - return ast; - }, - - /** - * call ident block - */ - - parseCall: function(){ - var tok = this.expect('call'); - var name = tok.val; - var args = tok.args; - var mixin = new nodes.Mixin(name, args, new nodes.Block, true); - - this.tag(mixin); - if (mixin.block.isEmpty()) mixin.block = null; - return mixin; - }, - - /** - * mixin block - */ - - parseMixin: function(){ - var tok = this.expect('mixin'); - var name = tok.val; - var args = tok.args; - var mixin; - - // definition - if ('indent' == this.peek().type) { - mixin = new nodes.Mixin(name, args, this.block(), false); - this.mixins[name] = mixin; - return mixin; - // call - } else { - return new nodes.Mixin(name, args, null, true); - } - }, - - /** - * indent (text | newline)* outdent - */ - - parseTextBlock: function(){ - var block = new nodes.Block; - block.line = this.line(); - var spaces = this.expect('indent').val; - if (null == this._spaces) this._spaces = spaces; - var indent = Array(spaces - this._spaces + 1).join(' '); - while ('outdent' != this.peek().type) { - switch (this.peek().type) { - case 'newline': - this.advance(); - break; - case 'indent': - this.parseTextBlock().nodes.forEach(function(node){ - block.push(node); - }); - break; - default: - var text = new nodes.Text(indent + this.advance().val); - text.line = this.line(); - block.push(text); - } - } - - if (spaces == this._spaces) this._spaces = null; - this.expect('outdent'); - return block; - }, - - /** - * indent expr* outdent - */ - - block: function(){ - var block = new nodes.Block; - block.line = this.line(); - this.expect('indent'); - while ('outdent' != this.peek().type) { - if ('newline' == this.peek().type) { - this.advance(); - } else { - block.push(this.parseExpr()); - } - } - this.expect('outdent'); - return block; - }, - - /** - * interpolation (attrs | class | id)* (text | code | ':')? newline* block? - */ - - parseInterpolation: function(){ - var tok = this.advance(); - var tag = new nodes.Tag(tok.val); - tag.buffer = true; - return this.tag(tag); - }, - - /** - * tag (attrs | class | id)* (text | code | ':')? newline* block? - */ - - parseTag: function(){ - // ast-filter look-ahead - var i = 2; - if ('attrs' == this.lookahead(i).type) ++i; - - var tok = this.advance(); - var tag = new nodes.Tag(tok.val); - - tag.selfClosing = tok.selfClosing; - - return this.tag(tag); - }, - - /** - * Parse tag. - */ - - tag: function(tag){ - var dot; - - tag.line = this.line(); - - // (attrs | class | id)* - out: - while (true) { - switch (this.peek().type) { - case 'id': - case 'class': - var tok = this.advance(); - tag.setAttribute(tok.type, "'" + tok.val + "'"); - continue; - case 'attrs': - var tok = this.advance() - , obj = tok.attrs - , escaped = tok.escaped - , names = Object.keys(obj); - - if (tok.selfClosing) tag.selfClosing = true; - - for (var i = 0, len = names.length; i < len; ++i) { - var name = names[i] - , val = obj[name]; - tag.setAttribute(name, val, escaped[name]); - } - continue; - default: - break out; - } - } - - // check immediate '.' - if ('.' == this.peek().val) { - dot = tag.textOnly = true; - this.advance(); - } - - // (text | code | ':')? - switch (this.peek().type) { - case 'text': - tag.block.push(this.parseText()); - break; - case 'code': - tag.code = this.parseCode(); - break; - case ':': - this.advance(); - tag.block = new nodes.Block; - tag.block.push(this.parseExpr()); - break; - } - - // newline* - while ('newline' == this.peek().type) this.advance(); - - tag.textOnly = tag.textOnly || ~textOnly.indexOf(tag.name); - - // script special-case - if ('script' == tag.name) { - var type = tag.getAttribute('type'); - if (!dot && type && 'text/javascript' != type.replace(/^['"]|['"]$/g, '')) { - tag.textOnly = false; - } - } - - // block? - if ('indent' == this.peek().type) { - if (tag.textOnly) { - this.lexer.pipeless = true; - tag.block = this.parseTextBlock(); - this.lexer.pipeless = false; - } else { - var block = this.block(); - if (tag.block) { - for (var i = 0, len = block.nodes.length; i < len; ++i) { - tag.block.push(block.nodes[i]); - } - } else { - tag.block = block; - } - } - } - - return tag; - } -}; diff --git a/node_modules/jade/lib/runtime.js b/node_modules/jade/lib/runtime.js deleted file mode 100644 index fb711f5..0000000 --- a/node_modules/jade/lib/runtime.js +++ /dev/null @@ -1,174 +0,0 @@ - -/*! - * Jade - runtime - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Lame Array.isArray() polyfill for now. - */ - -if (!Array.isArray) { - Array.isArray = function(arr){ - return '[object Array]' == Object.prototype.toString.call(arr); - }; -} - -/** - * Lame Object.keys() polyfill for now. - */ - -if (!Object.keys) { - Object.keys = function(obj){ - var arr = []; - for (var key in obj) { - if (obj.hasOwnProperty(key)) { - arr.push(key); - } - } - return arr; - } -} - -/** - * Merge two attribute objects giving precedence - * to values in object `b`. Classes are special-cased - * allowing for arrays and merging/joining appropriately - * resulting in a string. - * - * @param {Object} a - * @param {Object} b - * @return {Object} a - * @api private - */ - -exports.merge = function merge(a, b) { - var ac = a['class']; - var bc = b['class']; - - if (ac || bc) { - ac = ac || []; - bc = bc || []; - if (!Array.isArray(ac)) ac = [ac]; - if (!Array.isArray(bc)) bc = [bc]; - ac = ac.filter(nulls); - bc = bc.filter(nulls); - a['class'] = ac.concat(bc).join(' '); - } - - for (var key in b) { - if (key != 'class') { - a[key] = b[key]; - } - } - - return a; -}; - -/** - * Filter null `val`s. - * - * @param {Mixed} val - * @return {Mixed} - * @api private - */ - -function nulls(val) { - return val != null; -} - -/** - * Render the given attributes object. - * - * @param {Object} obj - * @param {Object} escaped - * @return {String} - * @api private - */ - -exports.attrs = function attrs(obj, escaped){ - var buf = [] - , terse = obj.terse; - - delete obj.terse; - var keys = Object.keys(obj) - , len = keys.length; - - if (len) { - buf.push(''); - for (var i = 0; i < len; ++i) { - var key = keys[i] - , val = obj[key]; - - if ('boolean' == typeof val || null == val) { - if (val) { - terse - ? buf.push(key) - : buf.push(key + '="' + key + '"'); - } - } else if (0 == key.indexOf('data') && 'string' != typeof val) { - buf.push(key + "='" + JSON.stringify(val) + "'"); - } else if ('class' == key && Array.isArray(val)) { - buf.push(key + '="' + exports.escape(val.join(' ')) + '"'); - } else if (escaped && escaped[key]) { - buf.push(key + '="' + exports.escape(val) + '"'); - } else { - buf.push(key + '="' + val + '"'); - } - } - } - - return buf.join(' '); -}; - -/** - * Escape the given string of `html`. - * - * @param {String} html - * @return {String} - * @api private - */ - -exports.escape = function escape(html){ - return String(html) - .replace(/&(?!(\w+|\#\d+);)/g, '&') - .replace(//g, '>') - .replace(/"/g, '"'); -}; - -/** - * Re-throw the given `err` in context to the - * the jade in `filename` at the given `lineno`. - * - * @param {Error} err - * @param {String} filename - * @param {String} lineno - * @api private - */ - -exports.rethrow = function rethrow(err, filename, lineno){ - if (!filename) throw err; - - var context = 3 - , str = require('fs').readFileSync(filename, 'utf8') - , lines = str.split('\n') - , start = Math.max(lineno - context, 0) - , end = Math.min(lines.length, lineno + context); - - // Error context - var context = lines.slice(start, end).map(function(line, i){ - var curr = i + start + 1; - return (curr == lineno ? ' > ' : ' ') - + curr - + '| ' - + line; - }).join('\n'); - - // Alter exception message - err.path = filename; - err.message = (filename || 'Jade') + ':' + lineno - + '\n' + context + '\n\n' + err.message; - throw err; -}; diff --git a/node_modules/jade/lib/self-closing.js b/node_modules/jade/lib/self-closing.js deleted file mode 100644 index 0548771..0000000 --- a/node_modules/jade/lib/self-closing.js +++ /dev/null @@ -1,19 +0,0 @@ - -/*! - * Jade - self closing tags - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -module.exports = [ - 'meta' - , 'img' - , 'link' - , 'input' - , 'source' - , 'area' - , 'base' - , 'col' - , 'br' - , 'hr' -]; \ No newline at end of file diff --git a/node_modules/jade/lib/utils.js b/node_modules/jade/lib/utils.js deleted file mode 100644 index ca4a7fa..0000000 --- a/node_modules/jade/lib/utils.js +++ /dev/null @@ -1,68 +0,0 @@ - -/*! - * Jade - utils - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Convert interpolation in the given string to JavaScript. - * - * @param {String} str - * @return {String} - * @api private - */ - -var interpolate = exports.interpolate = function(str){ - return str.replace(/(_SLASH_)?([#!]){(.*?)}/g, function(str, escape, flag, code){ - code = code - .replace(/\\'/g, "'") - .replace(/_SLASH_/g, '\\'); - - return escape - ? str.slice(7) - : "' + " - + ('!' == flag ? '' : 'escape') - + "((interp = " + code - + ") == null ? '' : interp) + '"; - }); -}; - -/** - * Escape single quotes in `str`. - * - * @param {String} str - * @return {String} - * @api private - */ - -var escape = exports.escape = function(str) { - return str.replace(/'/g, "\\'"); -}; - -/** - * Interpolate, and escape the given `str`. - * - * @param {String} str - * @return {String} - * @api private - */ - -exports.text = function(str){ - return interpolate(escape(str)); -}; - -/** - * Merge `b` into `a`. - * - * @param {Object} a - * @param {Object} b - * @return {Object} - * @api public - */ - -exports.merge = function(a, b) { - for (var key in b) a[key] = b[key]; - return a; -}; - diff --git a/node_modules/jade/node_modules/commander/.npmignore b/node_modules/jade/node_modules/commander/.npmignore deleted file mode 100644 index f1250e5..0000000 --- a/node_modules/jade/node_modules/commander/.npmignore +++ /dev/null @@ -1,4 +0,0 @@ -support -test -examples -*.sock diff --git a/node_modules/jade/node_modules/commander/.travis.yml b/node_modules/jade/node_modules/commander/.travis.yml deleted file mode 100644 index f1d0f13..0000000 --- a/node_modules/jade/node_modules/commander/.travis.yml +++ /dev/null @@ -1,4 +0,0 @@ -language: node_js -node_js: - - 0.4 - - 0.6 diff --git a/node_modules/jade/node_modules/commander/History.md b/node_modules/jade/node_modules/commander/History.md deleted file mode 100644 index 4961d2e..0000000 --- a/node_modules/jade/node_modules/commander/History.md +++ /dev/null @@ -1,107 +0,0 @@ - -0.6.1 / 2012-06-01 -================== - - * Added: append (yes or no) on confirmation - * Added: allow node.js v0.7.x - -0.6.0 / 2012-04-10 -================== - - * Added `.prompt(obj, callback)` support. Closes #49 - * Added default support to .choose(). Closes #41 - * Fixed the choice example - -0.5.1 / 2011-12-20 -================== - - * Fixed `password()` for recent nodes. Closes #36 - -0.5.0 / 2011-12-04 -================== - - * Added sub-command option support [itay] - -0.4.3 / 2011-12-04 -================== - - * Fixed custom help ordering. Closes #32 - -0.4.2 / 2011-11-24 -================== - - * Added travis support - * Fixed: line-buffered input automatically trimmed. Closes #31 - -0.4.1 / 2011-11-18 -================== - - * Removed listening for "close" on --help - -0.4.0 / 2011-11-15 -================== - - * Added support for `--`. Closes #24 - -0.3.3 / 2011-11-14 -================== - - * Fixed: wait for close event when writing help info [Jerry Hamlet] - -0.3.2 / 2011-11-01 -================== - - * Fixed long flag definitions with values [felixge] - -0.3.1 / 2011-10-31 -================== - - * Changed `--version` short flag to `-V` from `-v` - * Changed `.version()` so it's configurable [felixge] - -0.3.0 / 2011-10-31 -================== - - * Added support for long flags only. Closes #18 - -0.2.1 / 2011-10-24 -================== - - * "node": ">= 0.4.x < 0.7.0". Closes #20 - -0.2.0 / 2011-09-26 -================== - - * Allow for defaults that are not just boolean. Default peassignment only occurs for --no-*, optional, and required arguments. [Jim Isaacs] - -0.1.0 / 2011-08-24 -================== - - * Added support for custom `--help` output - -0.0.5 / 2011-08-18 -================== - - * Changed: when the user enters nothing prompt for password again - * Fixed issue with passwords beginning with numbers [NuckChorris] - -0.0.4 / 2011-08-15 -================== - - * Fixed `Commander#args` - -0.0.3 / 2011-08-15 -================== - - * Added default option value support - -0.0.2 / 2011-08-15 -================== - - * Added mask support to `Command#password(str[, mask], fn)` - * Added `Command#password(str, fn)` - -0.0.1 / 2010-01-03 -================== - - * Initial release diff --git a/node_modules/jade/node_modules/commander/Makefile b/node_modules/jade/node_modules/commander/Makefile deleted file mode 100644 index 0074625..0000000 --- a/node_modules/jade/node_modules/commander/Makefile +++ /dev/null @@ -1,7 +0,0 @@ - -TESTS = $(shell find test/test.*.js) - -test: - @./test/run $(TESTS) - -.PHONY: test \ No newline at end of file diff --git a/node_modules/jade/node_modules/commander/Readme.md b/node_modules/jade/node_modules/commander/Readme.md deleted file mode 100644 index b8328c3..0000000 --- a/node_modules/jade/node_modules/commander/Readme.md +++ /dev/null @@ -1,262 +0,0 @@ -# Commander.js - - The complete solution for [node.js](http://nodejs.org) command-line interfaces, inspired by Ruby's [commander](https://github.com/visionmedia/commander). - - [![Build Status](https://secure.travis-ci.org/visionmedia/commander.js.png)](http://travis-ci.org/visionmedia/commander.js) - -## Installation - - $ npm install commander - -## Option parsing - - Options with commander are defined with the `.option()` method, also serving as documentation for the options. The example below parses args and options from `process.argv`, leaving remaining args as the `program.args` array which were not consumed by options. - -```js -#!/usr/bin/env node - -/** - * Module dependencies. - */ - -var program = require('commander'); - -program - .version('0.0.1') - .option('-p, --peppers', 'Add peppers') - .option('-P, --pineapple', 'Add pineapple') - .option('-b, --bbq', 'Add bbq sauce') - .option('-c, --cheese [type]', 'Add the specified type of cheese [marble]', 'marble') - .parse(process.argv); - -console.log('you ordered a pizza with:'); -if (program.peppers) console.log(' - peppers'); -if (program.pineapple) console.log(' - pineappe'); -if (program.bbq) console.log(' - bbq'); -console.log(' - %s cheese', program.cheese); -``` - - Short flags may be passed as a single arg, for example `-abc` is equivalent to `-a -b -c`. Multi-word options such as "--template-engine" are camel-cased, becoming `program.templateEngine` etc. - -## Automated --help - - The help information is auto-generated based on the information commander already knows about your program, so the following `--help` info is for free: - -``` - $ ./examples/pizza --help - - Usage: pizza [options] - - Options: - - -V, --version output the version number - -p, --peppers Add peppers - -P, --pineapple Add pineappe - -b, --bbq Add bbq sauce - -c, --cheese Add the specified type of cheese [marble] - -h, --help output usage information - -``` - -## Coercion - -```js -function range(val) { - return val.split('..').map(Number); -} - -function list(val) { - return val.split(','); -} - -program - .version('0.0.1') - .usage('[options] ') - .option('-i, --integer ', 'An integer argument', parseInt) - .option('-f, --float ', 'A float argument', parseFloat) - .option('-r, --range ..', 'A range', range) - .option('-l, --list ', 'A list', list) - .option('-o, --optional [value]', 'An optional value') - .parse(process.argv); - -console.log(' int: %j', program.integer); -console.log(' float: %j', program.float); -console.log(' optional: %j', program.optional); -program.range = program.range || []; -console.log(' range: %j..%j', program.range[0], program.range[1]); -console.log(' list: %j', program.list); -console.log(' args: %j', program.args); -``` - -## Custom help - - You can display arbitrary `-h, --help` information - by listening for "--help". Commander will automatically - exit once you are done so that the remainder of your program - does not execute causing undesired behaviours, for example - in the following executable "stuff" will not output when - `--help` is used. - -```js -#!/usr/bin/env node - -/** - * Module dependencies. - */ - -var program = require('../'); - -function list(val) { - return val.split(',').map(Number); -} - -program - .version('0.0.1') - .option('-f, --foo', 'enable some foo') - .option('-b, --bar', 'enable some bar') - .option('-B, --baz', 'enable some baz'); - -// must be before .parse() since -// node's emit() is immediate - -program.on('--help', function(){ - console.log(' Examples:'); - console.log(''); - console.log(' $ custom-help --help'); - console.log(' $ custom-help -h'); - console.log(''); -}); - -program.parse(process.argv); - -console.log('stuff'); -``` - -yielding the following help output: - -``` - -Usage: custom-help [options] - -Options: - - -h, --help output usage information - -V, --version output the version number - -f, --foo enable some foo - -b, --bar enable some bar - -B, --baz enable some baz - -Examples: - - $ custom-help --help - $ custom-help -h - -``` - -## .prompt(msg, fn) - - Single-line prompt: - -```js -program.prompt('name: ', function(name){ - console.log('hi %s', name); -}); -``` - - Multi-line prompt: - -```js -program.prompt('description:', function(name){ - console.log('hi %s', name); -}); -``` - - Coercion: - -```js -program.prompt('Age: ', Number, function(age){ - console.log('age: %j', age); -}); -``` - -```js -program.prompt('Birthdate: ', Date, function(date){ - console.log('date: %s', date); -}); -``` - -## .password(msg[, mask], fn) - -Prompt for password without echoing: - -```js -program.password('Password: ', function(pass){ - console.log('got "%s"', pass); - process.stdin.destroy(); -}); -``` - -Prompt for password with mask char "*": - -```js -program.password('Password: ', '*', function(pass){ - console.log('got "%s"', pass); - process.stdin.destroy(); -}); -``` - -## .confirm(msg, fn) - - Confirm with the given `msg`: - -```js -program.confirm('continue? ', function(ok){ - console.log(' got %j', ok); -}); -``` - -## .choose(list, fn) - - Let the user choose from a `list`: - -```js -var list = ['tobi', 'loki', 'jane', 'manny', 'luna']; - -console.log('Choose the coolest pet:'); -program.choose(list, function(i){ - console.log('you chose %d "%s"', i, list[i]); -}); -``` - -## Links - - - [API documentation](http://visionmedia.github.com/commander.js/) - - [ascii tables](https://github.com/LearnBoost/cli-table) - - [progress bars](https://github.com/visionmedia/node-progress) - - [more progress bars](https://github.com/substack/node-multimeter) - - [examples](https://github.com/visionmedia/commander.js/tree/master/examples) - -## License - -(The MIT License) - -Copyright (c) 2011 TJ Holowaychuk <tj@vision-media.ca> - -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. \ No newline at end of file diff --git a/node_modules/jade/node_modules/commander/index.js b/node_modules/jade/node_modules/commander/index.js deleted file mode 100644 index 06ec1e4..0000000 --- a/node_modules/jade/node_modules/commander/index.js +++ /dev/null @@ -1,2 +0,0 @@ - -module.exports = require('./lib/commander'); \ No newline at end of file diff --git a/node_modules/jade/node_modules/commander/lib/commander.js b/node_modules/jade/node_modules/commander/lib/commander.js deleted file mode 100644 index 5ba87eb..0000000 --- a/node_modules/jade/node_modules/commander/lib/commander.js +++ /dev/null @@ -1,1026 +0,0 @@ - -/*! - * commander - * Copyright(c) 2011 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Module dependencies. - */ - -var EventEmitter = require('events').EventEmitter - , path = require('path') - , tty = require('tty') - , basename = path.basename; - -/** - * Expose the root command. - */ - -exports = module.exports = new Command; - -/** - * Expose `Command`. - */ - -exports.Command = Command; - -/** - * Expose `Option`. - */ - -exports.Option = Option; - -/** - * Initialize a new `Option` with the given `flags` and `description`. - * - * @param {String} flags - * @param {String} description - * @api public - */ - -function Option(flags, description) { - this.flags = flags; - this.required = ~flags.indexOf('<'); - this.optional = ~flags.indexOf('['); - this.bool = !~flags.indexOf('-no-'); - flags = flags.split(/[ ,|]+/); - if (flags.length > 1 && !/^[[<]/.test(flags[1])) this.short = flags.shift(); - this.long = flags.shift(); - this.description = description; -} - -/** - * Return option name. - * - * @return {String} - * @api private - */ - -Option.prototype.name = function(){ - return this.long - .replace('--', '') - .replace('no-', ''); -}; - -/** - * Check if `arg` matches the short or long flag. - * - * @param {String} arg - * @return {Boolean} - * @api private - */ - -Option.prototype.is = function(arg){ - return arg == this.short - || arg == this.long; -}; - -/** - * Initialize a new `Command`. - * - * @param {String} name - * @api public - */ - -function Command(name) { - this.commands = []; - this.options = []; - this.args = []; - this.name = name; -} - -/** - * Inherit from `EventEmitter.prototype`. - */ - -Command.prototype.__proto__ = EventEmitter.prototype; - -/** - * Add command `name`. - * - * The `.action()` callback is invoked when the - * command `name` is specified via __ARGV__, - * and the remaining arguments are applied to the - * function for access. - * - * When the `name` is "*" an un-matched command - * will be passed as the first arg, followed by - * the rest of __ARGV__ remaining. - * - * Examples: - * - * program - * .version('0.0.1') - * .option('-C, --chdir ', 'change the working directory') - * .option('-c, --config ', 'set config path. defaults to ./deploy.conf') - * .option('-T, --no-tests', 'ignore test hook') - * - * program - * .command('setup') - * .description('run remote setup commands') - * .action(function(){ - * console.log('setup'); - * }); - * - * program - * .command('exec ') - * .description('run the given remote command') - * .action(function(cmd){ - * console.log('exec "%s"', cmd); - * }); - * - * program - * .command('*') - * .description('deploy the given env') - * .action(function(env){ - * console.log('deploying "%s"', env); - * }); - * - * program.parse(process.argv); - * - * @param {String} name - * @return {Command} the new command - * @api public - */ - -Command.prototype.command = function(name){ - var args = name.split(/ +/); - var cmd = new Command(args.shift()); - this.commands.push(cmd); - cmd.parseExpectedArgs(args); - cmd.parent = this; - return cmd; -}; - -/** - * Parse expected `args`. - * - * For example `["[type]"]` becomes `[{ required: false, name: 'type' }]`. - * - * @param {Array} args - * @return {Command} for chaining - * @api public - */ - -Command.prototype.parseExpectedArgs = function(args){ - if (!args.length) return; - var self = this; - args.forEach(function(arg){ - switch (arg[0]) { - case '<': - self.args.push({ required: true, name: arg.slice(1, -1) }); - break; - case '[': - self.args.push({ required: false, name: arg.slice(1, -1) }); - break; - } - }); - return this; -}; - -/** - * Register callback `fn` for the command. - * - * Examples: - * - * program - * .command('help') - * .description('display verbose help') - * .action(function(){ - * // output help here - * }); - * - * @param {Function} fn - * @return {Command} for chaining - * @api public - */ - -Command.prototype.action = function(fn){ - var self = this; - this.parent.on(this.name, function(args, unknown){ - // Parse any so-far unknown options - unknown = unknown || []; - var parsed = self.parseOptions(unknown); - - // Output help if necessary - outputHelpIfNecessary(self, parsed.unknown); - - // If there are still any unknown options, then we simply - // die, unless someone asked for help, in which case we give it - // to them, and then we die. - if (parsed.unknown.length > 0) { - self.unknownOption(parsed.unknown[0]); - } - - self.args.forEach(function(arg, i){ - if (arg.required && null == args[i]) { - self.missingArgument(arg.name); - } - }); - - // Always append ourselves to the end of the arguments, - // to make sure we match the number of arguments the user - // expects - if (self.args.length) { - args[self.args.length] = self; - } else { - args.push(self); - } - - fn.apply(this, args); - }); - return this; -}; - -/** - * Define option with `flags`, `description` and optional - * coercion `fn`. - * - * The `flags` string should contain both the short and long flags, - * separated by comma, a pipe or space. The following are all valid - * all will output this way when `--help` is used. - * - * "-p, --pepper" - * "-p|--pepper" - * "-p --pepper" - * - * Examples: - * - * // simple boolean defaulting to false - * program.option('-p, --pepper', 'add pepper'); - * - * --pepper - * program.pepper - * // => Boolean - * - * // simple boolean defaulting to false - * program.option('-C, --no-cheese', 'remove cheese'); - * - * program.cheese - * // => true - * - * --no-cheese - * program.cheese - * // => true - * - * // required argument - * program.option('-C, --chdir ', 'change the working directory'); - * - * --chdir /tmp - * program.chdir - * // => "/tmp" - * - * // optional argument - * program.option('-c, --cheese [type]', 'add cheese [marble]'); - * - * @param {String} flags - * @param {String} description - * @param {Function|Mixed} fn or default - * @param {Mixed} defaultValue - * @return {Command} for chaining - * @api public - */ - -Command.prototype.option = function(flags, description, fn, defaultValue){ - var self = this - , option = new Option(flags, description) - , oname = option.name() - , name = camelcase(oname); - - // default as 3rd arg - if ('function' != typeof fn) defaultValue = fn, fn = null; - - // preassign default value only for --no-*, [optional], or - if (false == option.bool || option.optional || option.required) { - // when --no-* we make sure default is true - if (false == option.bool) defaultValue = true; - // preassign only if we have a default - if (undefined !== defaultValue) self[name] = defaultValue; - } - - // register the option - this.options.push(option); - - // when it's passed assign the value - // and conditionally invoke the callback - this.on(oname, function(val){ - // coercion - if (null != val && fn) val = fn(val); - - // unassigned or bool - if ('boolean' == typeof self[name] || 'undefined' == typeof self[name]) { - // if no value, bool true, and we have a default, then use it! - if (null == val) { - self[name] = option.bool - ? defaultValue || true - : false; - } else { - self[name] = val; - } - } else if (null !== val) { - // reassign - self[name] = val; - } - }); - - return this; -}; - -/** - * Parse `argv`, settings options and invoking commands when defined. - * - * @param {Array} argv - * @return {Command} for chaining - * @api public - */ - -Command.prototype.parse = function(argv){ - // store raw args - this.rawArgs = argv; - - // guess name - if (!this.name) this.name = basename(argv[1]); - - // process argv - var parsed = this.parseOptions(this.normalize(argv.slice(2))); - this.args = parsed.args; - return this.parseArgs(this.args, parsed.unknown); -}; - -/** - * Normalize `args`, splitting joined short flags. For example - * the arg "-abc" is equivalent to "-a -b -c". - * - * @param {Array} args - * @return {Array} - * @api private - */ - -Command.prototype.normalize = function(args){ - var ret = [] - , arg; - - for (var i = 0, len = args.length; i < len; ++i) { - arg = args[i]; - if (arg.length > 1 && '-' == arg[0] && '-' != arg[1]) { - arg.slice(1).split('').forEach(function(c){ - ret.push('-' + c); - }); - } else { - ret.push(arg); - } - } - - return ret; -}; - -/** - * Parse command `args`. - * - * When listener(s) are available those - * callbacks are invoked, otherwise the "*" - * event is emitted and those actions are invoked. - * - * @param {Array} args - * @return {Command} for chaining - * @api private - */ - -Command.prototype.parseArgs = function(args, unknown){ - var cmds = this.commands - , len = cmds.length - , name; - - if (args.length) { - name = args[0]; - if (this.listeners(name).length) { - this.emit(args.shift(), args, unknown); - } else { - this.emit('*', args); - } - } else { - outputHelpIfNecessary(this, unknown); - - // If there were no args and we have unknown options, - // then they are extraneous and we need to error. - if (unknown.length > 0) { - this.unknownOption(unknown[0]); - } - } - - return this; -}; - -/** - * Return an option matching `arg` if any. - * - * @param {String} arg - * @return {Option} - * @api private - */ - -Command.prototype.optionFor = function(arg){ - for (var i = 0, len = this.options.length; i < len; ++i) { - if (this.options[i].is(arg)) { - return this.options[i]; - } - } -}; - -/** - * Parse options from `argv` returning `argv` - * void of these options. - * - * @param {Array} argv - * @return {Array} - * @api public - */ - -Command.prototype.parseOptions = function(argv){ - var args = [] - , len = argv.length - , literal - , option - , arg; - - var unknownOptions = []; - - // parse options - for (var i = 0; i < len; ++i) { - arg = argv[i]; - - // literal args after -- - if ('--' == arg) { - literal = true; - continue; - } - - if (literal) { - args.push(arg); - continue; - } - - // find matching Option - option = this.optionFor(arg); - - // option is defined - if (option) { - // requires arg - if (option.required) { - arg = argv[++i]; - if (null == arg) return this.optionMissingArgument(option); - if ('-' == arg[0]) return this.optionMissingArgument(option, arg); - this.emit(option.name(), arg); - // optional arg - } else if (option.optional) { - arg = argv[i+1]; - if (null == arg || '-' == arg[0]) { - arg = null; - } else { - ++i; - } - this.emit(option.name(), arg); - // bool - } else { - this.emit(option.name()); - } - continue; - } - - // looks like an option - if (arg.length > 1 && '-' == arg[0]) { - unknownOptions.push(arg); - - // If the next argument looks like it might be - // an argument for this option, we pass it on. - // If it isn't, then it'll simply be ignored - if (argv[i+1] && '-' != argv[i+1][0]) { - unknownOptions.push(argv[++i]); - } - continue; - } - - // arg - args.push(arg); - } - - return { args: args, unknown: unknownOptions }; -}; - -/** - * Argument `name` is missing. - * - * @param {String} name - * @api private - */ - -Command.prototype.missingArgument = function(name){ - console.error(); - console.error(" error: missing required argument `%s'", name); - console.error(); - process.exit(1); -}; - -/** - * `Option` is missing an argument, but received `flag` or nothing. - * - * @param {String} option - * @param {String} flag - * @api private - */ - -Command.prototype.optionMissingArgument = function(option, flag){ - console.error(); - if (flag) { - console.error(" error: option `%s' argument missing, got `%s'", option.flags, flag); - } else { - console.error(" error: option `%s' argument missing", option.flags); - } - console.error(); - process.exit(1); -}; - -/** - * Unknown option `flag`. - * - * @param {String} flag - * @api private - */ - -Command.prototype.unknownOption = function(flag){ - console.error(); - console.error(" error: unknown option `%s'", flag); - console.error(); - process.exit(1); -}; - -/** - * Set the program version to `str`. - * - * This method auto-registers the "-V, --version" flag - * which will print the version number when passed. - * - * @param {String} str - * @param {String} flags - * @return {Command} for chaining - * @api public - */ - -Command.prototype.version = function(str, flags){ - if (0 == arguments.length) return this._version; - this._version = str; - flags = flags || '-V, --version'; - this.option(flags, 'output the version number'); - this.on('version', function(){ - console.log(str); - process.exit(0); - }); - return this; -}; - -/** - * Set the description `str`. - * - * @param {String} str - * @return {String|Command} - * @api public - */ - -Command.prototype.description = function(str){ - if (0 == arguments.length) return this._description; - this._description = str; - return this; -}; - -/** - * Set / get the command usage `str`. - * - * @param {String} str - * @return {String|Command} - * @api public - */ - -Command.prototype.usage = function(str){ - var args = this.args.map(function(arg){ - return arg.required - ? '<' + arg.name + '>' - : '[' + arg.name + ']'; - }); - - var usage = '[options' - + (this.commands.length ? '] [command' : '') - + ']' - + (this.args.length ? ' ' + args : ''); - if (0 == arguments.length) return this._usage || usage; - this._usage = str; - - return this; -}; - -/** - * Return the largest option length. - * - * @return {Number} - * @api private - */ - -Command.prototype.largestOptionLength = function(){ - return this.options.reduce(function(max, option){ - return Math.max(max, option.flags.length); - }, 0); -}; - -/** - * Return help for options. - * - * @return {String} - * @api private - */ - -Command.prototype.optionHelp = function(){ - var width = this.largestOptionLength(); - - // Prepend the help information - return [pad('-h, --help', width) + ' ' + 'output usage information'] - .concat(this.options.map(function(option){ - return pad(option.flags, width) - + ' ' + option.description; - })) - .join('\n'); -}; - -/** - * Return command help documentation. - * - * @return {String} - * @api private - */ - -Command.prototype.commandHelp = function(){ - if (!this.commands.length) return ''; - return [ - '' - , ' Commands:' - , '' - , this.commands.map(function(cmd){ - var args = cmd.args.map(function(arg){ - return arg.required - ? '<' + arg.name + '>' - : '[' + arg.name + ']'; - }).join(' '); - - return cmd.name - + (cmd.options.length - ? ' [options]' - : '') + ' ' + args - + (cmd.description() - ? '\n' + cmd.description() - : ''); - }).join('\n\n').replace(/^/gm, ' ') - , '' - ].join('\n'); -}; - -/** - * Return program help documentation. - * - * @return {String} - * @api private - */ - -Command.prototype.helpInformation = function(){ - return [ - '' - , ' Usage: ' + this.name + ' ' + this.usage() - , '' + this.commandHelp() - , ' Options:' - , '' - , '' + this.optionHelp().replace(/^/gm, ' ') - , '' - , '' - ].join('\n'); -}; - -/** - * Prompt for a `Number`. - * - * @param {String} str - * @param {Function} fn - * @api private - */ - -Command.prototype.promptForNumber = function(str, fn){ - var self = this; - this.promptSingleLine(str, function parseNumber(val){ - val = Number(val); - if (isNaN(val)) return self.promptSingleLine(str + '(must be a number) ', parseNumber); - fn(val); - }); -}; - -/** - * Prompt for a `Date`. - * - * @param {String} str - * @param {Function} fn - * @api private - */ - -Command.prototype.promptForDate = function(str, fn){ - var self = this; - this.promptSingleLine(str, function parseDate(val){ - val = new Date(val); - if (isNaN(val.getTime())) return self.promptSingleLine(str + '(must be a date) ', parseDate); - fn(val); - }); -}; - -/** - * Single-line prompt. - * - * @param {String} str - * @param {Function} fn - * @api private - */ - -Command.prototype.promptSingleLine = function(str, fn){ - if ('function' == typeof arguments[2]) { - return this['promptFor' + (fn.name || fn)](str, arguments[2]); - } - - process.stdout.write(str); - process.stdin.setEncoding('utf8'); - process.stdin.once('data', function(val){ - fn(val.trim()); - }).resume(); -}; - -/** - * Multi-line prompt. - * - * @param {String} str - * @param {Function} fn - * @api private - */ - -Command.prototype.promptMultiLine = function(str, fn){ - var buf = []; - console.log(str); - process.stdin.setEncoding('utf8'); - process.stdin.on('data', function(val){ - if ('\n' == val || '\r\n' == val) { - process.stdin.removeAllListeners('data'); - fn(buf.join('\n')); - } else { - buf.push(val.trimRight()); - } - }).resume(); -}; - -/** - * Prompt `str` and callback `fn(val)` - * - * Commander supports single-line and multi-line prompts. - * To issue a single-line prompt simply add white-space - * to the end of `str`, something like "name: ", whereas - * for a multi-line prompt omit this "description:". - * - * - * Examples: - * - * program.prompt('Username: ', function(name){ - * console.log('hi %s', name); - * }); - * - * program.prompt('Description:', function(desc){ - * console.log('description was "%s"', desc.trim()); - * }); - * - * @param {String|Object} str - * @param {Function} fn - * @api public - */ - -Command.prototype.prompt = function(str, fn){ - var self = this; - - if ('string' == typeof str) { - if (/ $/.test(str)) return this.promptSingleLine.apply(this, arguments); - this.promptMultiLine(str, fn); - } else { - var keys = Object.keys(str) - , obj = {}; - - function next() { - var key = keys.shift() - , label = str[key]; - - if (!key) return fn(obj); - self.prompt(label, function(val){ - obj[key] = val; - next(); - }); - } - - next(); - } -}; - -/** - * Prompt for password with `str`, `mask` char and callback `fn(val)`. - * - * The mask string defaults to '', aka no output is - * written while typing, you may want to use "*" etc. - * - * Examples: - * - * program.password('Password: ', function(pass){ - * console.log('got "%s"', pass); - * process.stdin.destroy(); - * }); - * - * program.password('Password: ', '*', function(pass){ - * console.log('got "%s"', pass); - * process.stdin.destroy(); - * }); - * - * @param {String} str - * @param {String} mask - * @param {Function} fn - * @api public - */ - -Command.prototype.password = function(str, mask, fn){ - var self = this - , buf = ''; - - // default mask - if ('function' == typeof mask) { - fn = mask; - mask = ''; - } - - process.stdin.resume(); - tty.setRawMode(true); - process.stdout.write(str); - - // keypress - process.stdin.on('keypress', function(c, key){ - if (key && 'enter' == key.name) { - console.log(); - process.stdin.removeAllListeners('keypress'); - tty.setRawMode(false); - if (!buf.trim().length) return self.password(str, mask, fn); - fn(buf); - return; - } - - if (key && key.ctrl && 'c' == key.name) { - console.log('%s', buf); - process.exit(); - } - - process.stdout.write(mask); - buf += c; - }).resume(); -}; - -/** - * Confirmation prompt with `str` and callback `fn(bool)` - * - * Examples: - * - * program.confirm('continue? ', function(ok){ - * console.log(' got %j', ok); - * process.stdin.destroy(); - * }); - * - * @param {String} str - * @param {Function} fn - * @api public - */ - - -Command.prototype.confirm = function(str, fn, verbose){ - var self = this; - this.prompt(str, function(ok){ - if (!ok.trim()) { - if (!verbose) str += '(yes or no) '; - return self.confirm(str, fn, true); - } - fn(parseBool(ok)); - }); -}; - -/** - * Choice prompt with `list` of items and callback `fn(index, item)` - * - * Examples: - * - * var list = ['tobi', 'loki', 'jane', 'manny', 'luna']; - * - * console.log('Choose the coolest pet:'); - * program.choose(list, function(i){ - * console.log('you chose %d "%s"', i, list[i]); - * process.stdin.destroy(); - * }); - * - * @param {Array} list - * @param {Number|Function} index or fn - * @param {Function} fn - * @api public - */ - -Command.prototype.choose = function(list, index, fn){ - var self = this - , hasDefault = 'number' == typeof index; - - if (!hasDefault) { - fn = index; - index = null; - } - - list.forEach(function(item, i){ - if (hasDefault && i == index) { - console.log('* %d) %s', i + 1, item); - } else { - console.log(' %d) %s', i + 1, item); - } - }); - - function again() { - self.prompt(' : ', function(val){ - val = parseInt(val, 10) - 1; - if (hasDefault && isNaN(val)) val = index; - - if (null == list[val]) { - again(); - } else { - fn(val, list[val]); - } - }); - } - - again(); -}; - -/** - * Camel-case the given `flag` - * - * @param {String} flag - * @return {String} - * @api private - */ - -function camelcase(flag) { - return flag.split('-').reduce(function(str, word){ - return str + word[0].toUpperCase() + word.slice(1); - }); -} - -/** - * Parse a boolean `str`. - * - * @param {String} str - * @return {Boolean} - * @api private - */ - -function parseBool(str) { - return /^y|yes|ok|true$/i.test(str); -} - -/** - * Pad `str` to `width`. - * - * @param {String} str - * @param {Number} width - * @return {String} - * @api private - */ - -function pad(str, width) { - var len = Math.max(0, width - str.length); - return str + Array(len + 1).join(' '); -} - -/** - * Output help information if necessary - * - * @param {Command} command to output help for - * @param {Array} array of options to search for -h or --help - * @api private - */ - -function outputHelpIfNecessary(cmd, options) { - options = options || []; - for (var i = 0; i < options.length; i++) { - if (options[i] == '--help' || options[i] == '-h') { - process.stdout.write(cmd.helpInformation()); - cmd.emit('--help'); - process.exit(0); - } - } -} diff --git a/node_modules/jade/node_modules/commander/package.json b/node_modules/jade/node_modules/commander/package.json deleted file mode 100644 index 6f9d567..0000000 --- a/node_modules/jade/node_modules/commander/package.json +++ /dev/null @@ -1,38 +0,0 @@ -{ - "name": "commander", - "version": "0.6.1", - "description": "the complete solution for node.js command-line programs", - "keywords": [ - "command", - "option", - "parser", - "prompt", - "stdin" - ], - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca" - }, - "repository": { - "type": "git", - "url": "https://github.com/visionmedia/commander.js.git" - }, - "dependencies": {}, - "devDependencies": { - "should": ">= 0.0.1" - }, - "scripts": { - "test": "make test" - }, - "main": "index", - "engines": { - "node": ">= 0.4.x" - }, - "readme": "# Commander.js\n\n The complete solution for [node.js](http://nodejs.org) command-line interfaces, inspired by Ruby's [commander](https://github.com/visionmedia/commander).\n\n [![Build Status](https://secure.travis-ci.org/visionmedia/commander.js.png)](http://travis-ci.org/visionmedia/commander.js)\n\n## Installation\n\n $ npm install commander\n\n## Option parsing\n\n Options with commander are defined with the `.option()` method, also serving as documentation for the options. The example below parses args and options from `process.argv`, leaving remaining args as the `program.args` array which were not consumed by options.\n\n```js\n#!/usr/bin/env node\n\n/**\n * Module dependencies.\n */\n\nvar program = require('commander');\n\nprogram\n .version('0.0.1')\n .option('-p, --peppers', 'Add peppers')\n .option('-P, --pineapple', 'Add pineapple')\n .option('-b, --bbq', 'Add bbq sauce')\n .option('-c, --cheese [type]', 'Add the specified type of cheese [marble]', 'marble')\n .parse(process.argv);\n\nconsole.log('you ordered a pizza with:');\nif (program.peppers) console.log(' - peppers');\nif (program.pineapple) console.log(' - pineappe');\nif (program.bbq) console.log(' - bbq');\nconsole.log(' - %s cheese', program.cheese);\n```\n\n Short flags may be passed as a single arg, for example `-abc` is equivalent to `-a -b -c`. Multi-word options such as \"--template-engine\" are camel-cased, becoming `program.templateEngine` etc.\n\n## Automated --help\n\n The help information is auto-generated based on the information commander already knows about your program, so the following `--help` info is for free:\n\n``` \n $ ./examples/pizza --help\n\n Usage: pizza [options]\n\n Options:\n\n -V, --version output the version number\n -p, --peppers Add peppers\n -P, --pineapple Add pineappe\n -b, --bbq Add bbq sauce\n -c, --cheese Add the specified type of cheese [marble]\n -h, --help output usage information\n\n```\n\n## Coercion\n\n```js\nfunction range(val) {\n return val.split('..').map(Number);\n}\n\nfunction list(val) {\n return val.split(',');\n}\n\nprogram\n .version('0.0.1')\n .usage('[options] ')\n .option('-i, --integer ', 'An integer argument', parseInt)\n .option('-f, --float ', 'A float argument', parseFloat)\n .option('-r, --range ..', 'A range', range)\n .option('-l, --list ', 'A list', list)\n .option('-o, --optional [value]', 'An optional value')\n .parse(process.argv);\n\nconsole.log(' int: %j', program.integer);\nconsole.log(' float: %j', program.float);\nconsole.log(' optional: %j', program.optional);\nprogram.range = program.range || [];\nconsole.log(' range: %j..%j', program.range[0], program.range[1]);\nconsole.log(' list: %j', program.list);\nconsole.log(' args: %j', program.args);\n```\n\n## Custom help\n\n You can display arbitrary `-h, --help` information\n by listening for \"--help\". Commander will automatically\n exit once you are done so that the remainder of your program\n does not execute causing undesired behaviours, for example\n in the following executable \"stuff\" will not output when\n `--help` is used.\n\n```js\n#!/usr/bin/env node\n\n/**\n * Module dependencies.\n */\n\nvar program = require('../');\n\nfunction list(val) {\n return val.split(',').map(Number);\n}\n\nprogram\n .version('0.0.1')\n .option('-f, --foo', 'enable some foo')\n .option('-b, --bar', 'enable some bar')\n .option('-B, --baz', 'enable some baz');\n\n// must be before .parse() since\n// node's emit() is immediate\n\nprogram.on('--help', function(){\n console.log(' Examples:');\n console.log('');\n console.log(' $ custom-help --help');\n console.log(' $ custom-help -h');\n console.log('');\n});\n\nprogram.parse(process.argv);\n\nconsole.log('stuff');\n```\n\nyielding the following help output:\n\n```\n\nUsage: custom-help [options]\n\nOptions:\n\n -h, --help output usage information\n -V, --version output the version number\n -f, --foo enable some foo\n -b, --bar enable some bar\n -B, --baz enable some baz\n\nExamples:\n\n $ custom-help --help\n $ custom-help -h\n\n```\n\n## .prompt(msg, fn)\n\n Single-line prompt:\n\n```js\nprogram.prompt('name: ', function(name){\n console.log('hi %s', name);\n});\n```\n\n Multi-line prompt:\n\n```js\nprogram.prompt('description:', function(name){\n console.log('hi %s', name);\n});\n```\n\n Coercion:\n\n```js\nprogram.prompt('Age: ', Number, function(age){\n console.log('age: %j', age);\n});\n```\n\n```js\nprogram.prompt('Birthdate: ', Date, function(date){\n console.log('date: %s', date);\n});\n```\n\n## .password(msg[, mask], fn)\n\nPrompt for password without echoing:\n\n```js\nprogram.password('Password: ', function(pass){\n console.log('got \"%s\"', pass);\n process.stdin.destroy();\n});\n```\n\nPrompt for password with mask char \"*\":\n\n```js\nprogram.password('Password: ', '*', function(pass){\n console.log('got \"%s\"', pass);\n process.stdin.destroy();\n});\n```\n\n## .confirm(msg, fn)\n\n Confirm with the given `msg`:\n\n```js\nprogram.confirm('continue? ', function(ok){\n console.log(' got %j', ok);\n});\n```\n\n## .choose(list, fn)\n\n Let the user choose from a `list`:\n\n```js\nvar list = ['tobi', 'loki', 'jane', 'manny', 'luna'];\n\nconsole.log('Choose the coolest pet:');\nprogram.choose(list, function(i){\n console.log('you chose %d \"%s\"', i, list[i]);\n});\n```\n\n## Links\n\n - [API documentation](http://visionmedia.github.com/commander.js/)\n - [ascii tables](https://github.com/LearnBoost/cli-table)\n - [progress bars](https://github.com/visionmedia/node-progress)\n - [more progress bars](https://github.com/substack/node-multimeter)\n - [examples](https://github.com/visionmedia/commander.js/tree/master/examples)\n\n## License \n\n(The MIT License)\n\nCopyright (c) 2011 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.", - "readmeFilename": "Readme.md", - "bugs": { - "url": "https://github.com/visionmedia/commander.js/issues" - }, - "_id": "commander@0.6.1", - "_from": "commander@0.6.1" -} diff --git a/node_modules/jade/node_modules/mkdirp/.npmignore b/node_modules/jade/node_modules/mkdirp/.npmignore deleted file mode 100644 index 9303c34..0000000 --- a/node_modules/jade/node_modules/mkdirp/.npmignore +++ /dev/null @@ -1,2 +0,0 @@ -node_modules/ -npm-debug.log \ No newline at end of file diff --git a/node_modules/jade/node_modules/mkdirp/.travis.yml b/node_modules/jade/node_modules/mkdirp/.travis.yml deleted file mode 100644 index 84fd7ca..0000000 --- a/node_modules/jade/node_modules/mkdirp/.travis.yml +++ /dev/null @@ -1,5 +0,0 @@ -language: node_js -node_js: - - 0.6 - - 0.8 - - 0.9 diff --git a/node_modules/jade/node_modules/mkdirp/LICENSE b/node_modules/jade/node_modules/mkdirp/LICENSE deleted file mode 100644 index 432d1ae..0000000 --- a/node_modules/jade/node_modules/mkdirp/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -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/node_modules/jade/node_modules/mkdirp/examples/pow.js b/node_modules/jade/node_modules/mkdirp/examples/pow.js deleted file mode 100644 index e692421..0000000 --- a/node_modules/jade/node_modules/mkdirp/examples/pow.js +++ /dev/null @@ -1,6 +0,0 @@ -var mkdirp = require('mkdirp'); - -mkdirp('/tmp/foo/bar/baz', function (err) { - if (err) console.error(err) - else console.log('pow!') -}); diff --git a/node_modules/jade/node_modules/mkdirp/index.js b/node_modules/jade/node_modules/mkdirp/index.js deleted file mode 100644 index fda6de8..0000000 --- a/node_modules/jade/node_modules/mkdirp/index.js +++ /dev/null @@ -1,82 +0,0 @@ -var path = require('path'); -var fs = require('fs'); - -module.exports = mkdirP.mkdirp = mkdirP.mkdirP = mkdirP; - -function mkdirP (p, mode, f, made) { - if (typeof mode === 'function' || mode === undefined) { - f = mode; - mode = 0777 & (~process.umask()); - } - if (!made) made = null; - - var cb = f || function () {}; - if (typeof mode === 'string') mode = parseInt(mode, 8); - p = path.resolve(p); - - fs.mkdir(p, mode, function (er) { - if (!er) { - made = made || p; - return cb(null, made); - } - switch (er.code) { - case 'ENOENT': - mkdirP(path.dirname(p), mode, function (er, made) { - if (er) cb(er, made); - else mkdirP(p, mode, cb, made); - }); - break; - - // In the case of any other error, just see if there's a dir - // there already. If so, then hooray! If not, then something - // is borked. - default: - fs.stat(p, function (er2, stat) { - // if the stat fails, then that's super weird. - // let the original error be the failure reason. - if (er2 || !stat.isDirectory()) cb(er, made) - else cb(null, made); - }); - break; - } - }); -} - -mkdirP.sync = function sync (p, mode, made) { - if (mode === undefined) { - mode = 0777 & (~process.umask()); - } - if (!made) made = null; - - if (typeof mode === 'string') mode = parseInt(mode, 8); - p = path.resolve(p); - - try { - fs.mkdirSync(p, mode); - made = made || p; - } - catch (err0) { - switch (err0.code) { - case 'ENOENT' : - made = sync(path.dirname(p), mode, made); - sync(p, mode, made); - break; - - // In the case of any other error, just see if there's a dir - // there already. If so, then hooray! If not, then something - // is borked. - default: - var stat; - try { - stat = fs.statSync(p); - } - catch (err1) { - throw err0; - } - if (!stat.isDirectory()) throw err0; - break; - } - } - - return made; -}; diff --git a/node_modules/jade/node_modules/mkdirp/package.json b/node_modules/jade/node_modules/mkdirp/package.json deleted file mode 100644 index bd9194b..0000000 --- a/node_modules/jade/node_modules/mkdirp/package.json +++ /dev/null @@ -1,33 +0,0 @@ -{ - "name": "mkdirp", - "description": "Recursively mkdir, like `mkdir -p`", - "version": "0.3.5", - "author": { - "name": "James Halliday", - "email": "mail@substack.net", - "url": "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.4.0" - }, - "license": "MIT", - "readme": "# mkdirp\n\nLike `mkdir -p`, but in node.js!\n\n[![build status](https://secure.travis-ci.org/substack/node-mkdirp.png)](http://travis-ci.org/substack/node-mkdirp)\n\n# example\n\n## pow.js\n\n```js\nvar mkdirp = require('mkdirp');\n \nmkdirp('/tmp/foo/bar/baz', function (err) {\n if (err) console.error(err)\n else console.log('pow!')\n});\n```\n\nOutput\n\n```\npow!\n```\n\nAnd now /tmp/foo/bar/baz exists, huzzah!\n\n# methods\n\n```js\nvar mkdirp = require('mkdirp');\n```\n\n## mkdirp(dir, mode, cb)\n\nCreate a new directory and any necessary subdirectories at `dir` with octal\npermission string `mode`.\n\nIf `mode` isn't specified, it defaults to `0777 & (~process.umask())`.\n\n`cb(err, made)` fires with the error or the first directory `made`\nthat had to be created, if any.\n\n## mkdirp.sync(dir, mode)\n\nSynchronously create a new directory and any necessary subdirectories at `dir`\nwith octal permission string `mode`.\n\nIf `mode` isn't specified, it defaults to `0777 & (~process.umask())`.\n\nReturns the first directory that had to be created, if any.\n\n# install\n\nWith [npm](http://npmjs.org) do:\n\n```\nnpm install mkdirp\n```\n\n# license\n\nMIT\n", - "readmeFilename": "readme.markdown", - "bugs": { - "url": "https://github.com/substack/node-mkdirp/issues" - }, - "_id": "mkdirp@0.3.5", - "_from": "mkdirp@0.3.x" -} diff --git a/node_modules/jade/node_modules/mkdirp/readme.markdown b/node_modules/jade/node_modules/mkdirp/readme.markdown deleted file mode 100644 index 83b0216..0000000 --- a/node_modules/jade/node_modules/mkdirp/readme.markdown +++ /dev/null @@ -1,63 +0,0 @@ -# mkdirp - -Like `mkdir -p`, but in node.js! - -[![build status](https://secure.travis-ci.org/substack/node-mkdirp.png)](http://travis-ci.org/substack/node-mkdirp) - -# example - -## pow.js - -```js -var mkdirp = require('mkdirp'); - -mkdirp('/tmp/foo/bar/baz', function (err) { - if (err) console.error(err) - else console.log('pow!') -}); -``` - -Output - -``` -pow! -``` - -And now /tmp/foo/bar/baz exists, huzzah! - -# methods - -```js -var mkdirp = require('mkdirp'); -``` - -## mkdirp(dir, mode, cb) - -Create a new directory and any necessary subdirectories at `dir` with octal -permission string `mode`. - -If `mode` isn't specified, it defaults to `0777 & (~process.umask())`. - -`cb(err, made)` fires with the error or the first directory `made` -that had to be created, if any. - -## mkdirp.sync(dir, mode) - -Synchronously create a new directory and any necessary subdirectories at `dir` -with octal permission string `mode`. - -If `mode` isn't specified, it defaults to `0777 & (~process.umask())`. - -Returns the first directory that had to be created, if any. - -# install - -With [npm](http://npmjs.org) do: - -``` -npm install mkdirp -``` - -# license - -MIT diff --git a/node_modules/jade/node_modules/mkdirp/test/chmod.js b/node_modules/jade/node_modules/mkdirp/test/chmod.js deleted file mode 100644 index 520dcb8..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/chmod.js +++ /dev/null @@ -1,38 +0,0 @@ -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.end(); - }); - }); -}); diff --git a/node_modules/jade/node_modules/mkdirp/test/clobber.js b/node_modules/jade/node_modules/mkdirp/test/clobber.js deleted file mode 100644 index 0eb7099..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/clobber.js +++ /dev/null @@ -1,37 +0,0 @@ -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/node_modules/jade/node_modules/mkdirp/test/mkdirp.js b/node_modules/jade/node_modules/mkdirp/test/mkdirp.js deleted file mode 100644 index b07cd70..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/mkdirp.js +++ /dev/null @@ -1,28 +0,0 @@ -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/node_modules/jade/node_modules/mkdirp/test/perm.js b/node_modules/jade/node_modules/mkdirp/test/perm.js deleted file mode 100644 index 23a7abb..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/perm.js +++ /dev/null @@ -1,32 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('async perm', function (t) { - t.plan(2); - var file = '/tmp/' + (Math.random() * (1<<30)).toString(16); - - 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(); - } - }) - }) - }); -}); - -test('async root perm', function (t) { - mkdirp('/tmp', 0755, function (err) { - if (err) t.fail(err); - t.end(); - }); - t.end(); -}); diff --git a/node_modules/jade/node_modules/mkdirp/test/perm_sync.js b/node_modules/jade/node_modules/mkdirp/test/perm_sync.js deleted file mode 100644 index f685f60..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/perm_sync.js +++ /dev/null @@ -1,39 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('sync perm', function (t) { - t.plan(2); - var file = '/tmp/' + (Math.random() * (1<<30)).toString(16) + '.json'; - - mkdirp.sync(file, 0755); - 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(); - } - }) - }); -}); - -test('sync root perm', function (t) { - t.plan(1); - - var file = '/tmp'; - mkdirp.sync(file, 0755); - 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.ok(stat.isDirectory(), 'target not a directory'); - t.end(); - } - }) - }); -}); diff --git a/node_modules/jade/node_modules/mkdirp/test/race.js b/node_modules/jade/node_modules/mkdirp/test/race.js deleted file mode 100644 index 96a0447..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/race.js +++ /dev/null @@ -1,41 +0,0 @@ -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/node_modules/jade/node_modules/mkdirp/test/rel.js b/node_modules/jade/node_modules/mkdirp/test/rel.js deleted file mode 100644 index 7985824..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/rel.js +++ /dev/null @@ -1,32 +0,0 @@ -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/node_modules/jade/node_modules/mkdirp/test/return.js b/node_modules/jade/node_modules/mkdirp/test/return.js deleted file mode 100644 index bce68e5..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/return.js +++ /dev/null @@ -1,25 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('return value', function (t) { - t.plan(4); - 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('/'); - - // should return the first dir created. - // By this point, it would be profoundly surprising if /tmp didn't - // already exist, since every other test makes things in there. - mkdirp(file, function (err, made) { - t.ifError(err); - t.equal(made, '/tmp/' + x); - mkdirp(file, function (err, made) { - t.ifError(err); - t.equal(made, null); - }); - }); -}); diff --git a/node_modules/jade/node_modules/mkdirp/test/return_sync.js b/node_modules/jade/node_modules/mkdirp/test/return_sync.js deleted file mode 100644 index 7c222d3..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/return_sync.js +++ /dev/null @@ -1,24 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('return value', 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('/'); - - // should return the first dir created. - // By this point, it would be profoundly surprising if /tmp didn't - // already exist, since every other test makes things in there. - // Note that this will throw on failure, which will fail the test. - var made = mkdirp.sync(file); - t.equal(made, '/tmp/' + x); - - // making the same file again should have no effect. - made = mkdirp.sync(file); - t.equal(made, null); -}); diff --git a/node_modules/jade/node_modules/mkdirp/test/root.js b/node_modules/jade/node_modules/mkdirp/test/root.js deleted file mode 100644 index 97ad7a2..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/root.js +++ /dev/null @@ -1,18 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('root', function (t) { - // '/' on unix, 'c:/' on windows. - var file = path.resolve('/'); - - mkdirp(file, 0755, function (err) { - if (err) throw err - fs.stat(file, function (er, stat) { - if (er) throw er - t.ok(stat.isDirectory(), 'target is a directory'); - t.end(); - }) - }); -}); diff --git a/node_modules/jade/node_modules/mkdirp/test/sync.js b/node_modules/jade/node_modules/mkdirp/test/sync.js deleted file mode 100644 index 7530cad..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/sync.js +++ /dev/null @@ -1,32 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('sync', 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('/'); - - try { - mkdirp.sync(file, 0755); - } catch (err) { - t.fail(err); - return t.end(); - } - - 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/node_modules/jade/node_modules/mkdirp/test/umask.js b/node_modules/jade/node_modules/mkdirp/test/umask.js deleted file mode 100644 index 64ccafe..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/umask.js +++ /dev/null @@ -1,28 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('implicit mode from umask', 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, 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, 0777 & (~process.umask())); - t.ok(stat.isDirectory(), 'target not a directory'); - t.end(); - } - }) - }) - }); -}); diff --git a/node_modules/jade/node_modules/mkdirp/test/umask_sync.js b/node_modules/jade/node_modules/mkdirp/test/umask_sync.js deleted file mode 100644 index 35bd5cb..0000000 --- a/node_modules/jade/node_modules/mkdirp/test/umask_sync.js +++ /dev/null @@ -1,32 +0,0 @@ -var mkdirp = require('../'); -var path = require('path'); -var fs = require('fs'); -var test = require('tap').test; - -test('umask sync modes', 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('/'); - - try { - mkdirp.sync(file); - } catch (err) { - t.fail(err); - return t.end(); - } - - 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, (0777 & (~process.umask()))); - t.ok(stat.isDirectory(), 'target not a directory'); - t.end(); - } - }); - }); -}); diff --git a/node_modules/jade/package.json b/node_modules/jade/package.json deleted file mode 100644 index 84d581a..0000000 --- a/node_modules/jade/package.json +++ /dev/null @@ -1,50 +0,0 @@ -{ - "name": "jade", - "description": "Jade template engine", - "version": "0.28.2", - "author": { - "name": "TJ Holowaychuk", - "email": "tj@vision-media.ca" - }, - "repository": { - "type": "git", - "url": "git://github.com/visionmedia/jade" - }, - "main": "./index.js", - "bin": { - "jade": "./bin/jade" - }, - "man": [ - "./jade.1" - ], - "dependencies": { - "commander": "0.6.1", - "mkdirp": "0.3.x" - }, - "devDependencies": { - "coffee-script": "~1.4.0", - "mocha": "*", - "markdown": "*", - "stylus": "*", - "uubench": "*", - "should": "*", - "less": "*", - "uglify-js": "*" - }, - "component": { - "scripts": { - "jade": "runtime.js" - } - }, - "scripts": { - "test": "mocha -R spec", - "prepublish": "npm prune" - }, - "readme": "# Jade - template engine \n[![Build Status](https://secure.travis-ci.org/visionmedia/jade.png)](http://travis-ci.org/visionmedia/jade)\n[![Dependency Status](https://gemnasium.com/visionmedia/jade.png)](https://gemnasium.com/visionmedia/jade)\n\n Jade is a high performance template engine heavily influenced by [Haml](http://haml-lang.com)\n and implemented with JavaScript for [node](http://nodejs.org). For discussion join the [Google Group](http://groups.google.com/group/jadejs).\n\n## Test drive\n\n You can test drive Jade online [here](http://naltatis.github.com/jade-syntax-docs).\n\n## README Contents\n\n- [Features](#a1)\n- [Implementations](#a2)\n- [Installation](#a3)\n- [Browser Support](#a4)\n- [Public API](#a5)\n- [Syntax](#a6)\n - [Line Endings](#a6-1)\n - [Tags](#a6-2)\n - [Tag Text](#a6-3)\n - [Comments](#a6-4)\n - [Block Comments](#a6-5)\n - [Nesting](#a6-6)\n - [Block Expansion](#a6-7)\n - [Case](#a6-8)\n - [Attributes](#a6-9)\n - [HTML](#a6-10)\n - [Doctypes](#a6-11)\n- [Filters](#a7)\n- [Code](#a8)\n- [Iteration](#a9)\n- [Conditionals](#a10)\n- [Template inheritance](#a11)\n- [Block append / prepend](#a12)\n- [Includes](#a13)\n- [Mixins](#a14)\n- [Generated Output](#a15)\n- [Example Makefile](#a16)\n- [jade(1)](#a17)\n- [Tutorials](#a18)\n- [License](#a19)\n\n\n## Features\n\n - client-side support\n - great readability\n - flexible indentation\n - block-expansion\n - mixins\n - static includes\n - attribute interpolation\n - code is escaped by default for security\n - contextual error reporting at compile & run time\n - executable for compiling jade templates via the command line\n - html 5 mode (the default doctype)\n - optional memory caching\n - combine dynamic and static tag classes\n - parse tree manipulation via _filters_\n - template inheritance\n - block append / prepend\n - supports [Express JS](http://expressjs.com) out of the box\n - transparent iteration over objects, arrays, and even non-enumerables via `each`\n - block comments\n - no tag prefix\n - filters\n - :stylus must have [stylus](http://github.com/LearnBoost/stylus) installed\n - :less must have [less.js](http://github.com/cloudhead/less.js) installed\n - :markdown must have [markdown-js](http://github.com/evilstreak/markdown-js), [node-discount](http://github.com/visionmedia/node-discount), or [marked](http://github.com/chjj/marked) installed\n - :cdata\n - :coffeescript must have [coffee-script](http://jashkenas.github.com/coffee-script/) installed\n - [Emacs Mode](https://github.com/brianc/jade-mode)\n - [Vim Syntax](https://github.com/digitaltoad/vim-jade)\n - [TextMate Bundle](http://github.com/miksago/jade-tmbundle)\n - [Coda/SubEtha syntax Mode](https://github.com/aaronmccall/jade.mode)\n - [Screencasts](http://tjholowaychuk.com/post/1004255394/jade-screencast-template-engine-for-nodejs)\n - [html2jade](https://github.com/donpark/html2jade) converter\n\n\n## Implementations\n\n - [php](http://github.com/everzet/jade.php)\n - [scala](http://scalate.fusesource.org/versions/snapshot/documentation/scaml-reference.html)\n - [ruby](http://github.com/stonean/slim)\n - [python](https://github.com/SyrusAkbary/pyjade)\n - [java](https://github.com/neuland/jade4j)\n\n\n## Installation\n\nvia npm:\n\n```bash\n$ npm install jade\n```\n\n\n## Browser Support\n\n To compile jade to a single file compatible for client-side use simply execute:\n\n```bash\n$ make jade.js\n```\n\n Alternatively, if uglifyjs is installed via npm (`npm install uglify-js`) you may execute the following which will create both files. However each release builds these for you.\n\n```bash\n$ make jade.min.js\n```\n\n By default Jade instruments templates with line number statements such as `__.lineno = 3` for debugging purposes. When used in a browser it's useful to minimize this boiler plate, you can do so by passing the option `{ compileDebug: false }`. The following template\n\n```jade\np Hello #{name}\n```\n\n Can then be as small as the following generated function:\n\n```js\nfunction anonymous(locals, attrs, escape, rethrow) {\n var buf = [];\n with (locals || {}) {\n var interp;\n buf.push('\\n

    Hello ' + escape((interp = name) == null ? '' : interp) + '\\n

    ');\n }\n return buf.join(\"\");\n}\n```\n\n Through the use of Jade's `./runtime.js` you may utilize these pre-compiled templates on the client-side _without_ Jade itself, all you need is the associated utility functions (in runtime.js), which are then available as `jade.attrs`, `jade.escape` etc. To enable this you should pass `{ client: true }` to `jade.compile()` to tell Jade to reference the helper functions\n via `jade.attrs`, `jade.escape` etc.\n\n```js\nfunction anonymous(locals, attrs, escape, rethrow) {\n var attrs = jade.attrs, escape = jade.escape, rethrow = jade.rethrow;\n var buf = [];\n with (locals || {}) {\n var interp;\n buf.push('\\n

    Hello ' + escape((interp = name) == null ? '' : interp) + '\\n

    ');\n }\n return buf.join(\"\");\n}\n```\n\n
    \n## Public API\n\n```js\nvar jade = require('jade');\n\n// Compile a function\nvar fn = jade.compile('string of jade', options);\nfn(locals);\n```\n\n### Options\n\n - `self` Use a `self` namespace to hold the locals _(false by default)_\n - `locals` Local variable object\n - `filename` Used in exceptions, and required when using includes\n - `debug` Outputs tokens and function body generated\n - `compiler` Compiler to replace jade's default\n - `compileDebug` When `false` no debug instrumentation is compiled\n - `pretty` Add pretty-indentation whitespace to output _(false by default)_\n\n\n## Syntax\n\n\n### Line Endings\n\n**CRLF** and **CR** are converted to **LF** before parsing.\n\n\n### Tags\n\nA tag is simply a leading word:\n\n```jade\nhtml\n```\n\nfor example is converted to ``\n\ntags can also have ids:\n\n```jade\ndiv#container\n```\n\nwhich would render `
    `\n\nhow about some classes?\n\n```jade\ndiv.user-details\n```\n\nrenders `
    `\n\nmultiple classes? _and_ an id? sure:\n\n```jade\ndiv#foo.bar.baz\n```\n\nrenders `
    `\n\ndiv div div sure is annoying, how about:\n\n```jade\n#foo\n.bar\n```\n\nwhich is syntactic sugar for what we have already been doing, and outputs:\n\n```html\n
    \n```\n\n
    \n### Tag Text\n\nSimply place some content after the tag:\n\n```jade\np wahoo!\n```\n\nrenders `

    wahoo!

    `.\n\nwell cool, but how about large bodies of text:\n\n```jade\np\n | foo bar baz\n | rawr rawr\n | super cool\n | go jade go\n```\n\nrenders `

    foo bar baz rawr.....

    `\n\ninterpolation? yup! both types of text can utilize interpolation,\nif we passed `{ name: 'tj', email: 'tj@vision-media.ca' }` to the compiled function we can do the following:\n\n```jade\n#user #{name} <#{email}>\n```\n\noutputs `
    tj <tj@vision-media.ca>
    `\n\nActually want `#{}` for some reason? escape it!\n\n```jade\np \\#{something}\n```\n\nnow we have `

    #{something}

    `\n\nWe can also utilize the unescaped variant `!{html}`, so the following\nwill result in a literal script tag:\n\n```jade\n- var html = \"\"\n| !{html}\n```\n\nNested tags that also contain text can optionally use a text block:\n\n```jade\nlabel\n | Username:\n input(name='user[name]')\n```\n\nor immediate tag text:\n\n```jade\nlabel Username:\n input(name='user[name]')\n```\n\nTags that accept _only_ text such as `script` and `style` do not\nneed the leading `|` character, for example:\n\n```jade\nhtml\n head\n title Example\n script\n if (foo) {\n bar();\n } else {\n baz();\n }\n```\n\nOnce again as an alternative, we may use a trailing `.` to indicate a text block, for example:\n\n```jade\np.\n foo asdf\n asdf\n asdfasdfaf\n asdf\n asd.\n```\n\noutputs:\n\n```html\n

    foo asdf\nasdf\n asdfasdfaf\n asdf\nasd.\n

    \n```\n\nThis however differs from a trailing `.` followed by a space, which although is ignored by the Jade parser, tells Jade that this period is a literal:\n\n```jade\np .\n```\n\noutputs:\n\n```html\n

    .

    \n```\n\nIt should be noted that text blocks should be doubled escaped. For example if you desire the following output.\n\n```html\n

    foo\\bar

    \n```\n\nuse:\n\n```jade\np.\n foo\\\\bar\n```\n\n
    \n### Comments\n\nSingle line comments currently look the same as JavaScript comments,\naka `//` and must be placed on their own line:\n\n```jade\n// just some paragraphs\np foo\np bar\n```\n\nwould output\n\n```html\n\n

    foo

    \n

    bar

    \n```\n\nJade also supports unbuffered comments, by simply adding a hyphen:\n\n```jade\n//- will not output within markup\np foo\np bar\n```\n\noutputting\n\n```html\n

    foo

    \n

    bar

    \n```\n\n
    \n### Block Comments\n\n A block comment is legal as well:\n\n```jade\nbody\n //\n #content\n h1 Example\n```\n\noutputting\n\n```html\n\n \n\n```\n\nJade supports conditional-comments as well, for example:\n\n```jade\nhead\n //if lt IE 8\n script(src='/ie-sucks.js')\n```\n\noutputs:\n\n```html\n\n \n\n```\n\n\n### Nesting\n\n Jade supports nesting to define the tags in a natural way:\n\n```jade\nul\n li.first\n a(href='#') foo\n li\n a(href='#') bar\n li.last\n a(href='#') baz\n```\n\n\n### Block Expansion\n\n Block expansion allows you to create terse single-line nested tags,\n the following example is equivalent to the nesting example above.\n\n```jade\nul\n li.first: a(href='#') foo\n li: a(href='#') bar\n li.last: a(href='#') baz\n```\n\n\n### Case\n\n The case statement takes the following form:\n\n```jade\nhtml\n body\n friends = 10\n case friends\n when 0\n p you have no friends\n when 1\n p you have a friend\n default\n p you have #{friends} friends\n```\n\n Block expansion may also be used:\n\n```jade\nfriends = 5\n\nhtml\n body\n case friends\n when 0: p you have no friends\n when 1: p you have a friend\n default: p you have #{friends} friends\n```\n\n\n### Attributes\n\nJade currently supports `(` and `)` as attribute delimiters.\n\n```jade\na(href='/login', title='View login page') Login\n```\n\nWhen a value is `undefined` or `null` the attribute is _not_ added,\nso this is fine, it will not compile `something=\"null\"`.\n\n```jade\ndiv(something=null)\n```\n\nBoolean attributes are also supported:\n\n```jade\ninput(type=\"checkbox\", checked)\n```\n\nBoolean attributes with code will only output the attribute when `true`:\n\n```jade\ninput(type=\"checkbox\", checked=someValue)\n```\n\nMultiple lines work too:\n\n```jade\ninput(type='checkbox',\n name='agreement',\n checked)\n```\n\nMultiple lines without the comma work fine:\n\n```jade\ninput(type='checkbox'\n name='agreement'\n checked)\n```\n\nFunky whitespace? fine:\n\n```jade\ninput(\n type='checkbox'\n name='agreement'\n checked)\n```\n\nColons work:\n\n```jade\nrss(xmlns:atom=\"atom\")\n```\n\nSuppose we have the `user` local `{ id: 12, name: 'tobi' }`\nand we wish to create an anchor tag with `href` pointing to \"/user/12\"\nwe could use regular javascript concatenation:\n\n```jade\na(href='/user/' + user.id)= user.name\n```\n\nor we could use jade's interpolation, which I added because everyone\nusing Ruby or CoffeeScript seems to think this is legal js..:\n\n```jade\na(href='/user/#{user.id}')= user.name\n```\n\nThe `class` attribute is special-cased when an array is given,\nallowing you to pass an array such as `bodyClasses = ['user', 'authenticated']` directly:\n\n```jade\nbody(class=bodyClasses)\n```\n\n\n### HTML\n\n Inline html is fine, we can use the pipe syntax to\n write arbitrary text, in this case some html:\n\n```jade\nhtml\n body\n |

    Title

    \n |

    foo bar baz

    \n```\n\n Or we can use the trailing `.` to indicate to Jade that we\n only want text in this block, allowing us to omit the pipes:\n\n```jade\nhtml\n body.\n

    Title

    \n

    foo bar baz

    \n```\n\n Both of these examples yield the same result:\n\n```html\n

    Title

    \n

    foo bar baz

    \n\n```\n\n The same rule applies for anywhere you can have text\n in jade, raw html is fine:\n\n```jade\nhtml\n body\n h1 User #{name}\n```\n\n
    \n### Doctypes\n\nTo add a doctype simply use `!!!`, or `doctype` followed by an optional value:\n\n```jade\n!!!\n```\n\nor\n\n```jade\ndoctype\n```\n\nWill output the _html 5_ doctype, however:\n\n```jade\n!!! transitional\n```\n\nWill output the _transitional_ doctype.\n\nDoctypes are case-insensitive, so the following are equivalent:\n\n```jade\ndoctype Basic\ndoctype basic\n```\n\nit's also possible to simply pass a doctype literal:\n\n```jade\ndoctype html PUBLIC \"-//W3C//DTD XHTML Basic 1.1//EN\n```\n\nyielding:\n\n```html\n\n```\n\nBelow are the doctypes defined by default, which can easily be extended:\n\n```js\nvar doctypes = exports.doctypes = {\n '5': '',\n 'default': '',\n 'xml': '',\n 'transitional': '',\n 'strict': '',\n 'frameset': '',\n '1.1': '',\n 'basic': '',\n 'mobile': ''\n};\n```\n\nTo alter the default simply change:\n\n```js\njade.doctypes.default = 'whatever you want';\n```\n\n\n## Filters\n\nFilters are prefixed with `:`, for example `:markdown` and\npass the following block of text to an arbitrary function for processing. View the _features_\nat the top of this document for available filters.\n\n```jade\nbody\n :markdown\n Woah! jade _and_ markdown, very **cool**\n we can even link to [stuff](http://google.com)\n```\n\nRenders:\n\n```html\n

    Woah! jade and markdown, very cool we can even link to stuff

    \n```\n\n\n## Code\n\nJade currently supports three classifications of executable code. The first\nis prefixed by `-`, and is not buffered:\n\n```jade\n- var foo = 'bar';\n```\n\nThis can be used for conditionals, or iteration:\n\n```jade\n- for (var key in obj)\n p= obj[key]\n```\n\nDue to Jade's buffering techniques the following is valid as well:\n\n```jade\n- if (foo)\n ul\n li yay\n li foo\n li worked\n- else\n p oh no! didnt work\n```\n\nHell, even verbose iteration:\n\n```jade\n- if (items.length)\n ul\n - items.forEach(function(item){\n li= item\n - })\n```\n\nAnything you want!\n\nNext up we have _escaped_ buffered code, which is used to\nbuffer a return value, which is prefixed by `=`:\n\n```jade\n- var foo = 'bar'\n= foo\nh1= foo\n```\n\nWhich outputs `bar

    bar

    `. Code buffered by `=` is escaped\nby default for security, however to output unescaped return values\nyou may use `!=`:\n\n```jade\np!= aVarContainingMoreHTML\n```\n\n Jade also has designer-friendly variants, making the literal JavaScript\n more expressive and declarative. For example the following assignments\n are equivalent, and the expression is still regular javascript:\n\n```jade\n- var foo = 'foo ' + 'bar'\nfoo = 'foo ' + 'bar'\n```\n\n Likewise Jade has first-class `if`, `else if`, `else`, `until`, `while`, `unless` among others, however you must remember that the expressions are still regular javascript:\n\n```jade\nif foo == 'bar'\n ul\n li yay\n li foo\n li worked\nelse\n p oh no! didnt work\n```\n\n
    \n## Iteration\n\n Along with vanilla JavaScript Jade also supports a subset of\n constructs that allow you to create more designer-friendly templates,\n one of these constructs is `each`, taking the form:\n\n```jade\neach VAL[, KEY] in OBJ\n```\n\nAn example iterating over an array:\n\n```jade\n- var items = [\"one\", \"two\", \"three\"]\neach item in items\n li= item\n```\n\noutputs:\n\n```html\n
  • one
  • \n
  • two
  • \n
  • three
  • \n```\n\niterating an array with index:\n\n```jade\nitems = [\"one\", \"two\", \"three\"]\neach item, i in items\n li #{item}: #{i}\n```\n\noutputs:\n\n```html\n
  • one: 0
  • \n
  • two: 1
  • \n
  • three: 2
  • \n```\n\niterating an object's keys and values:\n\n```jade\nobj = { foo: 'bar' }\neach val, key in obj\n li #{key}: #{val}\n```\n\nwould output `
  • foo: bar
  • `\n\nInternally Jade converts these statements to regular\nJavaScript loops such as `users.forEach(function(user){`,\nso lexical scope and nesting applies as it would with regular\nJavaScript:\n\n```jade\neach user in users\n each role in user.roles\n li= role\n```\n\n You may also use `for` if you prefer:\n\n```jade\nfor user in users\n for role in user.roles\n li= role\n```\n\n
    \n## Conditionals\n\n Jade conditionals are equivalent to those using the code (`-`) prefix,\n however allow you to ditch parenthesis to become more designer friendly,\n however keep in mind the expression given is _regular_ JavaScript:\n\n```jade\nfor user in users\n if user.role == 'admin'\n p #{user.name} is an admin\n else\n p= user.name\n```\n\n is equivalent to the following using vanilla JavaScript literals:\n\n```jade\nfor user in users\n - if (user.role == 'admin')\n p #{user.name} is an admin\n - else\n p= user.name\n```\n\n Jade also provides `unless` which is equivalent to `if (!(expr))`:\n\n```jade\nfor user in users\n unless user.isAnonymous\n p\n | Click to view\n a(href='/users/' + user.id)= user.name\n```\n\n\n## Template inheritance\n\n Jade supports template inheritance via the `block` and `extends` keywords. A block is simply a \"block\" of Jade that may be replaced within a child template, this process is recursive. To activate template inheritance in Express 2.x you must add: `app.set('view options', { layout: false });`.\n\n Jade blocks can provide default content if desired, however optional as shown below by `block scripts`, `block content`, and `block foot`.\n\n```jade\nhtml\n head\n h1 My Site - #{title}\n block scripts\n script(src='/jquery.js')\n body\n block content\n block foot\n #footer\n p some footer content\n```\n\n Now to extend the layout, simply create a new file and use the `extends` directive as shown below, giving the path (with or without the .jade extension). You may now define one or more blocks that will override the parent block content, note that here the `foot` block is _not_ redefined and will output \"some footer content\".\n\n```jade\nextends layout\n\nblock scripts\n script(src='/jquery.js')\n script(src='/pets.js')\n\nblock content\n h1= title\n each pet in pets\n include pet\n```\n\n It's also possible to override a block to provide additional blocks, as shown in the following example where `content` now exposes a `sidebar` and `primary` block for overriding, or the child template could override `content` all together.\n\n```jade\nextends regular-layout\n\nblock content\n .sidebar\n block sidebar\n p nothing\n .primary\n block primary\n p nothing\n```\n\n\n## Block append / prepend\n\n Jade allows you to _replace_ (default), _prepend_, or _append_ blocks. Suppose for example you have default scripts in a \"head\" block that you wish to utilize on _every_ page, you might do this:\n\n```jade\nhtml\n head\n block head\n script(src='/vendor/jquery.js')\n script(src='/vendor/caustic.js')\n body\n block content\n```\n\n Now suppose you have a page of your application for a JavaScript game, you want some game related scripts as well as these defaults, you can simply `append` the block:\n\n```jade\nextends layout\n\nblock append head\n script(src='/vendor/three.js')\n script(src='/game.js')\n```\n\n When using `block append` or `block prepend` the `block` is optional:\n\n```jade\nextends layout\n\nappend head\n script(src='/vendor/three.js')\n script(src='/game.js')\n```\n\n\n## Includes\n\n Includes allow you to statically include chunks of Jade,\n or other content like css, or html which lives in separate files. The classical example is including a header and footer. Suppose we have the following directory structure:\n\n ./layout.jade\n ./includes/\n ./head.jade\n ./foot.jade\n\nand the following _layout.jade_:\n\n```jade\nhtml\n include includes/head\n body\n h1 My Site\n p Welcome to my super amazing site.\n include includes/foot\n```\n\nboth includes _includes/head_ and _includes/foot_ are\nread relative to the `filename` option given to _layout.jade_,\nwhich should be an absolute path to this file, however Express does this for you. Include then parses these files, and injects the AST produced to render what you would expect:\n\n```html\n\n \n My Site\n \n \n \n

    My Site

    \n

    Welcome to my super lame site.

    \n
    \n

    Copyright>(c) foobar

    \n
    \n \n\n```\n\n As mentioned `include` can be used to include other content\n such as html or css. By providing an extension Jade will not\n assume that the file is Jade source and will include it as\n a literal:\n\n```jade\nhtml\n body\n include content.html\n```\n\n Include directives may also accept a block, in which case the\n the given block will be appended to the _last_ block defined\n in the file. For example if `head.jade` contains:\n\n```jade\nhead\n script(src='/jquery.js')\n```\n\n We may append values by providing a block to `include head`\n as shown below, adding the two scripts.\n\n```jade\nhtml\n include head\n script(src='/foo.js')\n script(src='/bar.js')\n body\n h1 test\n```\n\n You may also `yield` within an included template, allowing you to explicitly mark where the block given to `include` will be placed. Suppose for example you wish to prepend scripts rather than append, you might do the following:\n\n```jade\nhead\n yield\n script(src='/jquery.js')\n script(src='/jquery.ui.js')\n```\n\n Since included Jade is parsed and literally merges the AST, lexically scoped variables function as if the included Jade was written right in the same file. This means `include` may be used as sort of partial, for example suppose we have `user.jade` which utilizes a `user` variable.\n\n```jade\nh1= user.name\np= user.occupation\n```\n\nWe could then simply `include user` while iterating users, and since the `user` variable is already defined within the loop the included template will have access to it.\n\n```jade\nusers = [{ name: 'Tobi', occupation: 'Ferret' }]\n\neach user in users\n .user\n include user\n```\n\nyielding:\n\n```html\n
    \n

    Tobi

    \n

    Ferret

    \n
    \n```\n\nIf we wanted to expose a different variable name as `user` since `user.jade` references that name, we could simply define a new variable as shown here with `user = person`:\n\n```jade\neach person in users\n .user\n user = person\n include user\n```\n\n
    \n## Mixins\n\n Mixins are converted to regular JavaScript functions in\n the compiled template that Jade constructs. Mixins may\n take arguments, though not required:\n\n```jade\nmixin list\n ul\n li foo\n li bar\n li baz\n```\n\n Utilizing a mixin without args looks similar, just without a block:\n\n```jade\nh2 Groceries\nmixin list\n```\n\n Mixins may take one or more arguments as well, the arguments\n are regular javascripts expressions, so for example the following:\n\n```jade\nmixin pets(pets)\n ul.pets\n - each pet in pets\n li= pet\n\nmixin profile(user)\n .user\n h2= user.name\n mixin pets(user.pets)\n```\n\n Would yield something similar to the following html:\n\n```html\n
    \n

    tj

    \n
      \n
    • tobi
    • \n
    • loki
    • \n
    • jane
    • \n
    • manny
    • \n
    \n
    \n```\n\n
    \n## Generated Output\n\n Suppose we have the following Jade:\n\n```jade\n- var title = 'yay'\nh1.title #{title}\np Just an example\n```\n\n When the `compileDebug` option is not explicitly `false`, Jade\n will compile the function instrumented with `__.lineno = n;`, which\n in the event of an exception is passed to `rethrow()` which constructs\n a useful message relative to the initial Jade input.\n\n```js\nfunction anonymous(locals) {\n var __ = { lineno: 1, input: \"- var title = 'yay'\\nh1.title #{title}\\np Just an example\", filename: \"testing/test.js\" };\n var rethrow = jade.rethrow;\n try {\n var attrs = jade.attrs, escape = jade.escape;\n var buf = [];\n with (locals || {}) {\n var interp;\n __.lineno = 1;\n var title = 'yay'\n __.lineno = 2;\n buf.push('');\n buf.push('' + escape((interp = title) == null ? '' : interp) + '');\n buf.push('');\n __.lineno = 3;\n buf.push('

    ');\n buf.push('Just an example');\n buf.push('

    ');\n }\n return buf.join(\"\");\n } catch (err) {\n rethrow(err, __.input, __.filename, __.lineno);\n }\n}\n```\n\nWhen the `compileDebug` option _is_ explicitly `false`, this instrumentation\nis stripped, which is very helpful for light-weight client-side templates. Combining Jade's options with the `./runtime.js` file in this repo allows you\nto toString() compiled templates and avoid running the entire Jade library on\nthe client, increasing performance, and decreasing the amount of JavaScript\nrequired.\n\n```js\nfunction anonymous(locals) {\n var attrs = jade.attrs, escape = jade.escape;\n var buf = [];\n with (locals || {}) {\n var interp;\n var title = 'yay'\n buf.push('');\n buf.push('' + escape((interp = title) == null ? '' : interp) + '');\n buf.push('');\n buf.push('

    ');\n buf.push('Just an example');\n buf.push('

    ');\n }\n return buf.join(\"\");\n}\n```\n\n
    \n## Example Makefile\n\n Below is an example Makefile used to compile _pages/*.jade_\n into _pages/*.html_ files by simply executing `make`.\n\n_Note:_ If you try to run this snippet and `make` throws a `missing separator` error, you should make sure all indented lines use a tab for indentation instead of spaces. (For whatever reason, GitHub renders this code snippet with 4-space indentation although the actual README file uses tabs in this snippet.)\n\n```make\nJADE = $(shell find . -wholename './pages/*.jade')\nHTML = $(JADE:.jade=.html)\n\nall: $(HTML)\n\n%.html: %.jade\n\tjade < $< --path $< > $@\n\nclean:\n\trm -f $(HTML)\n\n.PHONY: clean\n```\n\nthis can be combined with the `watch(1)` command to produce\na watcher-like behaviour:\n\n```bash\n$ watch make\n```\n\n\n## jade(1)\n\n```\n\nUsage: jade [options] [dir|file ...]\n\nOptions:\n\n -h, --help output usage information\n -V, --version output the version number\n -o, --obj javascript options object\n -O, --out output the compiled html to \n -p, --path filename used to resolve includes\n -P, --pretty compile pretty html output\n -c, --client compile for client-side runtime.js\n -D, --no-debug compile without debugging (smaller functions)\n\nExamples:\n\n # translate jade the templates dir\n $ jade templates\n\n # create {foo,bar}.html\n $ jade {foo,bar}.jade\n\n # jade over stdio\n $ jade < my.jade > my.html\n\n # jade over stdio\n $ echo \"h1 Jade!\" | jade\n\n # foo, bar dirs rendering to /tmp\n $ jade foo bar --out /tmp\n\n```\n\n\n## Tutorials\n\n - cssdeck interactive [Jade syntax tutorial](http://cssdeck.com/labs/learning-the-jade-templating-engine-syntax)\n - cssdeck interactive [Jade logic tutorial](http://cssdeck.com/labs/jade-templating-tutorial-codecast-part-2)\n - in [Japanese](http://blog.craftgear.net/4f501e97c1347ec934000001/title/10%E5%88%86%E3%81%A7%E3%82%8F%E3%81%8B%E3%82%8Bjade%E3%83%86%E3%83%B3%E3%83%97%E3%83%AC%E3%83%BC%E3%83%88%E3%82%A8%E3%83%B3%E3%82%B8%E3%83%B3)\n\n\n## License\n\n(The MIT License)\n\nCopyright (c) 2009-2010 TJ Holowaychuk <tj@vision-media.ca>\n\nPermission is hereby granted, free of charge, to any person obtaining\na copy of this software and associated documentation files (the\n'Software'), to deal in the Software without restriction, including\nwithout limitation the rights to use, copy, modify, merge, publish,\ndistribute, sublicense, and/or sell copies of the Software, and to\npermit persons to whom the Software is furnished to do so, subject to\nthe following conditions:\n\nThe above copyright notice and this permission notice shall be\nincluded in all copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,\nEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\nMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.\nIN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY\nCLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,\nTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE\nSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n", - "readmeFilename": "Readme.md", - "bugs": { - "url": "https://github.com/visionmedia/jade/issues" - }, - "_id": "jade@0.28.2", - "_from": "jade@~0.28.2" -} diff --git a/node_modules/jade/runtime.js b/node_modules/jade/runtime.js deleted file mode 100644 index 0f54907..0000000 --- a/node_modules/jade/runtime.js +++ /dev/null @@ -1,179 +0,0 @@ - -jade = (function(exports){ -/*! - * Jade - runtime - * Copyright(c) 2010 TJ Holowaychuk - * MIT Licensed - */ - -/** - * Lame Array.isArray() polyfill for now. - */ - -if (!Array.isArray) { - Array.isArray = function(arr){ - return '[object Array]' == Object.prototype.toString.call(arr); - }; -} - -/** - * Lame Object.keys() polyfill for now. - */ - -if (!Object.keys) { - Object.keys = function(obj){ - var arr = []; - for (var key in obj) { - if (obj.hasOwnProperty(key)) { - arr.push(key); - } - } - return arr; - } -} - -/** - * Merge two attribute objects giving precedence - * to values in object `b`. Classes are special-cased - * allowing for arrays and merging/joining appropriately - * resulting in a string. - * - * @param {Object} a - * @param {Object} b - * @return {Object} a - * @api private - */ - -exports.merge = function merge(a, b) { - var ac = a['class']; - var bc = b['class']; - - if (ac || bc) { - ac = ac || []; - bc = bc || []; - if (!Array.isArray(ac)) ac = [ac]; - if (!Array.isArray(bc)) bc = [bc]; - ac = ac.filter(nulls); - bc = bc.filter(nulls); - a['class'] = ac.concat(bc).join(' '); - } - - for (var key in b) { - if (key != 'class') { - a[key] = b[key]; - } - } - - return a; -}; - -/** - * Filter null `val`s. - * - * @param {Mixed} val - * @return {Mixed} - * @api private - */ - -function nulls(val) { - return val != null; -} - -/** - * Render the given attributes object. - * - * @param {Object} obj - * @param {Object} escaped - * @return {String} - * @api private - */ - -exports.attrs = function attrs(obj, escaped){ - var buf = [] - , terse = obj.terse; - - delete obj.terse; - var keys = Object.keys(obj) - , len = keys.length; - - if (len) { - buf.push(''); - for (var i = 0; i < len; ++i) { - var key = keys[i] - , val = obj[key]; - - if ('boolean' == typeof val || null == val) { - if (val) { - terse - ? buf.push(key) - : buf.push(key + '="' + key + '"'); - } - } else if (0 == key.indexOf('data') && 'string' != typeof val) { - buf.push(key + "='" + JSON.stringify(val) + "'"); - } else if ('class' == key && Array.isArray(val)) { - buf.push(key + '="' + exports.escape(val.join(' ')) + '"'); - } else if (escaped && escaped[key]) { - buf.push(key + '="' + exports.escape(val) + '"'); - } else { - buf.push(key + '="' + val + '"'); - } - } - } - - return buf.join(' '); -}; - -/** - * Escape the given string of `html`. - * - * @param {String} html - * @return {String} - * @api private - */ - -exports.escape = function escape(html){ - return String(html) - .replace(/&(?!(\w+|\#\d+);)/g, '&') - .replace(//g, '>') - .replace(/"/g, '"'); -}; - -/** - * Re-throw the given `err` in context to the - * the jade in `filename` at the given `lineno`. - * - * @param {Error} err - * @param {String} filename - * @param {String} lineno - * @api private - */ - -exports.rethrow = function rethrow(err, filename, lineno){ - if (!filename) throw err; - - var context = 3 - , str = require('fs').readFileSync(filename, 'utf8') - , lines = str.split('\n') - , start = Math.max(lineno - context, 0) - , end = Math.min(lines.length, lineno + context); - - // Error context - var context = lines.slice(start, end).map(function(line, i){ - var curr = i + start + 1; - return (curr == lineno ? ' > ' : ' ') - + curr - + '| ' - + line; - }).join('\n'); - - // Alter exception message - err.path = filename; - err.message = (filename || 'Jade') + ':' + lineno - + '\n' + context + '\n\n' + err.message; - throw err; -}; - - return exports; - -})({}); diff --git a/node_modules/jade/runtime.min.js b/node_modules/jade/runtime.min.js deleted file mode 100644 index 1714efb..0000000 --- a/node_modules/jade/runtime.min.js +++ /dev/null @@ -1 +0,0 @@ -jade=function(exports){Array.isArray||(Array.isArray=function(arr){return"[object Array]"==Object.prototype.toString.call(arr)}),Object.keys||(Object.keys=function(obj){var arr=[];for(var key in obj)obj.hasOwnProperty(key)&&arr.push(key);return arr}),exports.merge=function merge(a,b){var ac=a["class"],bc=b["class"];if(ac||bc)ac=ac||[],bc=bc||[],Array.isArray(ac)||(ac=[ac]),Array.isArray(bc)||(bc=[bc]),ac=ac.filter(nulls),bc=bc.filter(nulls),a["class"]=ac.concat(bc).join(" ");for(var key in b)key!="class"&&(a[key]=b[key]);return a};function nulls(val){return val!=null}return exports.attrs=function attrs(obj,escaped){var buf=[],terse=obj.terse;delete obj.terse;var keys=Object.keys(obj),len=keys.length;if(len){buf.push("");for(var i=0;i/g,">").replace(/"/g,""")},exports.rethrow=function rethrow(err,filename,lineno){if(!filename)throw err;var context=3,str=require("fs").readFileSync(filename,"utf8"),lines=str.split("\n"),start=Math.max(lineno-context,0),end=Math.min(lines.length,lineno+context),context=lines.slice(start,end).map(function(line,i){var curr=i+start+1;return(curr==lineno?" > ":" ")+curr+"| "+line}).join("\n");throw err.path=filename,err.message=(filename||"Jade")+":"+lineno+"\n"+context+"\n\n"+err.message,err},exports}({}); \ No newline at end of file diff --git a/node_modules/jade/testing/index.html b/node_modules/jade/testing/index.html deleted file mode 100644 index 105c9f4..0000000 --- a/node_modules/jade/testing/index.html +++ /dev/null @@ -1,5 +0,0 @@ - \ No newline at end of file diff --git a/node_modules/jade/testing/index.jade b/node_modules/jade/testing/index.jade deleted file mode 100644 index a22e82a..0000000 --- a/node_modules/jade/testing/index.jade +++ /dev/null @@ -1,4 +0,0 @@ - -html - body - include some.js diff --git a/node_modules/jade/testing/layout.html b/node_modules/jade/testing/layout.html deleted file mode 100644 index cbbfc7a..0000000 --- a/node_modules/jade/testing/layout.html +++ /dev/null @@ -1 +0,0 @@ -Application \ No newline at end of file diff --git a/node_modules/jade/testing/layout.jade b/node_modules/jade/testing/layout.jade deleted file mode 100644 index a48765c..0000000 --- a/node_modules/jade/testing/layout.jade +++ /dev/null @@ -1,6 +0,0 @@ -!!! 5 -html - head - title Application - body - block content diff --git a/node_modules/jade/testing/mobile.html b/node_modules/jade/testing/mobile.html deleted file mode 100644 index e69de29..0000000 diff --git a/node_modules/jade/testing/mobile.jade b/node_modules/jade/testing/mobile.jade deleted file mode 100644 index e69de29..0000000 diff --git a/node_modules/jade/testing/nested/something.html b/node_modules/jade/testing/nested/something.html deleted file mode 100644 index d6b0493..0000000 --- a/node_modules/jade/testing/nested/something.html +++ /dev/null @@ -1 +0,0 @@ -

    out

    \ No newline at end of file diff --git a/node_modules/jade/testing/nested/something.jade b/node_modules/jade/testing/nested/something.jade deleted file mode 100644 index 552c797..0000000 --- a/node_modules/jade/testing/nested/something.jade +++ /dev/null @@ -1 +0,0 @@ -p out \ No newline at end of file diff --git a/node_modules/jade/testing/some.js b/node_modules/jade/testing/some.js deleted file mode 100644 index 3104e71..0000000 --- a/node_modules/jade/testing/some.js +++ /dev/null @@ -1,4 +0,0 @@ - -if (something) { - something('hey'); -} diff --git a/node_modules/jade/testing/test.md b/node_modules/jade/testing/test.md deleted file mode 100644 index 9af1bb2..0000000 --- a/node_modules/jade/testing/test.md +++ /dev/null @@ -1,5 +0,0 @@ -Just a _test_ of some **markdown**: - - - foo - - bar - - baz diff --git a/node_modules/lazy/.npmignore b/node_modules/lazy/.npmignore deleted file mode 100644 index 1377554..0000000 --- a/node_modules/lazy/.npmignore +++ /dev/null @@ -1 +0,0 @@ -*.swp diff --git a/node_modules/lazy/README.md b/node_modules/lazy/README.md deleted file mode 100644 index f2d853f..0000000 --- a/node_modules/lazy/README.md +++ /dev/null @@ -1,185 +0,0 @@ -Lazy lists for node -=================== - - -# Table of contents: - -[Introduction](#Introduction) - -[Documentation](#Documentation) - -
    -# Introduction -Lazy comes really handy when you need to treat a stream of events like a list. -The best use case currently is returning a lazy list from an asynchronous -function, and having data pumped into it via events. In asynchronous -programming you can't just return a regular list because you don't yet have -data for it. The usual solution so far has been to provide a callback that gets -called when the data is available. But doing it this way you lose the power of -chaining functions and creating pipes, which leads to not that nice interfaces. -(See the 2nd example below to see how it improved the interface in one of my -modules.) - -Check out this toy example, first you create a Lazy object: -```javascript - var Lazy = require('lazy'); - - var lazy = new Lazy; - lazy - .filter(function (item) { - return item % 2 == 0 - }) - .take(5) - .map(function (item) { - return item*2; - }) - .join(function (xs) { - console.log(xs); - }); -``` - -This code says that 'lazy' is going to be a lazy list that filters even -numbers, takes first five of them, then multiplies all of them by 2, and then -calls the join function (think of join as in threads) on the final list. - -And now you can emit 'data' events with data in them at some point later, -```javascript - [0,1,2,3,4,5,6,7,8,9,10].forEach(function (x) { - lazy.emit('data', x); - }); -``` - -The output will be produced by the 'join' function, which will output the -expected [0, 4, 8, 12, 16]. - -And here is a real-world example. Some time ago I wrote a hash database for -node.js called node-supermarket (think of key-value store except greater). Now -it had a similar interface as a list, you could .forEach on the stored -elements, .filter them, etc. But being asynchronous in nature it lead to the -following code, littered with callbacks and temporary lists: -```javascript - var Store = require('supermarket'); - - var db = new Store({ filename : 'users.db', json : true }); - - var users_over_20 = []; - db.filter( - function (user, meta) { - // predicate function - return meta.age > 20; - }, - function (err, user, meta) { - // function that gets executed when predicate is true - if (users_over_20.length < 5) - users_over_20.push(meta); - }, - function () { - // done function, called when all records have been filtered - - // now do something with users_over_20 - } - ) -``` -This code selects first five users who are over 20 years old and stores them -in users_over_20. - -But now we changed the node-supermarket interface to return lazy lists, and -the code became: -```javascript - var Store = require('supermarket'); - - var db = new Store({ filename : 'users.db', json : true }); - - db.filter(function (user, meta) { - return meta.age > 20; - }) - .take(5) - .join(function (xs) { - // xs contains the first 5 users who are over 20! - }); -``` -This is so much nicer! - -Here is the latest feature: .lines. Given a stream of data that has \n's in it, -.lines converts that into a list of lines. - -Here is an example from node-iptables that I wrote the other week, -```javascript - var Lazy = require('lazy'); - var spawn = require('child_process').spawn; - var iptables = spawn('iptables', ['-L', '-n', '-v']); - - Lazy(iptables.stdout) - .lines - .map(String) - .skip(2) // skips the two lines that are iptables header - .map(function (line) { - // packets, bytes, target, pro, opt, in, out, src, dst, opts - var fields = line.trim().split(/\s+/, 9); - return { - parsed : { - packets : fields[0], - bytes : fields[1], - target : fields[2], - protocol : fields[3], - opt : fields[4], - in : fields[5], - out : fields[6], - src : fields[7], - dst : fields[8] - }, - raw : line.trim() - }; - }); -``` -This example takes the `iptables -L -n -v` command and uses .lines on its output. -Then it .skip's two lines from input and maps a function on all other lines that -creates a data structure from the output. - - -# Documentation - -Supports the following operations: - -* lazy.filter(f) -* lazy.forEach(f) -* lazy.map(f) -* lazy.take(n) -* lazy.takeWhile(f) -* lazy.bucket(init, f) -* lazy.lines -* lazy.sum(f) -* lazy.product(f) -* lazy.foldr(op, i, f) -* lazy.skip(n) -* lazy.head(f) -* lazy.tail(f) -* lazy.join(f) - -The Lazy object itself has a .range property for generating all the possible ranges. - -Here are several examples: - -* Lazy.range('10..') - infinite range starting from 10 -* Lazy.range('(10..') - infinite range starting from 11 -* Lazy.range(10) - range from 0 to 9 -* Lazy.range(-10, 10) - range from -10 to 9 (-10, -9, ... 0, 1, ... 9) -* Lazy.range(-10, 10, 2) - range from -10 to 8, skipping every 2nd element (-10, -8, ... 0, 2, 4, 6, 8) -* Lazy.range(10, 0, 2) - reverse range from 10 to 1, skipping every 2nd element (10, 8, 6, 4, 2) -* Lazy.range(10, 0) - reverse range from 10 to 1 -* Lazy.range('5..50') - range from 5 to 49 -* Lazy.range('50..44') - range from 50 to 45 -* Lazy.range('1,1.1..4') - range from 1 to 4 with increment of 0.1 (1, 1.1, 1.2, ... 3.9) -* Lazy.range('4,3.9..1') - reverse range from 4 to 1 with decerement of 0.1 -* Lazy.range('[1..10]') - range from 1 to 10 (all inclusive) -* Lazy.range('[10..1]') - range from 10 to 1 (all inclusive) -* Lazy.range('[1..10)') - range grom 1 to 9 -* Lazy.range('[10..1)') - range from 10 to 2 -* Lazy.range('(1..10]') - range from 2 to 10 -* Lazy.range('(10..1]') - range from 9 to 1 -* Lazy.range('(1..10)') - range from 2 to 9 -* Lazy.range('[5,10..50]') - range from 5 to 50 with a step of 5 (all inclusive) - -Then you can use other lazy functions on these ranges. - - diff --git a/node_modules/lazy/lazy.js b/node_modules/lazy/lazy.js deleted file mode 100644 index 08a01e7..0000000 --- a/node_modules/lazy/lazy.js +++ /dev/null @@ -1,349 +0,0 @@ -var EventEmitter = require('events').EventEmitter; -var util = require('util'); -var stream = require('stream'); - -function Lazy(em, opts) { - if (!(this instanceof Lazy)) return new Lazy(em, opts); - EventEmitter.call(this); - var self = this; - - - self.once = function (name, f) { - self.on(name, function g () { - self.removeListener(name, g); - f.apply(this, arguments); - }); - } - - if (!opts) opts = {}; - var dataName = opts.data || 'data'; - var pipeName = opts.pipe || 'pipe'; - var endName = opts.pipe || 'end'; - - if (pipeName != endName) { - var piped = false; - self.once(pipeName, function () { piped = true }); - self.once(endName, function () { - if (!piped) self.emit(pipeName); - }); - } - - self.push = function (x) { - self.emit(dataName, x); - } - - self.end = function () { - self.emit(endName); - } - - if (em && em.on) { - em.on(endName, function () { - self.emit(endName); - }); - self.on(pipeName, function () { - em.emit(pipeName); - }); - // Check for v0.10 or Greater (Stream2 has Duplex type) - if (stream.Duplex && em instanceof(stream)) { - em.on('readable', function () { - var x = em.read(); - self.emit(dataName, x); - }); - } else { - // Old Stream1 or Event support - em.on(dataName, function (x) { - self.emit(dataName, x); - }); - } - } - - function newLazy (g, h, l) { - if (!g) { - g = function () { - return true; - }; - } - if (!h) { - h = function (x) { - return x; - }; - } - var lazy = new Lazy(null, opts, l); - self.on(dataName, function (x, y) { - if (g.call(lazy, x)) { - lazy.emit(dataName, h(x), y); - } - }); - self.once(pipeName, function () { - lazy.emit(pipeName); - }); - return lazy; - } - - self.filter = function (f) { - return newLazy(function (x) { - return f(x); - }); - } - - self.forEach = function (f) { - return newLazy(function (x) { - f(x); - return true; - }); - } - - self.map = function (f) { - return newLazy( - function () { return true }, - function (x) { return f(x) } - ); - } - - self.head = function (f) { - var lazy = newLazy(); - lazy.on(dataName, function g (x) { - f(x) - lazy.removeListener(dataName, g) - }) - } - - self.tail = function () { - var skip = true; - return newLazy(function () { - if (skip) { - skip = false; - return false; - } - return true; - }); - } - - self.skip = function (n) { - return newLazy(function () { - if (n > 0) { - n--; - return false; - } - return true; - }); - } - - self.take = function (n) { - return newLazy(function () { - if (n == 0) self.emit(pipeName); - return n-- > 0; - }); - } - - self.takeWhile = function (f) { - var cond = true; - return newLazy(function (x) { - if (cond && f(x)) return true; - cond = false; - self.emit(pipeName); - return false; - }); - } - - self.foldr = function (op, i, f) { - var acc = i; - var lazy = newLazy(); - lazy.on(dataName, function g (x) { - acc = op(x, acc); - }); - lazy.once(pipeName, function () { - f(acc); - }); - } - - self.sum = function (f) { - return self.foldr(function (x, acc) { return x + acc }, 0, f); - } - - self.product = function (f) { - return self.foldr(function (x, acc) { return x*acc }, 1, f); - } - - self.join = function (f) { - var data = [] - var lazy = newLazy(function (x) { - data.push(x); - return true; - }); - lazy.once(pipeName, function () { f(data) }); - return self; - } - - self.bucket = function (init, f) { - var lazy = new Lazy(null, opts); - var yieldTo = function (x) { - lazy.emit(dataName, x); - }; - - var acc = init; - - self.on(dataName, function (x) { - acc = f.call(yieldTo, acc, x); - }); - - self.once(pipeName, function () { - lazy.emit(pipeName); - }); - - // flush on end event - self.once(endName, function () { - var finalBuffer = mergeBuffers(acc); - if (finalBuffer) { - yieldTo(finalBuffer); - } - }); - - return lazy; - } - - // Streams that use this should emit strings or buffers only - self.__defineGetter__('lines', function () { - return self.bucket([], function (chunkArray, chunk) { - var newline = '\n'.charCodeAt(0), lastNewLineIndex = 0; - if (typeof chunk === 'string') chunk = new Buffer(chunk); - if (chunk){ - for (var i = 0; i < chunk.length; i++) { - if (chunk[i] === newline) { - // If we have content from the current chunk to append to our buffers, do it. - if (i > 0) { - chunkArray.push(chunk.slice(lastNewLineIndex, i)); - } - - // Wrap all our buffers and emit it. - this(mergeBuffers(chunkArray)); - lastNewLineIndex = i + 1; - } - } - } - - if (lastNewLineIndex > 0) { - // New line found in the chunk, push the remaining part of the buffer. - if (lastNewLineIndex < chunk.length) { - chunkArray.push(chunk.slice(lastNewLineIndex)); - } - } else { - // No new line found, push the whole buffer. - if (chunk && chunk.length) { - chunkArray.push(chunk); - } - } - return chunkArray; - }); - }); -} - -Lazy.range = function () { - var args = arguments; - var step = 1; - var infinite = false; - - if (args.length == 1 && typeof args[0] == 'number') { - var i = 0, j = args[0]; - } - else if (args.length == 1 && typeof args[0] == 'string') { // 'start[,next]..[end]' - var arg = args[0]; - var startOpen = false, endClosed = false; - if (arg[0] == '(' || arg[0] == '[') { - if (arg[0] == '(') startOpen = true; - arg = arg.slice(1); - } - if (arg.slice(-1) == ']') endClosed = true; - - var parts = arg.split('..'); - if (parts.length != 2) - throw new Error("single argument range takes 'start..' or 'start..end' or 'start,next..end'"); - - if (parts[1] == '') { // 'start..' - var i = parts[0]; - infinite = true; - } - else { // 'start[,next]..end' - var progression = parts[0].split(','); - if (progression.length == 1) { // start..end - var i = parts[0], j = parts[1]; - } - else { // 'start,next..end' - var i = progression[0], j = parts[1]; - step = Math.abs(progression[1]-i); - } - } - - i = parseInt(i, 10); - j = parseInt(j, 10); - - if (startOpen) { - if (infinite || i < j) i++; - else i--; - } - - if (endClosed) { - if (i < j) j++; - else j--; - } - } - else if (args.length == 2 || args.length == 3) { // start, end[, step] - var i = args[0], j = args[1]; - if (args.length == 3) { - var step = args[2]; - } - } - else { - throw new Error("range takes 1, 2 or 3 arguments"); - } - var lazy = new Lazy; - var stopInfinite = false; - lazy.on('pipe', function () { - stopInfinite = true; - }); - if (infinite) { - process.nextTick(function g () { - if (stopInfinite) return; - lazy.emit('data', i++); - process.nextTick(g); - }); - } - else { - process.nextTick(function () { - if (i < j) { - for (; ij; i-=step) { - lazy.emit('data', i) - } - } - lazy.emit('end'); - }); - } - return lazy; -} - -var mergeBuffers = function mergeBuffers(buffers) { - // We expect buffers to be a non-empty Array - if (!buffers || !Array.isArray(buffers) || !buffers.length) return; - - var finalBufferLength, finalBuffer, currentBuffer, currentSize = 0; - - // Sum all the buffers lengths - finalBufferLength = buffers.reduce(function(left, right) { return (left.length||left) + (right.length||right); }, 0); - finalBuffer = new Buffer(finalBufferLength); - while(buffers.length) { - currentBuffer = buffers.shift(); - currentBuffer.copy(finalBuffer, currentSize); - currentSize += currentBuffer.length; - } - - return finalBuffer; -} - - -util.inherits(Lazy, EventEmitter); -module.exports = Lazy; diff --git a/node_modules/lazy/lazy.js~ b/node_modules/lazy/lazy.js~ deleted file mode 100644 index fbdf2a4..0000000 --- a/node_modules/lazy/lazy.js~ +++ /dev/null @@ -1,348 +0,0 @@ -var EventEmitter = require('events').EventEmitter; -var util = require('util'); -var stream = require('stream'); - -function Lazy(em, opts) { - if (!(this instanceof Lazy)) return new Lazy(em, opts); - EventEmitter.call(this); - var self = this; - - - self.once = function (name, f) { - self.on(name, function g () { - self.removeListener(name, g); - f.apply(this, arguments); - }); - } - - if (!opts) opts = {}; - var dataName = opts.data || 'data'; - var pipeName = opts.pipe || 'pipe'; - var endName = opts.pipe || 'end'; - - if (pipeName != endName) { - var piped = false; - self.once(pipeName, function () { piped = true }); - self.once(endName, function () { - if (!piped) self.emit(pipeName); - }); - } - - self.push = function (x) { - self.emit(dataName, x); - } - - self.end = function () { - self.emit(endName); - } - - if (em && em.on) { - em.on(endName, function () { - self.emit(endName); - }); - self.on(pipeName, function () { - em.emit(pipeName); - }); - // Check for v0.10 or Greater (Stream2 has Duplex type) - if (stream.Duplex && em instanceof(stream)) { - em.on('readable', function () { - var x = em.read(); - self.emit(dataName, x); - }); - } else { - // Old Stream1 or Event support - em.on(dataName, function (x) { - self.emit(dataName, x); - }); - } - } - - function newLazy (g, h, l) { - if (!g) { - g = function () { - return true; - }; - } - if (!h) { - h = function (x) { - return x; - }; - } - var lazy = new Lazy(null, opts, l); - self.on(dataName, function (x, y) { - if (g.call(lazy, x)) { - lazy.emit(dataName, h(x), y); - } - }); - self.once(pipeName, function () { - lazy.emit(pipeName); - }); - return lazy; - } - - self.filter = function (f) { - return newLazy(function (x) { - return f(x); - }); - } - - self.forEach = function (f) { - return newLazy(function (x) { - f(x); - return true; - }); - } - - self.map = function (f) { - return newLazy( - function () { return true }, - function (x) { return f(x) } - ); - } - - self.head = function (f) { - var lazy = newLazy(); - lazy.on(dataName, function g (x) { - f(x) - lazy.removeListener(dataName, g) - }) - } - - self.tail = function () { - var skip = true; - return newLazy(function () { - if (skip) { - skip = false; - return false; - } - return true; - }); - } - - self.skip = function (n) { - return newLazy(function () { - if (n > 0) { - n--; - return false; - } - return true; - }); - } - - self.take = function (n) { - return newLazy(function () { - if (n == 0) self.emit(pipeName); - return n-- > 0; - }); - } - - self.takeWhile = function (f) { - var cond = true; - return newLazy(function (x) { - if (cond && f(x)) return true; - cond = false; - self.emit(pipeName); - return false; - }); - } - - self.foldr = function (op, i, f) { - var acc = i; - var lazy = newLazy(); - lazy.on(dataName, function g (x) { - acc = op(x, acc); - }); - lazy.once(pipeName, function () { - f(acc); - }); - } - - self.sum = function (f) { - return self.foldr(function (x, acc) { return x + acc }, 0, f); - } - - self.product = function (f) { - return self.foldr(function (x, acc) { return x*acc }, 1, f); - } - - self.join = function (f) { - var data = [] - var lazy = newLazy(function (x) { - data.push(x); - return true; - }); - lazy.once(pipeName, function () { f(data) }); - return self; - } - - self.bucket = function (init, f) { - var lazy = new Lazy(null, opts); - var yieldTo = function (x) { - lazy.emit(dataName, x); - }; - - var acc = init; - - self.on(dataName, function (x) { - acc = f.call(yieldTo, acc, x); - }); - - self.once(pipeName, function () { - lazy.emit(pipeName); - }); - - // flush on end event - self.once(endName, function () { - var finalBuffer = mergeBuffers(acc); - if (finalBuffer) { - yieldTo(finalBuffer); - } - }); - - return lazy; - } - - // Streams that use this should emit strings or buffers only - self.__defineGetter__('lines', function () { - return self.bucket([], function (chunkArray, chunk) { - var newline = '\n'.charCodeAt(0), lastNewLineIndex = 0; - if (typeof chunk === 'string') chunk = new Buffer(chunk); - - for (var i = 0; i < chunk.length; i++) { - if (chunk[i] === newline) { - // If we have content from the current chunk to append to our buffers, do it. - if (i > 0) { - chunkArray.push(chunk.slice(lastNewLineIndex, i)); - } - - // Wrap all our buffers and emit it. - this(mergeBuffers(chunkArray)); - lastNewLineIndex = i + 1; - } - } - - if (lastNewLineIndex > 0) { - // New line found in the chunk, push the remaining part of the buffer. - if (lastNewLineIndex < chunk.length) { - chunkArray.push(chunk.slice(lastNewLineIndex)); - } - } else { - // No new line found, push the whole buffer. - if (chunk.length) { - chunkArray.push(chunk); - } - } - return chunkArray; - }); - }); -} - -Lazy.range = function () { - var args = arguments; - var step = 1; - var infinite = false; - - if (args.length == 1 && typeof args[0] == 'number') { - var i = 0, j = args[0]; - } - else if (args.length == 1 && typeof args[0] == 'string') { // 'start[,next]..[end]' - var arg = args[0]; - var startOpen = false, endClosed = false; - if (arg[0] == '(' || arg[0] == '[') { - if (arg[0] == '(') startOpen = true; - arg = arg.slice(1); - } - if (arg.slice(-1) == ']') endClosed = true; - - var parts = arg.split('..'); - if (parts.length != 2) - throw new Error("single argument range takes 'start..' or 'start..end' or 'start,next..end'"); - - if (parts[1] == '') { // 'start..' - var i = parts[0]; - infinite = true; - } - else { // 'start[,next]..end' - var progression = parts[0].split(','); - if (progression.length == 1) { // start..end - var i = parts[0], j = parts[1]; - } - else { // 'start,next..end' - var i = progression[0], j = parts[1]; - step = Math.abs(progression[1]-i); - } - } - - i = parseInt(i, 10); - j = parseInt(j, 10); - - if (startOpen) { - if (infinite || i < j) i++; - else i--; - } - - if (endClosed) { - if (i < j) j++; - else j--; - } - } - else if (args.length == 2 || args.length == 3) { // start, end[, step] - var i = args[0], j = args[1]; - if (args.length == 3) { - var step = args[2]; - } - } - else { - throw new Error("range takes 1, 2 or 3 arguments"); - } - var lazy = new Lazy; - var stopInfinite = false; - lazy.on('pipe', function () { - stopInfinite = true; - }); - if (infinite) { - process.nextTick(function g () { - if (stopInfinite) return; - lazy.emit('data', i++); - process.nextTick(g); - }); - } - else { - process.nextTick(function () { - if (i < j) { - for (; ij; i-=step) { - lazy.emit('data', i) - } - } - lazy.emit('end'); - }); - } - return lazy; -} - -var mergeBuffers = function mergeBuffers(buffers) { - // We expect buffers to be a non-empty Array - if (!buffers || !Array.isArray(buffers) || !buffers.length) return; - - var finalBufferLength, finalBuffer, currentBuffer, currentSize = 0; - - // Sum all the buffers lengths - finalBufferLength = buffers.reduce(function(left, right) { return (left.length||left) + (right.length||right); }, 0); - finalBuffer = new Buffer(finalBufferLength); - while(buffers.length) { - currentBuffer = buffers.shift(); - currentBuffer.copy(finalBuffer, currentSize); - currentSize += currentBuffer.length; - } - - return finalBuffer; -} - - -util.inherits(Lazy, EventEmitter); -module.exports = Lazy; diff --git a/node_modules/lazy/package.json b/node_modules/lazy/package.json deleted file mode 100644 index 5686143..0000000 --- a/node_modules/lazy/package.json +++ /dev/null @@ -1,38 +0,0 @@ -{ - "name": "lazy", - "version": "1.0.11", - "description": "Lazy lists for node", - "main": "./lazy.js", - "keywords": [ - "lazy lists", - "functional" - ], - "author": { - "name": "Peteris Krumins", - "email": "peteris.krumins@gmail.com", - "url": "http://www.catonmat.net" - }, - "license": { - "type": "MIT" - }, - "repository": { - "type": "git", - "url": "http://github.com/pkrumins/node-lazy.git" - }, - "devDependencies": { - "expresso": ">=0.7.5" - }, - "engines": { - "node": ">=0.2.0" - }, - "scripts": { - "test": "expresso" - }, - "readme": "Lazy lists for node\n===================\n\n\n# Table of contents:\n\n[Introduction](#Introduction)\n \n[Documentation](#Documentation)\n\n\n# Introduction\nLazy comes really handy when you need to treat a stream of events like a list.\nThe best use case currently is returning a lazy list from an asynchronous\nfunction, and having data pumped into it via events. In asynchronous\nprogramming you can't just return a regular list because you don't yet have\ndata for it. The usual solution so far has been to provide a callback that gets\ncalled when the data is available. But doing it this way you lose the power of\nchaining functions and creating pipes, which leads to not that nice interfaces.\n(See the 2nd example below to see how it improved the interface in one of my\nmodules.)\n\nCheck out this toy example, first you create a Lazy object:\n```javascript\n var Lazy = require('lazy');\n\n var lazy = new Lazy;\n lazy\n .filter(function (item) {\n return item % 2 == 0\n })\n .take(5)\n .map(function (item) {\n return item*2;\n })\n .join(function (xs) {\n console.log(xs);\n });\n```\n\nThis code says that 'lazy' is going to be a lazy list that filters even\nnumbers, takes first five of them, then multiplies all of them by 2, and then\ncalls the join function (think of join as in threads) on the final list.\n\nAnd now you can emit 'data' events with data in them at some point later,\n```javascript\n [0,1,2,3,4,5,6,7,8,9,10].forEach(function (x) {\n lazy.emit('data', x);\n });\n```\n\nThe output will be produced by the 'join' function, which will output the\nexpected [0, 4, 8, 12, 16].\n\nAnd here is a real-world example. Some time ago I wrote a hash database for\nnode.js called node-supermarket (think of key-value store except greater). Now\nit had a similar interface as a list, you could .forEach on the stored\nelements, .filter them, etc. But being asynchronous in nature it lead to the\nfollowing code, littered with callbacks and temporary lists:\n```javascript\n var Store = require('supermarket');\n\n var db = new Store({ filename : 'users.db', json : true });\n\n var users_over_20 = [];\n db.filter(\n function (user, meta) {\n // predicate function\n return meta.age > 20;\n },\n function (err, user, meta) {\n // function that gets executed when predicate is true\n if (users_over_20.length < 5)\n users_over_20.push(meta);\n },\n function () {\n // done function, called when all records have been filtered\n\n // now do something with users_over_20\n }\n )\n```\nThis code selects first five users who are over 20 years old and stores them\nin users_over_20.\n\nBut now we changed the node-supermarket interface to return lazy lists, and\nthe code became:\n```javascript\n var Store = require('supermarket');\n\n var db = new Store({ filename : 'users.db', json : true });\n\n db.filter(function (user, meta) {\n return meta.age > 20;\n })\n .take(5)\n .join(function (xs) {\n // xs contains the first 5 users who are over 20!\n });\n```\nThis is so much nicer!\n\nHere is the latest feature: .lines. Given a stream of data that has \\n's in it,\n.lines converts that into a list of lines.\n\nHere is an example from node-iptables that I wrote the other week,\n```javascript\n var Lazy = require('lazy');\n var spawn = require('child_process').spawn;\n var iptables = spawn('iptables', ['-L', '-n', '-v']);\n\n Lazy(iptables.stdout)\n .lines\n .map(String)\n .skip(2) // skips the two lines that are iptables header\n .map(function (line) {\n // packets, bytes, target, pro, opt, in, out, src, dst, opts\n var fields = line.trim().split(/\\s+/, 9);\n return {\n parsed : {\n packets : fields[0],\n bytes : fields[1],\n target : fields[2],\n protocol : fields[3],\n opt : fields[4],\n in : fields[5],\n out : fields[6],\n src : fields[7],\n dst : fields[8]\n },\n raw : line.trim()\n };\n });\n```\nThis example takes the `iptables -L -n -v` command and uses .lines on its output.\nThen it .skip's two lines from input and maps a function on all other lines that\ncreates a data structure from the output.\n\n\n# Documentation\n\nSupports the following operations:\n\n* lazy.filter(f)\n* lazy.forEach(f)\n* lazy.map(f)\n* lazy.take(n)\n* lazy.takeWhile(f)\n* lazy.bucket(init, f)\n* lazy.lines\n* lazy.sum(f)\n* lazy.product(f)\n* lazy.foldr(op, i, f)\n* lazy.skip(n)\n* lazy.head(f)\n* lazy.tail(f)\n* lazy.join(f)\n\nThe Lazy object itself has a .range property for generating all the possible ranges.\n\nHere are several examples:\n\n* Lazy.range('10..') - infinite range starting from 10\n* Lazy.range('(10..') - infinite range starting from 11\n* Lazy.range(10) - range from 0 to 9\n* Lazy.range(-10, 10) - range from -10 to 9 (-10, -9, ... 0, 1, ... 9)\n* Lazy.range(-10, 10, 2) - range from -10 to 8, skipping every 2nd element (-10, -8, ... 0, 2, 4, 6, 8)\n* Lazy.range(10, 0, 2) - reverse range from 10 to 1, skipping every 2nd element (10, 8, 6, 4, 2)\n* Lazy.range(10, 0) - reverse range from 10 to 1\n* Lazy.range('5..50') - range from 5 to 49\n* Lazy.range('50..44') - range from 50 to 45\n* Lazy.range('1,1.1..4') - range from 1 to 4 with increment of 0.1 (1, 1.1, 1.2, ... 3.9)\n* Lazy.range('4,3.9..1') - reverse range from 4 to 1 with decerement of 0.1\n* Lazy.range('[1..10]') - range from 1 to 10 (all inclusive)\n* Lazy.range('[10..1]') - range from 10 to 1 (all inclusive)\n* Lazy.range('[1..10)') - range grom 1 to 9\n* Lazy.range('[10..1)') - range from 10 to 2\n* Lazy.range('(1..10]') - range from 2 to 10\n* Lazy.range('(10..1]') - range from 9 to 1\n* Lazy.range('(1..10)') - range from 2 to 9\n* Lazy.range('[5,10..50]') - range from 5 to 50 with a step of 5 (all inclusive)\n\nThen you can use other lazy functions on these ranges.\n\n\n", - "readmeFilename": "README.md", - "bugs": { - "url": "https://github.com/pkrumins/node-lazy/issues" - }, - "_id": "lazy@1.0.11", - "_from": "lazy@~1.0.9" -} diff --git a/node_modules/lazy/package.json~ b/node_modules/lazy/package.json~ deleted file mode 100644 index d1762a1..0000000 --- a/node_modules/lazy/package.json~ +++ /dev/null @@ -1,32 +0,0 @@ -{ - "name": "lazy", - "version": "1.0.10", - "description": "Lazy lists for node", - "main": "./lazy.js", - "keywords": [ - "lazy lists", "functional" - ], - "author": { - "name": "Peteris Krumins", - "email": "peteris.krumins@gmail.com", - "web": "http://www.catonmat.net", - "twitter": "pkrumins" - }, - "license": { - "type": "MIT" - }, - "repository": { - "type": "git", - "url": "http://github.com/pkrumins/node-lazy.git" - }, - "devDependencies" : { - "expresso" : ">=0.7.5" - }, - "engines": { - "node": ">=0.2.0" - }, - "scripts": { - "test": "expresso" - } -} - diff --git a/node_modules/lazy/test/bucket.js b/node_modules/lazy/test/bucket.js deleted file mode 100644 index b1efdac..0000000 --- a/node_modules/lazy/test/bucket.js +++ /dev/null @@ -1,37 +0,0 @@ -var assert = require('assert'); -var Lazy = require('..'); - -exports.bucket = function () { - var joined = false; - var lazy = new Lazy; - lazy.bucket('', function splitter (acc, x) { - var accx = acc + x; - var i = accx.indexOf('\n'); - if (i >= 0) { - this(accx.slice(0, i)); - return splitter.call(this, accx.slice(i + 1), ''); - } - return accx; - }).join(function (lines) { - assert.deepEqual(lines, 'foo bar baz quux moo'.split(' ')); - joined = true; - }); - - setTimeout(function () { - lazy.emit('data', 'foo\nba'); - }, 50); - setTimeout(function () { - lazy.emit('data', 'r'); - }, 100); - setTimeout(function () { - lazy.emit('data', '\nbaz\nquux\nm'); - }, 150); - setTimeout(function () { - lazy.emit('data', 'oo'); - }, 200); - setTimeout(function () { - lazy.emit('data', '\nzoom'); - lazy.emit('end'); - assert.ok(joined); - }, 250); -}; diff --git a/node_modules/lazy/test/complex.js b/node_modules/lazy/test/complex.js deleted file mode 100644 index f79d619..0000000 --- a/node_modules/lazy/test/complex.js +++ /dev/null @@ -1,52 +0,0 @@ -var assert = require('assert'); -var Lazy = require('..'); -var expresso = expresso; - -function range(i, j) { - var r = []; - for (;i 5 }).join(function (xs) { - assert.deepEqual(xs, [6,7,8,9]); - executed = true; - }); - - range(0,10).forEach(function (x) { - lazy.emit('data', x); - }); - lazy.emit('pipe'); - - assert.ok(executed, 'join didn\'t execute'); -} - diff --git a/node_modules/lazy/test/foldr.js b/node_modules/lazy/test/foldr.js deleted file mode 100644 index 407fa30..0000000 --- a/node_modules/lazy/test/foldr.js +++ /dev/null @@ -1,26 +0,0 @@ -var assert = require('assert'); -var Lazy = require('..'); -var expresso = expresso; - -function range(i, j) { - var r = []; - for (;i 0); - }) - .join(function (lines) { - assert.deepEqual( - lines.map(function (x) { return x.toString() }), - 'foo bar baz quux moo'.split(' ') - ); - joined = true; - }); - ; - - setTimeout(function () { - lazy.push(new Buffer('foo\nbar')); - lazy.push(new Buffer('\nbaz\nquux\nmoo')); - lazy.push(new Buffer('')); - lazy.push(new Buffer('\ndoom')); - lazy.end(); - assert.ok(joined); - }, 50); -}; - -exports['string lines'] = function () { - var lazy = Lazy(); - var joined = false; - lazy.lines - .forEach(function (line) { - assert.ok(line instanceof Buffer); - assert.ok(!line.toString().match(/\n/)); - assert.ok(line.length > 0); - }) - .join(function (lines) { - assert.deepEqual( - lines.map(function (x) { return x.toString() }), - 'foo bar baz quux moo'.split(' ') - ); - joined = true; - }); - ; - - setTimeout(function () { - lazy.push('foo\nbar'); - lazy.push('\nbaz\nquux\nmoo'); - lazy.push(''); - lazy.push('\ndoom'); - lazy.end(); - assert.ok(joined); - }, 50); -}; - -exports.endStream = function () { - var to = setTimeout(function () { - assert.fail('never finished'); - }, 2500); - - var em = new EventEmitter; - var i = 0; - var lines = []; - Lazy(em).lines.forEach(function (line) { - i ++; - lines.push(line); - if (i == 2) { - clearTimeout(to); - assert.eql(lines.map(String), [ 'foo', 'bar' ]); - } - }); - - setTimeout(function () { - em.emit('data', 'fo'); - }, 100); - - setTimeout(function () { - em.emit('data', 'o\nbar'); - }, 150); - - setTimeout(function () { - em.emit('end'); - }, 200); -}; diff --git a/node_modules/lazy/test/map.js b/node_modules/lazy/test/map.js deleted file mode 100644 index a3010ef..0000000 --- a/node_modules/lazy/test/map.js +++ /dev/null @@ -1,29 +0,0 @@ -var assert = require('assert'); -var Lazy = require('..'); -var expresso = expresso; - -function range(i, j) { - var r = []; - for (;i i) for (;ij;i-=s) r.push(i); - return r; -} - -exports['infinite range'] = function () { - var joinExecuted = false; - Lazy.range('10..').take(10).join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(10, 20)); - assert.equal(xs.length, 10); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['infinite range half-open'] = function () { - var joinExecuted = false; - Lazy.range('(10..').take(10).join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(11, 21)); - assert.equal(xs.length, 10); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range i'] = function () { - var joinExecuted = false; - Lazy.range(10).join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(0, 10)); - assert.equal(xs.length, 10); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range i,j (ij)'] = function () { - var joinExecuted = false; - Lazy.range(10, 0, 2).join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(10, 0, 2)); - assert.equal(xs.length, 5); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range i,j (i>j)'] = function () { - var joinExecuted = false; - Lazy.range(10, -8).join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(10, -8)); - assert.equal(xs.length, 18); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range i..j (ij)'] = function () { - var joinExecuted = false; - Lazy.range('50..44').join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(50, 44)); - assert.equal(xs.length, 6); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range i,next..j (ij)'] = function () { - var joinExecuted = false; - Lazy.range('4,3.9..1').join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(4,1,0.1)); - assert.equal(xs.length, 30); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range [i..j] (ij)'] = function () { - var joinExecuted = false; - Lazy.range('[10..1]').join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(10,0)); - assert.equal(xs.length, 10); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range [i..j) (ij)'] = function () { - var joinExecuted = false; - Lazy.range('[10..1)').join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(10,1)); - assert.equal(xs.length, 9); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range (i..j] (ij)'] = function () { - var joinExecuted = false; - Lazy.range('(10..1]').join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(9,0)); - assert.equal(xs.length, 9); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range (i..j) (ij)'] = function () { - var joinExecuted = false; - Lazy.range('(10..1)').join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(9,1)); - assert.equal(xs.length, 8); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - -exports['range [i,step..j]'] = function () { - var joinExecuted = false; - Lazy.range('[5,10..50]').join(function (xs) { - joinExecuted = true; - assert.deepEqual(xs, range(5,51,5)); - assert.equal(xs.length, 10); - }); - - setTimeout(function () { - assert.ok(joinExecuted, 'join didn\'t execute'); - }, 2000); -} - diff --git a/node_modules/lazy/test/skip.js b/node_modules/lazy/test/skip.js deleted file mode 100644 index a7714ee..0000000 --- a/node_modules/lazy/test/skip.js +++ /dev/null @@ -1,27 +0,0 @@ -var assert = require('assert'); -var Lazy = require('..'); -var expresso = expresso; - -function range(i, j) { - var r = []; - for (;i1.4.0", "express": "~3.1.0", + "iced-coffee-script": "1.6.3-b", "jade": "~0.28.2", - "lazy": "~1.0.9" + "mkdirp": "^0.5.0", + "node-lixian": "^1.0.5", + "request": "^2.44.0", + "status-bar": "^2.0.1" }, "bin": { "lixian-portal": "./bin/lixian-portal" }, "scripts": { "test": "echo \"Error: no test specified\" && exit 1", - "fetch-xunlei-lixian": "rm -Rf xunlei-lixian && mkdir xunlei-lixian && curl -L https://github.com/iambus/xunlei-lixian/archive/master.tar.gz | tar -xz --strip-component 1 -C xunlei-lixian", "start": "node index.js" }, "main": "bin/lixian-portal", @@ -23,6 +25,5 @@ }, "author": "Michael Yin", "license": "BSD", - "readmeFilename": "README.md", - "gitHead": "6bdc29c4643ed1bc9e540ad481e2693018f748c9" + "readmeFilename": "README.md" } diff --git a/views/error.jade b/views/error.jade index 2758ac9..9103371 100644 --- a/views/error.jade +++ b/views/error.jade @@ -2,5 +2,8 @@ extends layout block content .alert.alert-warning h1 操作失败⋯⋯ - =error.message - a.btn.btn-primary(href='.') 返回 \ No newline at end of file + if error.message + = error.message + else + |#{error.stderr}; #{error.stdout} + a.btn.btn-primary(href='.') 返回 diff --git a/views/layout.jade b/views/layout.jade index 983d69b..823192f 100644 --- a/views/layout.jade +++ b/views/layout.jade @@ -12,9 +12,6 @@ html body .container h1 lixian... - if info.stats.task - h6 正在进行任务: #{info.stats.task.name} - h6 取回速度: #{info.stats.speed} .row .col-lg-12 block content diff --git a/views/login.jade b/views/login.jade index cb0b0e5..af0437e 100644 --- a/views/login.jade +++ b/views/login.jade @@ -12,7 +12,7 @@ block content .form-group input.form-control(name='vcode',type='text',placeholder='验证码') .form-group - img(src="data:image/jpeg;base64,#{info.stats.requireVerificationCode}") + img(src=info.stats.requireVerificationCode) .form-group button.btn.btn-primary(type='submit') i.icon-signin diff --git a/views/tasks.jade b/views/tasks.jade index dcdda02..51063d5 100644 --- a/views/tasks.jade +++ b/views/tasks.jade @@ -1,11 +1,30 @@ extends layout block content + script. + $(function(){ + var fn = function(){ + $.get('.', function(data){ + $('#stats').html($('#stats', data).html()); + $('#tasks').html($('#tasks', data).html()); + $('#queue').html($('#queue', data).html()); + $('#log').html($('#log', data).html()); + setTimeout(fn, 500); + }); + }; + fn(); + }); + #stats + if info.stats.executings.length + h6 正在进行任务: + ul + for executing in info.stats.executings + li #{executing} + + if info.stats.retrieving + h6 取回速度: #{info.stats.retrieving.format.speed(info.stats.retrieving.progress.speed)} form.form-inline(method='post',action='/', enctype='multipart/form-data',style='margin-bottom: 0.5em;') .pull-right .btn-group - a.btn.btn-info(href='/') - i.icon-refresh - |刷新 a.btn.btn-danger(href='/logout') i.icon-signout |登出 @@ -20,32 +39,34 @@ block content label input(name='bt',type='checkbox',accept='*/torrent',onclick='$(".url").toggle()') | BT任务 - table.table + table.table#tasks tr + th th 任务状态 th 取回状态 th 文件名 th 操作 for task in info.stats.tasks - tr(class=task.status) - td=task.statusLabel + tr + td(colspan=4) #{task.name} td - if info.stats.retrieving && info.stats.retrieving.task.id == task.id - .label.label-info 正在取回(#{info.stats.progress} 预计剩余时间:#{info.stats.time}) - if info.stats.error[task.id] - .label.label-error(title=info.stats.error[task.id]) 错误 - td=task.filename - td - .btn-toolbar(style='margin:0;') - .btn-group - a.btn-mini.btn.btn-danger(href='/tasks/#{task.id}?_method=delete') 删除 + a.btn-xs.btn.btn-inline.btn-danger(href='/tasks/#{task.id}?_method=delete') 删除 + for file in task.files + tr(class=file.status) + td + td=file.statusLabel + td + if info.stats.retrieving && info.stats.retrieving.file.url == file.url + .label.label-info 正在取回(#{info.stats.retrieving.format.storage(info.stats.retrieving.progress.currentSize)}, #{info.stats.retrieving.format.percentage(info.stats.retrieving.progress.percentage)} 预计剩余时间:#{info.stats.retrieving.format.time(info.stats.retrieving.progress.remainingTime)}) + td #{file.name} + td #accordion2.panel-group .panel.panel-default .panel-heading .panel-title - a.accordion-toggle(data-toggle='collapse', data-parent='#accordion2', href='#collapseOne') + a.accordion-toggle(data-toggle='collapse', data-parent='#accordion2', href='#queue') | 任务队列 - #collapseOne.panel-body.collapse.in + #queue.panel-body.collapse.in .panel-inner ul for item in info.queue @@ -53,8 +74,8 @@ block content .panel.panel-default .panel-heading .panel-title - a.accordion-toggle(data-toggle='collapse', data-parent='#accordion2', href='#collapseTwo') + a.accordion-toggle(data-toggle='collapse', data-parent='#accordion2', href='#log') | 任务日志 - #collapseTwo.panel-body.collapse + #log.panel-body.collapse .panel-inner pre=info.log.join('\n') diff --git a/xunlei-lixian/.gitignore b/xunlei-lixian/.gitignore deleted file mode 100644 index 0d20b64..0000000 --- a/xunlei-lixian/.gitignore +++ /dev/null @@ -1 +0,0 @@ -*.pyc diff --git a/xunlei-lixian/LICENSE b/xunlei-lixian/LICENSE deleted file mode 100644 index c62024a..0000000 --- a/xunlei-lixian/LICENSE +++ /dev/null @@ -1,23 +0,0 @@ -============================================== -This is a copy of the MIT license. -============================================== -Copyright (C) 2012 Boyu Guo - -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/xunlei-lixian/README.md b/xunlei-lixian/README.md deleted file mode 100644 index e849e5a..0000000 --- a/xunlei-lixian/README.md +++ /dev/null @@ -1,468 +0,0 @@ -xunlei-lixian -============= -迅雷离线下载脚本。欢迎继续关注[浏览器端的迅雷离线应用](https://github.com/iambus/xunlei-lixian-web)。 - -### 声明 -迅雷离线下载为会员功能。非会员无法使用。 - -Quick start ------------ - - python lixian_cli.py login "Your Xunlei account" "Your password" - python lixian_cli.py login "Your password" - python lixian_cli.py login - - python lixian_cli.py config username "Your Xunlei account" - python lixian_cli.py config password "Your password" - - python lixian_cli.py list - python lixian_cli.py list --completed - python lixian_cli.py list --completed --name --original-url --download-url --no-status --no-id - python lixian_cli.py list --deleted - python lixian_cli.py list --expired - python lixian_cli.py list id1 id2 - python lixian_cli.py list zip rar - python lixian_cli.py list 2012.04.04 2012.04.05 - - python lixian_cli.py download task-id - python lixian_cli.py download ed2k-url - python lixian_cli.py download --tool=wget ed2k-url - python lixian_cli.py download --tool=asyn ed2k-url - python lixian_cli.py download ed2k-url --output "file to save" - python lixian_cli.py download id1 id2 id3 - python lixian_cli.py download url1 url2 url3 - python lixian_cli.py download --input download-urls-file - python lixian_cli.py download --input download-urls-file --delete - python lixian_cli.py download --input download-urls-file --output-dir root-dir-to-save-files - python lixian_cli.py download bt://torrent-info-hash - python lixian_cli.py download 1.torrent - python lixian_cli.py download torrent-info-hash - python lixian_cli.py download --bt http://xxx/xxx.torrent - python lixian_cli.py download bt-task-id/file-id - python lixian_cli.py download --all - python lixian_cli.py download mkv - python lixian_cli.py download 2012.04.04 - python lixian_cli.py download 0 1 2 - python lixian_cli.py download 0-2 - - python lixian_cli.py add url - python lixian_cli.py add 1.torrent - python lixian_cli.py add torrent-info-hash - python lixian_cli.py add --bt http://xxx/xxx.torrent - - python lixian_cli.py delete task-id - python lixian_cli.py delete url - python lixian_cli.py delete file-name-on-cloud-to-delete - - python lixian_cli.py pause id - - python lixian_cli.py restart id - - python lixian_cli.py rename id name - - python lixian_cli.py logout - -安装指南 --------- - -1. 安装git(非github用户应该只需要执行第一步Download and Install Git) - - http://help.github.com/set-up-git-redirect - -2. 下载代码(Windows用户请在git-bash里执行) - - git clone git://github.com/iambus/xunlei-lixian.git - -3. 安装Python 2.x(请下载最新的2.7版本。不支持Python 3.x。) - - http://www.python.org/getit/ - -4. 在命令行里运行 - - python lixian_cli.py - -注:不方便安装git的用户可以选择跳过前两步,在github网页上下载最新的源代码包(选择"Download as zip"或者"Download as tar.gz"): - -https://github.com/iambus/xunlei-lixian/downloads - - -一些提示 --------- - -1. 你可以为python lixian_cli.py创建一个别名(比如lx),以减少敲键次数。 - - Linux上可以使用: - - ln -s 你的lixian_cli.py路径 ~/bin/lx - - Windows上可以创建一个lx.bat脚本,放在你的PATH中: - - @echo off - python 完整的lixian_cli.py路径 %* - - 注:下文中提到的lx都是指python lixian_cli.py的别名。 - -2. 你可以使用lx config保存一些配置。见“命令详解”一节。 - - lx config delete - lx config tool asyn - lx config username your-id - lx config password your-password - - 注:密码保存的时候会加密(hash) - -3. 部分命令有短名字。lx d相当于lx download,lx a相当于lx add,lx l相当于lx list,lx x相当于lx list。也可以通过plugin api自己添加alias。 - -4. 使用lx download下载的文件会自动验证hash。其中ed2k和bt会做完整的hash校验。http下载只做部分校验。 - - 注:包含多个文件的bt种子,如果没有完整下载所有文件,对于已下载的文件,可能有少量片段无法验证。如果很重视文件的正确性请选择下载bt种子中的所有文件。(目前还没有发现由于软件问题而导致hash验证失败的情况。) - -5. 如果觉得大文件的hash速度太慢,可以关掉: - - lx download --no-hash ... - - 也可以使用lx config默认关掉它: - - lx config no-hash - -6. lx hash命令可以用于手动计算hash。见“其他工具”一节。 - - -命令详解 --------- - -注:下文中提到的lx都是指python lixian_cli.py的别名。 - -常用命令: - -* lx login -* lx download -* lx list -* lx add -* lx delete -* lx pause -* lx restart -* lx rename -* lx readd -* lx config -* lx info -* lx help - -### lx login -登录,获得一个有效session,默认保存路径是~/.xunlei.lixian.cookies。一般来说,除非服务器故障或者执行了lx logout(或者你手动删除了cookies文件),否则session的有效期是一天左右。session过期之后需要手动重新执行login。但如果使用lx config password把密码保存到配置文件里,则会自动重新登录。后文会介绍[lx config](#lx-config)。 - -lx login接受两个参数,用户名和密码。第二次登录可以只填密码。 - - lx login username password - lx login password - -如果不希望明文显示密码,也可以直接运行 - - lx login - -或者使用-代替密码 - - lx login username - - -上面的命令会进入交互式不回显的密码输入。 - -可以用--cookies指定保存的session文件路径。-表示不保存(在login这个例子里,没什么实际意义)。 - - lx login username password --cookies some-path - lx login username password --cookies - - -注意,除了lx login外,大多数lx命令,比如lx download,都需要先执行登录。这些命令大多支持--username和--password,以及--cookies参数,根据传递进来的参数,检查用户是否已经登录,如果尚未登录则尝试登录。一般来说不建议在其他命令里使用这些参数(因为麻烦),除非你不希望保存session信息到硬盘。 - -### lx download -下载。目前支持普通的http下载,ed2k下载,和bt下载。可以使用thunder/flashget/qq旋风的连接(bt任务除外)。在信息足够的情况下(见“一些提示”一节的第3条),下载的文件会自动验证hash,出错了会重新下载(我个人目前还没遇到过下载文件损坏的情况)。见“一些提示”一节的第3条。 - - lx download id - lx download http://somewhere - lx download ed2k://somefile - lx download bt://info-hash - lx download link1 link2 link3 ... - lx download --all - lx download keywords - lx download date - -对于bt任务,可以指定本地.torrent文件路径,或者torrent文件的info hash。(很多网站使用info hash来标识一个bt种子文件,这种情况你就不需要下载种子了,lx download可以自动下载种子,不过前提是之前已经有人使用迅雷离线下载过同样的种子。[如后所述](#其他工具),你也可以使用lx hash --info-hash来手动生成bt种子的info hash。) - - lx download Community.S03E01.720p.HDTV.X264-DIMENSION.torrent - lx download 61AAA3C6FBB8B71EBE2F5A2A3481296B51D882F6 - lx download bt://61AAA3C6FBB8B71EBE2F5A2A3481296B51D882F6 - -如果url本身指向了要添加任务的种子文件,需要加上--bt参数告诉lx脚本这是一个种子。 - - lx download --bt http://tvu.org.ru/torrent.php?tid=64757 - -可以把多个连接保存到文件里,使用--input参数批量下载: - - lx download --input links.txt - -注意:在断点续传的情况下,如果文件已经存在,并且文件大小相等,并且使用了--continue,重新下载并不只是简单的忽略这个文件,而是先做hash校验,如果校验通过才忽略。如果文件比较多或者比较大,可能比较耗时。建议手动从--input文件里删除已经下载过的链接。也可以使用--mini-hash参数,如下。 - -如果指定了--mini-hash参数,对于已经下载过的文件,并且文件大小正确(一般意味着这个文件的正确性已经在前一次下载中验证过了),会做一个最简单的校验。对于尚未下载完成的任务,在完成之后还是会做完整的hash。 - -如果指定了--no-hash参数,永远不会做完整的hash。但还是会做文件大小检验和取样hash(很快)。 - -可以使用--delete参数在下载完成之后删除任务。 - - lx download link --delete - -如果一个文件已经存在,使用参数--continue支持断点续传,使用--overwrite覆盖已存在的文件,重新下载。 - -你可能需要用--tool参数来指定下载工具。默认的下载工具是wget,有些环境的wget是最低功能版本,不支持指定cookie或者断点续传。这种情况可以使用--tool=asyn。这在“支持的下载工具”一节有说明。 - - lx download --tool=wget link - lx download --tool=asyn link - ---output和--output-dir分别用来指定保存文件的路径和目录。 - -如果要下载的文件尚未在离线任务里,会被自动添加。 - -你也可以使用指定要下载的任务id(lx list命令可以用来查看任务id): - - lx download task-id - -但是要注意,多任务下载的时候,不能混用id和url(以后可能会支持)。 - -类似任务id,也可以指定任务的序列号。序列号从0开始。可以使用lx list -n查看序列号。如果希望lx list默认显示序列号,可以使用lx config n。若要下载任务列表中的第一个任务: - - lx download 0 - -要下载前三个任务: - - lx download 0-2 - -对于bt任务,如果只想下载部分文件,可以在task id后指定文件id: - - lx download bt-task-id/file-id bt-task-id/file-id2 - -或者: - - lx download bt-task-id/[1,3,5-7] - -注:上面的命令下载对应bt任务里文件id为1,3,5,6,7的五个文件。 - -也可以指定bt子文件的扩展名: - - lx download bt-task-id/.mkv - -或者: - - lx download bt-task-id/[.mkv,.mp4] - -更多的用法:TODO - -可以使用--all参数下载所有的任务(如果已经在参数中指定了要下载的链接或者任务id,--all参数会被忽略): - - lx download --all - -也可以使用一个简单的关键字匹配要下载的文件名: - - lx download mkv - -也可以搜索多个关键字(满足其中一个就算匹配): - - lx download mkv mp4 - -任务的添加日期也可以作为关键字: - - lx download 2012.04.04 - lx download 2012.04.04 2012.04.05 - -### lx list -列出已存在的离线任务。默认只会列出任务id,任务名,以及状态。可以使用--original-url和--download-url参数来列出原始链接和下载链接。--completed参数用于忽略未完成任务。 - - lx list - lx list --completed - lx list --no-status --original-url --download-url - -如果要列出bt任务的子文件,可以在任务id后面加上/: - - lx list id/ - -可以使用--deleted或者--expired参数来列出已删除和已过期的任务。 - -详细参数可以参考lx help list。 - -### lx add -添加任务到迅雷离线服务器上。 - - lx add url1 url2 url3 - lx add --input links.txt - lx add --bt torrent-file - lx add --bt torrent-url - lx add --bt info-hash - -提示:lx download会自动添加任务,而无需执行lx add。 - -### lx delete -从迅雷离线服务器上删除任务。 - - lx delete id1 id2 - lx delete ed2k://... - lx delete mkv - lx delete --all mkv - lx delete --all mkv mp4 - -### lx pause -暂停任务。 - - lx pause id1 id2 - lx pause --all mkv - -### lx restart -重新开始任务。 - - lx restart id1 id2 - lx restart --all mkv - -### lx rename -重命名任务 - - lx rename task-id task-name - -### lx logout -不想保留session可以使用lx logout退出。一般用不着。 - - lx logout - lx logout --cookies your-cookies-file - -### lx readd -重新添加已过期或者已删除的任务。 - - lx readd --deleted task-id - lx readd --expired task-name - -提示:可以用lx list --deleted或者lx list --expired列出已删除和过期的任务。 - -### lx config -保存配置。配置文件的保存路径是~/.xunlei.lixian.config。虽然你可以差不多可以保存任何参数,但是目前只有以下几个参数会真正起作用: - -* username -* password -* tool -* continue -* delete -* output-dir -* hash -* mini-hash -* id -* n -* size -* format-size -* colors -* wget-opts(见稍后的说明) -* aria2-opts(见稍后的说明)(见支持的下载工具一节) -* axel-opts(见稍后的说明) -* watch-interval -* log-level -* log-path - -(因为只有这几个参数我觉得是比较有用的。如果你觉得其他的参数有用可以发信给我或者直接open一个issue。) - -不加参数会打印当前保存的所有配置: - - lx config - -可以使用--print打印指定的配置: - - lx config --print password - -添加一个新的参数: - - lx config username your-username - lx config password your-password - lx config delete - lx config no-delete - -删除一个参数: - - lx config --delete password - -注:密码是hash过的,不是明文保存。 -注:如果不希望在命令行参数中明文保存密码,可以运行lx config password,或者lx config password -,会进入交互式不回显密码输入(只支持password配置)。 - -关于wget-opts/aria2-opts/axel-opts,因为这些工具的命令行参数一般都包含-,所以需要用额外的--转义。另外多个命令行参数需要用引号合并到一起: - - lx config -- aria2-opts "-s10 -x10 -c" - -### lx info -打印cookies文件里保存的迅雷内部id,包括登录的ID,一个内部使用的ID,以及gdriveid。 - -关于gdriveid:理论上gdriveid是下载迅雷离线链接需要的唯一cookie,你可以用lx list --download-url获取下载地址,然后用lx info获取gdriveid,然后手动使用其他工具下载,比如wget "--header=Cookie: gdriveid=your-gdriveid" download-url。 - --i参数可以只打印登录ID: - - lx info -i - -如果想把登录id复制到剪切板: - - lx info -i | clip - -### lx help -打印帮助信息。 - - lx help - lx help examples - lx help readme - lx help download - -支持的下载工具 --------------- - -* wget:默认下载工具。注意有些Linux发行版(比如某些运行在路由设备上的mini系统)自带的wget可能无法满足功能要求。可以尝试使用其他工具。 -* asyn:内置的下载工具。在命令行中加上--tool=asyn可以启用。注意此工具的下载表现一般,在高速下载或者设备性能不太好的情况(比如运行在低端路由上),CPU使用可能稍高。在我的RT-N16上,以250K/s的速度下载,CPU使用大概在10%~20%。 -* urllib2:内置下载工具。不支持断点续传错误重连,不建议使用。 -* curl:尚未测试。 -* aria2:测试通过。注意某些环境里的aria2c需要加上额外的参数才能运行。可以使用lx config进行配置:lx config -- aria2-opts --event-poll=select -* axel: 测试通过。注意官方版本的axel有一个URL重定向长度超过255被截断的bug,需要手动修改源代码编译。见issue #44. -* 其他工具,比如ProZilla,暂时都不支持。有需要请可以我,或者直接提交一个issue。 - - -其他工具 --------- - -* lx hash可以用于手动计算hash。 - - lx hash --ed2k filename - lx hash --info-hash torrent-file - lx hash --verify-sha1 filename sha1 - lx hash --verify-bt filename torrent-file - -* lixian_batch.py是我自己用的一个简单的“多任务”下载脚本。其实就是多个--input文件,每个文件里定义的链接下载到文件所在的目录里。 - - python lixian_batch.py folder1/links.txt folder2/links.txt ... - -既知问题 --------- - -1. --tool=asyn的性能不是很好。见“支持的下载工具”一节里的说明。 -2. 有些时候任务添加到服务器上,但是马上刷新拿不到这个数据。这应该是服务器同步的问题。技术上可以自动重刷一遍,但是暂时没有做。用户可以自己重试下。 -3. bt下载的校验如果失败,可能需要重新下载所有文件。从技术上来讲这是没有必要的。但是一来重下出错的片段有些繁琐,二来我自己都从来没遇到过bt校验失败需要重下的情况,所以暂时不考虑支持片段修复。更新:bt校验失败不会重下。 -4. 有时候因为帐号异常,登录需要验证码。目前还不支持验证码。 - -以后 ----- - -其实一开始是考虑做一个可以在路由器上运行的网页版离线下载管理器的。但是这个工作量比命令行版的大很多(不是一个数量级的),在资源消耗和出错概率上也大很多,而且可能还要有更多的依赖库,安装起来也不方便。当然主要还是精力和需求的原因。现在的这个命令行本对我来说已经够用了,也挺简单,短期就不考虑增加网页版了。 - -相关项目 --------- - -* [layerssss/lixian-portal](http://micy.in/lixian-portal/): 给iambus/xunlei-lixian做的一个简洁实用的webui - -特别感谢 --------- - -[群晖公司](http://www.synology.com/)在部分产品中绑定了迅雷离线脚本,并且捐赠了作者一台[DS213+](http://www.synology.com/products/product.php?product_name=DS213%2B)作为反馈。再此表示感谢! - -许可协议 --------- - -xunlei-lixian使用MIT许可协议。 - -此文档未完成。 --------------- - diff --git a/xunlei-lixian/lixian.py b/xunlei-lixian/lixian.py deleted file mode 100644 index 01d073e..0000000 --- a/xunlei-lixian/lixian.py +++ /dev/null @@ -1,1093 +0,0 @@ - -__all__ = ['XunleiClient'] - -import urllib -import urllib2 -import cookielib -import re -import time -import os.path -import json -from ast import literal_eval - - -def retry(f_or_arg, *args): - #retry_sleeps = [1, 1, 1] - retry_sleeps = [1, 2, 3, 5, 10, 20, 30, 60] + [60] * 60 - def decorator(f): - def withretry(*args, **kwargs): - for second in retry_sleeps: - try: - return f(*args, **kwargs) - except: - import traceback - logger.debug("Exception happened. Retrying...") - logger.debug(traceback.format_exc()) - time.sleep(second) - raise - return withretry - if callable(f_or_arg) and not args: - return decorator(f_or_arg) - else: - a = f_or_arg - assert type(a) == int - assert not args - retry_sleeps = [1] * a - return decorator - -class Logger: - def stdout(self, message): - print message - def info(self, message): - print message - def debug(self, message): - pass - def trace(self, message): - pass - -logger = Logger() - -class WithAttrSnapshot: - def __init__(self, object, **attrs): - self.object = object - self.attrs = attrs - def __enter__(self): - self.old_attrs = [] - for k in self.attrs: - if hasattr(self.object, k): - self.old_attrs.append((k, True, getattr(self.object, k))) - else: - self.old_attrs.append((k, False, None)) - for k in self.attrs: - setattr(self.object, k, self.attrs[k]) - def __exit__(self, exc_type, exc_val, exc_tb): - for k, has_old_attr, v in self.old_attrs: - if has_old_attr: - setattr(self.object, k, v) - else: - delattr(self.object, k) - -class WithAttr: - def __init__(self, object): - self.object = object - def __call__(self, **kwargs): - return WithAttrSnapshot(self.object, **kwargs) - def __getattr__(self, k): - return lambda (v): WithAttrSnapshot(self.object, **{k:v}) - -# TODO: write unit test -class OnDemandTaskList: - def __init__(self, fetch_page, page_size, limit): - self.fetch_page = fetch_page - if limit and page_size > limit: - page_size = limit - self.page_size = page_size - self.limit = limit - self.pages = {} - self.max_task_number = None - self.real_total_task_number = None - self.total_pages = None - - def is_out_of_range(self, n): - if self.limit: - if n >= self.limit: - return True - if self.max_task_number: - if n >= self.max_task_number: - return True - if self.real_total_task_number: - if n >= self.real_total_task_number: - return True - - def check_out_of_range(self, n): - if self.is_out_of_range(n): - raise IndexError('task index out of range') - - def is_out_of_page(self, page): - raise NotImplementedError() - - def get_nth_task(self, n): - self.check_out_of_range(n) - page = n / self.page_size - n_in_page = n - page * self.page_size - return self.hit_page(page)[n_in_page] - - def touch(self): - self.hit_page(0) - - def hit_page(self, page): - if page in self.pages: - return self.pages[page] - info = self.fetch_page(page, self.page_size) - tasks = info['tasks'] - if self.max_task_number is None: - self.max_task_number = info['total_task_number'] - if self.limit and self.max_task_number > self.limit: - self.max_task_number = self.limit - self.total_pages = self.max_task_number / self.page_size - if self.max_task_number % self.page_size != 0: - self.total_pages += 1 - if self.max_task_number == 0: - self.real_total_task_number = 0 - if page >= self.total_pages: - tasks = [] - elif page == self.total_pages - 1: - if self.page_size * page + len(tasks) > self.max_task_number: - tasks = tasks[0:self.max_task_number - self.page_size * page] - if len(tasks) > 0: - self.real_total_task_number = self.page_size * page + len(tasks) - else: - self.max_task_number -= self.page_size - self.total_pages -= 1 - if len(self.pages.get(page-1, [])) == self.page_size: - self.real_total_task_number = self.max_task_number - else: - if len(tasks) == 0: - self.max_task_number = self.page_size * page - self.total_pages = page - if len(self.pages.get(page-1, [])) == self.page_size: - self.real_total_task_number = self.max_task_number - elif len(tasks) < self.page_size: - self.real_total_task_number = self.page_size * page + len(tasks) - self.max_task_number = self.real_total_task_number - self.total_pages = page - else: - pass - for i, t in enumerate(tasks): - t['#'] = self.page_size * page + i - self.pages[page] = tasks - return tasks - - def __getitem__(self, n): - return self.get_nth_task(n) - - def __iter__(self): - class Iterator: - def __init__(self, container): - self.container = container - self.current = 0 - def next(self): - self.container.touch() - assert type(self.container.max_task_number) == int - if self.container.real_total_task_number is None: - if self.current < self.container.max_task_number: - try: - task = self.container[self.current] - except IndexError: - raise StopIteration() - else: - raise StopIteration() - else: - if self.current < self.container.real_total_task_number: - task = self.container[self.current] - else: - raise StopIteration() - self.current += 1 - return task - return Iterator(self) - - def __len__(self): - if self.real_total_task_number: - return self.real_total_task_number - self.touch() - self.hit_page(self.total_pages-1) - if self.real_total_task_number: - return self.real_total_task_number - count = 0 - for t in self: - count += 1 - return count - -class XunleiClient(object): - default_page_size = 100 - default_bt_page_size = 9999 - def __init__(self, username=None, password=None, cookie_path=None, login=True, verification_code_reader=None): - self.attr = WithAttr(self) - - self.username = username - self.password = password - self.cookie_path = cookie_path - if cookie_path: - self.cookiejar = cookielib.LWPCookieJar() - if os.path.exists(cookie_path): - self.load_cookies() - else: - self.cookiejar = cookielib.CookieJar() - - self.page_size = self.default_page_size - self.bt_page_size = self.default_bt_page_size - - self.limit = None - - self.opener = urllib2.build_opener(urllib2.HTTPCookieProcessor(self.cookiejar)) - self.verification_code_reader = verification_code_reader - self.login_time = None - if login: - self.id = self.get_userid_or_none() - if not self.id: - self.login() - self.id = self.get_userid() - - @property - def page_size(self): - return self._page_size - @page_size.setter - def page_size(self, size): - self._page_size = size - self.set_page_size(size) - - @retry - def urlopen(self, url, **args): - logger.debug(url) -# import traceback -# for line in traceback.format_stack(): -# print line.strip() - if 'data' in args and type(args['data']) == dict: - args['data'] = urlencode(args['data']) - return self.opener.open(urllib2.Request(url, **args), timeout=60) - - def urlread1(self, url, **args): - args.setdefault('headers', {}) - headers = args['headers'] - headers.setdefault('Accept-Encoding', 'gzip, deflate') -# headers.setdefault('Referer', 'http://lixian.vip.xunlei.com/task.html') -# headers.setdefault('User-Agent', 'Mozilla/5.0 (Windows NT 6.1; WOW64; rv:11.0) Gecko/20100101 Firefox/11.0') -# headers.setdefault('Accept', 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8') -# headers.setdefault('Accept-Language', 'zh-cn,zh;q=0.7,en-us;q=0.3') - response = self.urlopen(url, **args) - data = response.read() - if response.info().get('Content-Encoding') == 'gzip': - data = ungzip(data) - elif response.info().get('Content-Encoding') == 'deflate': - data = undeflate(data) - return data - - def urlread(self, url, **args): - data = self.urlread1(url, **args) - if self.is_session_timeout(data): - logger.debug('session timed out') - self.login() - data = self.urlread1(url, **args) - return data - - def load_cookies(self): - self.cookiejar.load(self.cookie_path, ignore_discard=True, ignore_expires=True) - - def save_cookies(self): - if self.cookie_path: - self.cookiejar.save(self.cookie_path, ignore_discard=True) - - def get_cookie(self, domain, k): - if self.has_cookie(domain, k): - return self.cookiejar._cookies[domain]['/'][k].value - - def has_cookie(self, domain, k): - return domain in self.cookiejar._cookies and k in self.cookiejar._cookies[domain]['/'] - - def get_userid(self): - if self.has_cookie('.xunlei.com', 'userid'): - return self.get_cookie('.xunlei.com', 'userid') - else: - raise Exception('Probably login failed') - - def get_userid_or_none(self): - return self.get_cookie('.xunlei.com', 'userid') - - def get_username(self): - return self.get_cookie('.xunlei.com', 'usernewno') - - def get_gdriveid(self): - return self.get_cookie('.vip.xunlei.com', 'gdriveid') - - def has_gdriveid(self): - return self.has_cookie('.vip.xunlei.com', 'gdriveid') - - def get_referer(self): - return 'http://dynamic.cloud.vip.xunlei.com/user_task?userid=%s' % self.id - - def set_cookie(self, domain, k, v): - c = cookielib.Cookie(version=0, name=k, value=v, port=None, port_specified=False, domain=domain, domain_specified=True, domain_initial_dot=False, path='/', path_specified=True, secure=False, expires=None, discard=True, comment=None, comment_url=None, rest={}, rfc2109=False) - self.cookiejar.set_cookie(c) - - def del_cookie(self, domain, k): - if self.has_cookie(domain, k): - self.cookiejar.clear(domain=domain, path="/", name=k) - - def set_gdriveid(self, id): - self.set_cookie('.vip.xunlei.com', 'gdriveid', id) - - def set_page_size(self, n): - self.set_cookie('.vip.xunlei.com', 'pagenum', str(n)) - - def get_cookie_header(self): - def domain_header(domain): - root = self.cookiejar._cookies[domain]['/'] - return '; '.join(k+'='+root[k].value for k in root) - return domain_header('.xunlei.com') + '; ' + domain_header('.vip.xunlei.com') - - def is_login_ok(self, html): - return len(html) > 512 - - def has_logged_in(self): - id = self.get_userid_or_none() - if not id: - return False - #print self.urlopen('http://dynamic.cloud.vip.xunlei.com/user_task?userid=%s&st=0' % id).read().decode('utf-8') - with self.attr(page_size=1): - url = 'http://dynamic.cloud.vip.xunlei.com/user_task?userid=%s&st=0' % id - #url = 'http://dynamic.lixian.vip.xunlei.com/login?cachetime=%d' % current_timestamp() - r = self.is_login_ok(self.urlread(url)) - return r - - def is_session_timeout(self, html): - is_timeout = html == '''''' or html == '''''' or html == '''''' - if is_timeout: - logger.trace(html) - return True - maybe_timeout = html == '''rebuild({"rtcode":-1,"list":[]})''' - if maybe_timeout: - if self.login_time and time.time() - self.login_time < 60 * 10: # 10 minutes - return False - else: - logger.trace(html) - return True - return is_timeout - - def read_verification_code(self): - if not self.verification_code_reader: - raise NotImplementedError('Verification code required') - else: - verification_code_url = 'http://verify.xunlei.com/image?cachetime=%s' % current_timestamp() - image = self.urlopen(verification_code_url).read() - return self.verification_code_reader(image) - - def login(self, username=None, password=None): - username = self.username - password = self.password - if not username and self.has_cookie('.xunlei.com', 'usernewno'): - username = self.get_username() - if not username: - # TODO: don't depend on lixian_config - import lixian_config - username = lixian_config.get_config('username') -# if not username: -# raise NotImplementedError('user is not logged in') - if not password: - raise NotImplementedError('user is not logged in') - - logger.debug('login') - cachetime = current_timestamp() - check_url = 'http://login.xunlei.com/check?u=%s&cachetime=%d' % (username, cachetime) - login_page = self.urlopen(check_url).read() - verification_code = self.get_cookie('.xunlei.com', 'check_result')[2:].upper() - if not verification_code: - verification_code = self.read_verification_code() - if verification_code: - verification_code = verification_code.upper() - assert verification_code - password = encypt_password(password) - password = md5(password+verification_code) - login_page = self.urlopen('http://login.xunlei.com/sec2login/', data={'u': username, 'p': password, 'verifycode': verification_code}) - self.id = self.get_userid() - with self.attr(page_size=1): - login_page = self.urlopen('http://dynamic.lixian.vip.xunlei.com/login?cachetime=%d&from=0'%current_timestamp()).read() - if not self.is_login_ok(login_page): - logger.trace(login_page) - raise RuntimeError('login failed') - self.save_cookies() - self.login_time = time.time() - - def logout(self): - logger.debug('logout') - #session_id = self.get_cookie('.xunlei.com', 'sessionid') - #timestamp = current_timestamp() - #url = 'http://login.xunlei.com/unregister?sessionid=%s&cachetime=%s&noCacheIE=%s' % (session_id, timestamp, timestamp) - #self.urlopen(url).read() - #self.urlopen('http://dynamic.vip.xunlei.com/login/indexlogin_contr/logout/').read() - ckeys = ["vip_isvip","lx_sessionid","vip_level","lx_login","dl_enable","in_xl","ucid","lixian_section"] - ckeys1 = ["sessionid","usrname","nickname","usernewno","userid"] - self.del_cookie('.vip.xunlei.com', 'gdriveid') - for k in ckeys: - self.set_cookie('.vip.xunlei.com', k, '') - for k in ckeys1: - self.set_cookie('.xunlei.com', k, '') - self.save_cookies() - self.login_time = None - - def to_page_url(self, type_id, page_index, page_size): - # type_id: 1 for downloading, 2 for completed, 4 for downloading+completed+expired, 11 for deleted, 13 for expired - if type_id == 0: - type_id = 4 - page = page_index + 1 - p = 1 # XXX: what is it? - # jsonp = 'jsonp%s' % current_timestamp() - # url = 'http://dynamic.cloud.vip.xunlei.com/interface/showtask_unfresh?type_id=%s&page=%s&tasknum=%s&p=%s&interfrom=task&callback=%s' % (type_id, page, page_size, p, jsonp) - url = 'http://dynamic.cloud.vip.xunlei.com/interface/showtask_unfresh?type_id=%s&page=%s&tasknum=%s&p=%s&interfrom=task' % (type_id, page, page_size, p) - return url - - @retry(10) - def read_task_page_info_by_url(self, url): - page = self.urlread(url).decode('utf-8', 'ignore') - data = parse_json_response(page) - if not self.has_gdriveid(): - gdriveid = data['info']['user']['cookie'] - self.set_gdriveid(gdriveid) - self.save_cookies() - # tasks = parse_json_tasks(data) - tasks = [t for t in parse_json_tasks(data) if not t['expired']] - for t in tasks: - t['client'] = self - # current_page = int(re.search(r'page=(\d+)', url).group(1)) - total_tasks = int(data['info']['total_num']) - # assert total_pages >= data['global_new']['page'].count('
  • .*?', page) - match_next_page = re.search(r'
  • ', page) - return tasks, match_next_page and 'http://dynamic.cloud.vip.xunlei.com'+match_next_page.group(1) - - def read_history_page(self, type=0, pg=None): - if pg is None: - url = 'http://dynamic.cloud.vip.xunlei.com/user_history?userid=%s&type=%d' % (self.id, type) - else: - url = 'http://dynamic.cloud.vip.xunlei.com/user_history?userid=%s&p=%d&type=%d' % (self.id, pg, type) - return self.read_history_page_url(url) - - def read_history(self, type=0): - '''read one page''' - tasks = self.read_history_page(type)[0] - for i, task in enumerate(tasks): - task['#'] = i - return tasks - - def read_all_history(self, type=0): - '''read all pages of deleted/expired tasks''' - all_tasks = [] - tasks, next_link = self.read_history_page(type) - all_tasks.extend(tasks) - while next_link: - if self.limit and len(all_tasks) > self.limit: - break - tasks, next_link = self.read_history_page_url(next_link) - all_tasks.extend(tasks) - if self.limit: - all_tasks = all_tasks[0:self.limit] - for i, task in enumerate(all_tasks): - task['#'] = i - return all_tasks - - def read_deleted(self): - return self.read_history() - - def read_all_deleted(self): - return self.read_all_history() - - def read_expired(self): - return self.read_history(1) - - def read_all_expired(self): - return self.read_all_history(1) - - def list_bt(self, task): - assert task['type'] == 'bt' - url = 'http://dynamic.cloud.vip.xunlei.com/interface/fill_bt_list?callback=fill_bt_list&tid=%s&infoid=%s&g_net=1&p=1&uid=%s&noCacheIE=%s' % (task['id'], task['bt_hash'], self.id, current_timestamp()) - with self.attr(page_size=self.bt_page_size): - html = remove_bom(self.urlread(url)).decode('utf-8') - sub_tasks = parse_bt_list(html) - for t in sub_tasks: - t['date'] = task['date'] - return sub_tasks - - def get_torrent_file_by_info_hash(self, info_hash): - url = 'http://dynamic.cloud.vip.xunlei.com/interface/get_torrent?userid=%s&infoid=%s' % (self.id, info_hash.upper()) - response = self.urlopen(url) - torrent = response.read() - if torrent == "": - raise Exception('Torrent file not found on xunlei cloud: '+info_hash) - assert response.headers['content-type'] == 'application/octet-stream' - return torrent - - def get_torrent_file(self, task): - return self.get_torrent_file_by_info_hash(task['bt_hash']) - - def add_task(self, url): - protocol = parse_url_protocol(url) - assert protocol in ('ed2k', 'http', 'https', 'ftp', 'thunder', 'Flashget', 'qqdl', 'bt', 'magnet'), 'protocol "%s" is not suppoted' % protocol - - from lixian_url import url_unmask - url = url_unmask(url) - protocol = parse_url_protocol(url) - assert protocol in ('ed2k', 'http', 'https', 'ftp', 'bt', 'magnet'), 'protocol "%s" is not suppoted' % protocol - - if protocol == 'bt': - return self.add_torrent_task_by_info_hash(url[5:]) - elif protocol == 'magnet': - return self.add_magnet_task(url) - - random = current_random() - check_url = 'http://dynamic.cloud.vip.xunlei.com/interface/task_check?callback=queryCid&url=%s&random=%s&tcache=%s' % (urllib.quote(url), random, current_timestamp()) - js = self.urlread(check_url).decode('utf-8') - qcid = re.match(r'^queryCid(\(.+\))\s*$', js).group(1) - qcid = literal_eval(qcid) - if len(qcid) == 8: - cid, gcid, size_required, filename, goldbean_need, silverbean_need, is_full, random = qcid - elif len(qcid) == 9: - cid, gcid, size_required, filename, goldbean_need, silverbean_need, is_full, random, ext = qcid - elif len(qcid) == 10: - cid, gcid, size_required, some_key, filename, goldbean_need, silverbean_need, is_full, random, ext = qcid - else: - raise NotImplementedError(qcid) - assert goldbean_need == 0 - assert silverbean_need == 0 - - if url.startswith('http://') or url.startswith('ftp://'): - task_type = 0 - elif url.startswith('ed2k://'): - task_type = 2 - else: - raise NotImplementedError() - task_url = 'http://dynamic.cloud.vip.xunlei.com/interface/task_commit?'+urlencode( - {'callback': 'ret_task', - 'uid': self.id, - 'cid': cid, - 'gcid': gcid, - 'size': size_required, - 'goldbean': goldbean_need, - 'silverbean': silverbean_need, - 't': filename, - 'url': url, - 'type': task_type, - 'o_page': 'task', - 'o_taskid': '0', - }) - - response = self.urlread(task_url) - assert response == 'ret_task(Array)', response - - def add_batch_tasks(self, urls, old_task_ids=None): - assert urls - urls = list(urls) - for url in urls: - if parse_url_protocol(url) not in ('http', 'https', 'ftp', 'ed2k', 'bt', 'thunder', 'magnet'): - raise NotImplementedError('Unsupported: '+url) - urls = filter(lambda u: parse_url_protocol(u) in ('http', 'https', 'ftp', 'ed2k', 'thunder'), urls) - if not urls: - return - #self.urlopen('http://dynamic.cloud.vip.xunlei.com/interface/batch_task_check', data={'url':'\r\n'.join(urls), 'random':current_random()}) - jsonp = 'jsonp%s' % current_timestamp() - url = 'http://dynamic.cloud.vip.xunlei.com/interface/batch_task_commit?callback=%s' % jsonp - if old_task_ids: - batch_old_taskid = ','.join(old_task_ids) - else: - batch_old_taskid = '0' + ',' * (len(urls) - 1) # XXX: what is it? - data = {} - for i in range(len(urls)): - data['cid[%d]' % i] = '' - data['url[%d]' % i] = urllib.quote(to_utf_8(urls[i])) # fix per request #98 - data['batch_old_taskid'] = batch_old_taskid - data['verify_code'] = '' - response = self.urlread(url, data=data) - - response_info = get_response_info(response, jsonp) - code = response_info['process'] - while code == -12 or code == -11: - verification_code = self.read_verification_code() - assert verification_code - data['verify_code'] = verification_code - response = self.urlread(url, data=data) - response_info = get_response_info(response, jsonp) - code = response_info['process'] - if code == len(urls): - return - else: - msg = response_info.get('msg') - assert not msg, repr(msg.decode('utf-8')) - assert code == len(urls), 'invalid response code: %s' % code - - def commit_torrent_task(self, data): - jsonp = 'jsonp%s' % current_timestamp() - commit_url = 'http://dynamic.cloud.vip.xunlei.com/interface/bt_task_commit?callback=%s' % jsonp - def commit(): - response = self.urlread(commit_url, data=data) - response_info = get_response_info(response, jsonp) - code = response_info['progress'] - while code == -12 or code == -11: - verification_code = self.read_verification_code() - assert verification_code - data['verify_code'] = verification_code - response = self.urlread(commit_url, data=data) - response_info = get_response_info(response, jsonp) - code = response_info['progress'] - return response_info - response_info = commit() - if is_dirty_resource(response_info): - data['btname'] = encode_dirty_name(data['btname']) - response_info = commit() - msg = response_info.get('msg') - assert not msg, repr(msg) - - def add_torrent_task_by_content(self, content, path='attachment.torrent'): - assert re.match(r'd\d+:', content), 'Probably not a valid content file [%s...]' % repr(content[:17]) - upload_url = 'http://dynamic.cloud.vip.xunlei.com/interface/torrent_upload' - content_type, body = encode_multipart_formdata([], [('filepath', path, content)]) - response = self.urlread(upload_url, data=body, headers={'Content-Type': content_type}).decode('utf-8') - - upload_success = re.search(r'', response, flags=re.S) - if upload_success: - bt = json.loads(upload_success.group(1)) - bt_hash = bt['infoid'] - bt_name = bt['ftitle'] - bt_size = bt['btsize'] - data = {'uid':self.id, 'btname':bt_name, 'cid':bt_hash, 'tsize':bt_size, - 'findex':''.join(f['id']+'_' for f in bt['filelist']), - 'size':''.join(f['subsize']+'_' for f in bt['filelist']), - 'from':'0'} - self.commit_torrent_task(data) - return bt_hash - already_exists = re.search(r"parent\.edit_bt_list\((\{.*\}),'','0'\)", response, flags=re.S) - if already_exists: - bt = json.loads(already_exists.group(1)) - bt_hash = bt['infoid'] - return bt_hash - raise NotImplementedError(response) - - def add_torrent_task_by_info_hash(self, sha1): - return self.add_torrent_task_by_content(self.get_torrent_file_by_info_hash(sha1), sha1.upper()+'.torrent') - - def add_torrent_task(self, path): - with open(path, 'rb') as x: - return self.add_torrent_task_by_content(x.read(), os.path.basename(path)) - - def add_torrent_task_by_info_hash2(self, sha1, old_task_id=None): - '''similar to add_torrent_task_by_info_hash, but faster. I may delete current add_torrent_task_by_info_hash completely in future''' - link = 'http://dynamic.cloud.vip.xunlei.com/interface/get_torrent?userid=%s&infoid=%s' % (self.id, sha1.upper()) - return self.add_torrent_task_by_link(link, old_task_id=old_task_id) - - def add_magnet_task(self, link): - return self.add_torrent_task_by_link(link) - - def add_torrent_task_by_link(self, link, old_task_id=None): - url = 'http://dynamic.cloud.vip.xunlei.com/interface/url_query?callback=queryUrl&u=%s&random=%s' % (urllib.quote(link), current_timestamp()) - response = self.urlread(url) - success = re.search(r'queryUrl(\(1,.*\))\s*$', response, flags=re.S) # XXX: sometimes it returns queryUrl(0,...)? - if not success: - already_exists = re.search(r"queryUrl\(-1,'([^']{40})", response, flags=re.S) - if already_exists: - return already_exists.group(1) - raise NotImplementedError(repr(response)) - args = success.group(1).decode('utf-8') - args = literal_eval(args.replace('new Array', '')) - _, cid, tsize, btname, _, names, sizes_, sizes, _, types, findexes, _, timestamp, _ = args - def toList(x): - if type(x) in (list, tuple): - return x - else: - return [x] - data = {'uid':self.id, 'btname':btname, 'cid':cid, 'tsize':tsize, - 'findex':''.join(x+'_' for x in toList(findexes)), - 'size':''.join(x+'_' for x in toList(sizes)), - 'from':'0'} - if old_task_id: - data['o_taskid'] = old_task_id - data['o_page'] = 'history' - self.commit_torrent_task(data) - return cid - - def readd_all_expired_tasks(self): - url = 'http://dynamic.cloud.vip.xunlei.com/interface/delay_once?callback=anything' - response = self.urlread(url) - - def delete_tasks_by_id(self, ids): - jsonp = 'jsonp%s' % current_timestamp() - data = {'taskids': ','.join(ids)+',', 'databases': '0,'} - url = 'http://dynamic.cloud.vip.xunlei.com/interface/task_delete?callback=%s&type=%s&noCacheIE=%s' % (jsonp, 2, current_timestamp()) # XXX: what is 'type'? - response = self.urlread(url, data=data) - response = remove_bom(response) - assert_response(response, jsonp, '{"result":1,"type":2}') - - def delete_task_by_id(self, id): - self.delete_tasks_by_id([id]) - - def delete_task(self, task): - self.delete_task_by_id(task['id']) - - def delete_tasks(self, tasks): - self.delete_tasks_by_id([t['id'] for t in tasks]) - - def pause_tasks_by_id(self, ids): - url = 'http://dynamic.cloud.vip.xunlei.com/interface/task_pause?tid=%s&uid=%s&noCacheIE=%s' % (','.join(ids)+',', self.id, current_timestamp()) - assert self.urlread(url) == 'pause_task_resp()' - - def pause_task_by_id(self, id): - self.pause_tasks_by_id([id]) - - def pause_task(self, task): - self.pause_task_by_id(task['id']) - - def pause_tasks(self, tasks): - self.pause_tasks_by_id(t['id'] for t in tasks) - - def restart_tasks(self, tasks): - jsonp = 'jsonp%s' % current_timestamp() - url = 'http://dynamic.cloud.vip.xunlei.com/interface/redownload?callback=%s' % jsonp - form = [] - for task in tasks: - assert task['type'] in ('ed2k', 'http', 'https', 'ftp', 'https', 'bt'), "'%s' is not tested" % task['type'] - data = {'id[]': task['id'], - 'cid[]': '', # XXX: should I set this? - 'url[]': task['original_url'], - 'download_status[]': task['status']} - if task['type'] == 'ed2k': - data['taskname[]'] = task['name'].encode('utf-8') # XXX: shouldn't I set this for other task types? - form.append(urlencode(data)) - form.append(urlencode({'type':1})) - data = '&'.join(form) - response = self.urlread(url, data=data) - assert_response(response, jsonp) - - def rename_task(self, task, new_name): - assert type(new_name) == unicode - url = 'http://dynamic.cloud.vip.xunlei.com/interface/rename' - taskid = task['id'] - bt = '1' if task['type'] == 'bt' else '0' - url = url+'?'+urlencode({'taskid':taskid, 'bt':bt, 'filename':new_name.encode('utf-8')}) - response = self.urlread(url) - assert '"result":0' in response, response - - def restart_task(self, task): - self.restart_tasks([task]) - - def get_task_by_id(self, id): - tasks = self.read_all_tasks(0) - for x in tasks: - if x['id'] == id: - return x - raise Exception('No task found for id '+id) - - -def current_timestamp(): - return int(time.time()*1000) - -def current_random(): - from random import randint - return '%s%06d.%s' % (current_timestamp(), randint(0, 999999), randint(100000000, 9999999999)) - -def convert_task(data): - expired = {'0':False, '4': True}[data['flag']] - assert re.match(r'[^:]+', data['url']), 'Invalid URL in: ' + repr(data) - task = {'id': data['id'], - 'type': re.match(r'[^:]+', data['url']).group().lower(), - 'name': decode_dirty_name(unescape_html(data['taskname'])), - 'status': int(data['download_status']), - 'status_text': {'0':'waiting', '1':'downloading', '2':'completed', '3':'failed', '5':'pending'}[data['download_status']], - 'expired': expired, - 'size': int(data['ysfilesize']), - 'original_url': unescape_html(data['url']), - 'xunlei_url': data['lixian_url'] or None, - 'bt_hash': data['cid'], - 'dcid': data['cid'], - 'gcid': data['gcid'], - 'date': data['dt_committed'][:10].replace('-', '.'), - 'progress': '%s%%' % data['progress'], - 'speed': '%s' % data['speed'], - } - return task - -def parse_json_response(html): - m = re.match(ur'^\ufeff?rebuild\((\{.*\})\)$', html) - if not m: - logger.trace(html) - raise RuntimeError('Invalid response') - return json.loads(m.group(1)) - -def parse_json_tasks(result): - tasks = result['info']['tasks'] - return map(convert_task, tasks) - -def parse_task(html): - inputs = re.findall(r']+/>', html) - def parse_attrs(html): - return dict((k, v1 or v2) for k, v1, v2 in re.findall(r'''\b(\w+)=(?:'([^']*)'|"([^"]*)")''', html)) - info = dict((x['id'], unescape_html(x['value'])) for x in map(parse_attrs, inputs)) - mini_info = {} - mini_map = {} - #mini_info = dict((re.sub(r'\d+$', '', k), info[k]) for k in info) - for k in info: - mini_key = re.sub(r'\d+$', '', k) - mini_info[mini_key] = info[k] - mini_map[mini_key] = k - taskid = mini_map['taskname'][8:] - url = mini_info['f_url'] - task_type = re.match(r'[^:]+', url).group().lower() - task = {'id': taskid, - 'type': task_type, - 'name': mini_info['taskname'], - 'status': int(mini_info['d_status']), - 'status_text': {'0':'waiting', '1':'downloading', '2':'completed', '3':'failed', '5':'pending'}[mini_info['d_status']], - 'size': int(mini_info.get('ysfilesize', 0)), - 'original_url': mini_info['f_url'], - 'xunlei_url': mini_info.get('dl_url', None), - 'bt_hash': mini_info['dcid'], - 'dcid': mini_info['dcid'], - 'gcid': parse_gcid(mini_info.get('dl_url', None)), - } - - m = re.search(r']*>([^<>]*)', html) - task['progress'] = m and m.group(1) or '' - m = re.search(r']*id="speed\d+">([^<>]*)', html) - task['speed'] = m and m.group(1).replace(' ', '') or '' - m = re.search(r'([^<>]*)', html) - task['date'] = m and m.group(1) or '' - - return task - -def parse_history(html): - rwbox = re.search(r'
    ', html, re.S).group() - rw_lists = re.findall(r'
    ]*/>', rwbox, re.S) - return map(parse_task, rw_lists) - -def parse_bt_list(js): - result = json.loads(re.match(r'^fill_bt_list\((.+)\)\s*$', js).group(1))['Result'] - files = [] - for record in result['Record']: - files.append({ - 'id': record['taskid'], - 'index': record['id'], - 'type': 'bt', - 'name': record['title'], # TODO: support folder - 'status': int(record['download_status']), - 'status_text': {'0':'waiting', '1':'downloading', '2':'completed', '3':'failed', '5':'pending'}[record['download_status']], - 'size': int(record['filesize']), - 'original_url': record['url'], - 'xunlei_url': record['downurl'], - 'dcid': record['cid'], - 'gcid': parse_gcid(record['downurl']), - 'speed': '', - 'progress': '%s%%' % record['percent'], - 'date': '', - }) - return files - -def parse_gcid(url): - if not url: - return - m = re.search(r'&g=([A-F0-9]{40})&', url) - if not m: - return - return m.group(1) - -def urlencode(x): - def unif8(u): - if type(u) == unicode: - u = u.encode('utf-8') - return u - return urllib.urlencode([(unif8(k), unif8(v)) for k, v in x.items()]) - -def encode_multipart_formdata(fields, files): - #http://code.activestate.com/recipes/146306/ - """ - fields is a sequence of (name, value) elements for regular form fields. - files is a sequence of (name, filename, value) elements for data to be uploaded as files - Return (content_type, body) ready for httplib.HTTP instance - """ - BOUNDARY = '----------ThIs_Is_tHe_bouNdaRY_$' - CRLF = '\r\n' - L = [] - for (key, value) in fields: - L.append('--' + BOUNDARY) - L.append('Content-Disposition: form-data; name="%s"' % key) - L.append('') - L.append(value) - for (key, filename, value) in files: - L.append('--' + BOUNDARY) - L.append('Content-Disposition: form-data; name="%s"; filename="%s"' % (key, filename)) - L.append('Content-Type: %s' % get_content_type(filename)) - L.append('') - L.append(value) - L.append('--' + BOUNDARY + '--') - L.append('') - body = CRLF.join(L) - content_type = 'multipart/form-data; boundary=%s' % BOUNDARY - return content_type, body - -def get_content_type(filename): - import mimetypes - return mimetypes.guess_type(filename)[0] or 'application/octet-stream' - -def assert_default_page(response, id): - #assert response == "" % id - assert re.match(r"^$" % id, response), response - -def remove_bom(response): - if response.startswith('\xef\xbb\xbf'): - response = response[3:] - return response - -def assert_response(response, jsonp, value=1): - response = remove_bom(response) - assert response == '%s(%s)' % (jsonp, value), repr(response) - -def get_response_info(response, jsonp): - response = remove_bom(response) - m = re.match(r'^%s\((.+)\)$' % jsonp, response) - assert m, 'invalid jsonp response: %s' % response - logger.trace('get_response_info') - logger.trace(response) - parameter = m.group(1) - m = re.match(r"^\{process:(-?\d+),msg:'(.*)'\}$", parameter) - if m: - return {'process': int(m.group(1)), 'msg': m.group(2)} - return json.loads(parameter) - -def parse_url_protocol(url): - m = re.match(r'([^:]+)://', url) - if m: - return m.group(1) - elif url.startswith('magnet:'): - return 'magnet' - else: - return url - -def unescape_html(html): - import xml.sax.saxutils - return xml.sax.saxutils.unescape(html) - -def to_utf_8(s): - if type(s) == unicode: - return s.encode('utf-8') - else: - return s - -def md5(s): - import hashlib - return hashlib.md5(s).hexdigest().lower() - -def encypt_password(password): - if not re.match(r'^[0-9a-f]{32}$', password): - password = md5(md5(password)) - return password - -def ungzip(s): - from StringIO import StringIO - import gzip - buffer = StringIO(s) - f = gzip.GzipFile(fileobj=buffer) - return f.read() - -def undeflate(s): - import zlib - return zlib.decompress(s, -zlib.MAX_WBITS) - -def is_dirty_resource(response_info): - return response_info['progress'] == 2 and response_info.get('rtcode') == '76' and response_info.get('msg') == u"\u6587\u4ef6\u540d\u4e2d\u5305\u542b\u8fdd\u89c4\u5185\u5bb9\uff0c\u65e0\u6cd5\u6dfb\u52a0\u5230\u79bb\u7ebf\u7a7a\u95f4[0976]" - -def encode_dirty_name(x): - import base64 - try: - return unicode('[base64]' + base64.encodestring(x.encode('utf-8')).replace('\n', '')) - except: - return x - -def decode_dirty_name(x): - import base64 - try: - if x.startswith('[base64]'): - return base64.decodestring(x[len('[base64]'):]).decode('utf-8') - else: - return x - except: - return x - diff --git a/xunlei-lixian/lixian_alias.py b/xunlei-lixian/lixian_alias.py deleted file mode 100644 index 3c7a34a..0000000 --- a/xunlei-lixian/lixian_alias.py +++ /dev/null @@ -1,20 +0,0 @@ - - -__all__ = ['register_alias', 'to_alias'] - -aliases = {'d': 'download', 'l': 'list', 'a': 'add', 'x': 'delete'} - -def register_alias(alias, command): - aliases[alias] = command - -def get_aliases(): - return aliases - -def get_alias(a): - aliases = get_aliases() - if a in aliases: - return aliases[a] - -def to_alias(a): - return get_alias(a) or a - diff --git a/xunlei-lixian/lixian_batch.py b/xunlei-lixian/lixian_batch.py deleted file mode 100755 index fadeba0..0000000 --- a/xunlei-lixian/lixian_batch.py +++ /dev/null @@ -1,15 +0,0 @@ -#!/usr/bin/env python - -import sys -import os.path -import lixian_cli - -def download_batch(files): - for f in map(os.path.abspath, files): - print 'Downloading', f, '...' - os.chdir(os.path.dirname(f)) - lixian_cli.execute_command(['download', '--input', f, '--delete', '--continue']) - -if __name__ == '__main__': - download_batch(sys.argv[1:]) - diff --git a/xunlei-lixian/lixian_cli.py b/xunlei-lixian/lixian_cli.py deleted file mode 100755 index b495214..0000000 --- a/xunlei-lixian/lixian_cli.py +++ /dev/null @@ -1,65 +0,0 @@ -#!/usr/bin/env python - -from lixian_commands.util import * -import lixian_help -import sys - -from lixian_commands.login import login -from lixian_commands.logout import logout -from lixian_commands.download import download_task -from lixian_commands.list import list_task -from lixian_commands.add import add_task -from lixian_commands.delete import delete_task -from lixian_commands.pause import pause_task -from lixian_commands.restart import restart_task -from lixian_commands.rename import rename_task -from lixian_commands.readd import readd_task -from lixian_commands.info import lixian_info -from lixian_commands.config import lx_config -from lixian_commands.help import lx_help - - -def execute_command(args=sys.argv[1:]): - import lixian_plugins # load plugins at import - if not args: - usage() - sys.exit(1) - command = args[0] - if command.startswith('-'): - if command in ('-h', '--help'): - usage(lixian_help.welcome_help) - elif command in ('-v', '--version'): - print '0.0.x' - else: - usage() - sys.exit(1) - sys.exit(0) - import lixian_alias - command = lixian_alias.to_alias(command) - commands = {'login': login, - 'logout': logout, - 'download': download_task, - 'list': list_task, - 'add': add_task, - 'delete': delete_task, - 'pause': pause_task, - 'restart': restart_task, - 'rename': rename_task, - 'readd': readd_task, - 'info': lixian_info, - 'config': lx_config, - 'help': lx_help} - import lixian_plugins.commands - commands.update(lixian_plugins.commands.commands) - if command not in commands: - usage() - sys.exit(1) - if '-h' in args or '--help' in args: - lx_help([command]) - else: - commands[command](args[1:]) - -if __name__ == '__main__': - execute_command() - - diff --git a/xunlei-lixian/lixian_cli_parser.py b/xunlei-lixian/lixian_cli_parser.py deleted file mode 100644 index 4e4f02a..0000000 --- a/xunlei-lixian/lixian_cli_parser.py +++ /dev/null @@ -1,178 +0,0 @@ - -__all__ = ['expand_command_line', 'parse_command_line', 'Parser', 'command_line_parse', 'command_line_option', 'command_line_value', 'command_line_parser', 'with_parser'] - -def expand_windows_command_line(args): - from glob import glob - expanded = [] - for x in args: - try: - xx = glob(x) - except: - xx = None - if xx: - expanded += xx - else: - expanded.append(x) - return expanded - -def expand_command_line(args): - import platform - return expand_windows_command_line(args) if platform.system() == 'Windows' else args - -def parse_command_line(args, keys=[], bools=[], alias={}, default={}, help=None): - args = expand_command_line(args) - options = {} - for k in keys: - options[k] = None - for k in bools: - options[k] = None - left = [] - args = args[:] - while args: - x = args.pop(0) - if x == '--': - left.extend(args) - break - if x.startswith('-') and len(x) > 1: - k = x.lstrip('-') - if k in bools: - options[k] = True - elif k.startswith('no-') and k[3:] in bools: - options[k[3:]] = False - elif k in keys: - options[k] = args.pop(0) - elif '=' in k and k[:k.index('=')] in keys: - options[k[:k.index('=')]] = k[k.index('=')+1:] - elif k in alias: - k = alias[k] - if k in bools: - options[k] = True - else: - options[k] = args.pop(0) - elif '=' in k and k[:k.index('=')] in alias: - k, v = k[:k.index('=')], k[k.index('=')+1:] - k = alias[k] - if k not in keys: - raise RuntimeError('Invalid boolean option '+x) - options[k] = v - else: - if help: - print 'Unknown option ' + x - print - print help - exit(1) - else: - raise RuntimeError('Unknown option '+x) - else: - left.append(x) - - for k in default: - if options[k] is None: - options[k] = default[k] - - class Args(object): - def __init__(self, args, left): - self.__dict__['_args'] = args - self.__dict__['_left'] = left - def __getattr__(self, k): - v = self._args.get(k, None) - if v: - return v - if '_' in k: - return self._args.get(k.replace('_', '-'), None) - def __setattr__(self, k, v): - self._args[k] = v - def __getitem__(self, i): - if type(i) == int: - return self._left[i] - else: - return self._args[i] - def __setitem__(self, i, v): - if type(i) == int: - self._left[i] = v - else: - self._args[i] = v - def __len__(self): - return len(self._left) - def __str__(self): - return '' % (self._args, self._left) - return Args(options, left) - -class Stack: - def __init__(self, **args): - self.__dict__.update(args) - -class Parser: - def __init__(self): - self.stack = [] - def with_parser(self, parser): - self.stack.append(parser) - return self - def __call__(self, args, keys=[], bools=[], alias={}, default={}, help=None): - stack = Stack(keys=list(keys), bools=list(bools), alias=dict(alias), default=dict(default)) - keys = [] - bools = [] - alias = {} - default = {} - for stack in [x.args_stack for x in self.stack] + [stack]: - keys += stack.keys - bools += stack.bools - alias.update(stack.alias) - default.update(stack.default) - args = parse_command_line(args, keys=keys, bools=bools, alias=alias, default=default, help=help) - for fn in self.stack: - new_args = fn(args) - if new_args: - args = new_args - return args - -def command_line_parse(keys=[], bools=[], alias={}, default={}): - def wrapper(fn): - if hasattr(fn, 'args_stack'): - stack = fn.args_stack - stack.keys += keys - stack.bools += bools - stack.alias.update(alias) - stack.default.update(default) - else: - fn.args_stack = Stack(keys=list(keys), bools=list(bools), alias=dict(alias), default=dict(default)) - return fn - return wrapper - -def command_line_option(name, alias=None, default=None): - alias = {alias:name} if alias else {} - default = {name:default} if default is not None else {} - return command_line_parse(bools=[name], alias=alias, default=default) - -def command_line_value(name, alias=None, default=None): - alias = {alias:name} if alias else {} - default = {name:default} if default else {} - return command_line_parse(keys=[name], alias=alias, default=default) - -def command_line_parser(*args, **kwargs): - def wrapper(f): - parser = Parser() - for x in reversed(getattr(f, 'args_parsers', [])): - parser = parser.with_parser(x) - if hasattr(f, 'args_stack'): - def parse_no_body(args): - pass - parse_no_body.args_stack = f.args_stack - parser = parser.with_parser(parse_no_body) - import functools - @functools.wraps(f) - def parse(args_list): - return f(parser(args_list, *args, **kwargs)) - return parse - return wrapper - -def with_parser(parser): - def wrapper(f): - if hasattr(f, 'args_parsers'): - f.args_parsers.append(parser) - else: - f.args_parsers = [parser] - return f - return wrapper - - diff --git a/xunlei-lixian/lixian_colors.py b/xunlei-lixian/lixian_colors.py deleted file mode 100644 index 2c25ccb..0000000 --- a/xunlei-lixian/lixian_colors.py +++ /dev/null @@ -1,70 +0,0 @@ - -import os -import sys - -def get_console_type(use_colors=True): - if use_colors and sys.stdout.isatty() and sys.stderr.isatty(): - import platform - if platform.system() == 'Windows': - import lixian_colors_win32 - return lixian_colors_win32.WinConsole - else: - import lixian_colors_linux - return lixian_colors_linux.AnsiConsole - else: - import lixian_colors_console - return lixian_colors_console.Console - -console_type = get_console_type() -raw_console_type = get_console_type(False) - -def Console(use_colors=True): - return get_console_type(use_colors)() - -def get_softspace(output): - if hasattr(output, 'softspace'): - return output.softspace - import lixian_colors_console - if isinstance(output, lixian_colors_console.Console): - return get_softspace(output.output) - return 0 - -class ScopedColors(console_type): - def __init__(self, *args): - console_type.__init__(self, *args) - def __call__(self): - console = self - class Scoped: - def __enter__(self): - self.stdout = sys.stdout - softspace = get_softspace(sys.stdout) - sys.stdout = console - sys.stdout.softspace = softspace - def __exit__(self, type, value, traceback): - softspace = get_softspace(sys.stdout) - sys.stdout = self.stdout - sys.stdout.softspace = softspace - return Scoped() - -class RawScopedColors(raw_console_type): - def __init__(self, *args): - raw_console_type.__init__(self, *args) - def __call__(self): - class Scoped: - def __enter__(self): - pass - def __exit__(self, type, value, traceback): - pass - return Scoped() - -class RootColors: - def __init__(self, use_colors=True): - self.use_colors = use_colors - def __getattr__(self, name): - return getattr(ScopedColors() if self.use_colors else RawScopedColors(), name) - def __call__(self, use_colors): - assert use_colors in (True, False, None), use_colors - return RootColors(use_colors) - -colors = RootColors() - diff --git a/xunlei-lixian/lixian_colors_console.py b/xunlei-lixian/lixian_colors_console.py deleted file mode 100644 index 258d75f..0000000 --- a/xunlei-lixian/lixian_colors_console.py +++ /dev/null @@ -1,46 +0,0 @@ - -__all__ = ['Console'] - -import sys - -styles = [ - 'black', - 'blue', - 'green', - 'red', - 'cyan', - 'yellow', - 'purple', - 'white', - - 'bold', - 'italic', - 'underline', - 'inverse', -] - - -class Console: - def __init__(self, output=None, styles=[]): - output = output or sys.stdout - if isinstance(output, Console): - self.output = output.output - self.styles = output.styles + styles - else: - self.output = output - self.styles = styles - assert not isinstance(self.output, Console) - def __getattr__(self, name): - if name in styles: - return self.ansi(name) - else: - raise AttributeError(name) - def ansi(self, code): - return self.__class__(self.output, self.styles + [code]) if code not in (None, '') else self - def __call__(self, s): - self.write(s) - def write(self, s): - self.output.write(s) - def flush(self, *args): - self.output.flush(*args) - diff --git a/xunlei-lixian/lixian_colors_linux.py b/xunlei-lixian/lixian_colors_linux.py deleted file mode 100644 index c224dcd..0000000 --- a/xunlei-lixian/lixian_colors_linux.py +++ /dev/null @@ -1,62 +0,0 @@ - -__all__ = ['AnsiConsole'] - -from lixian_colors_console import Console - -import sys - -colors = { - 'bold' : [1, 22], - 'italic' : [3, 23], - 'underline' : [4, 24], - 'inverse' : [7, 27], - 'white' : [37, 39], - 'grey' : [90, 39], - 'black' : [30, 39], - 'blue' : [34, 39], - 'cyan' : [36, 39], - 'green' : [32, 39], - 'purple' : [35, 39], - 'magenta' : [35, 39], - 'red' : [31, 39], - 'yellow' : [33, 39] -} - -class Render: - def __init__(self, output, code): - self.output = output - self.left, self.right = code - def __enter__(self): - self.output.write(self.left) - self.output.flush() - def __exit__(self, type, value, traceback): - self.output.write(self.right) - self.output.flush() - -def mix_styles(styles): - left = [] - right = [] - for style in styles: - if style in colors: - color = colors[style] - left.append(color[0]) - right.append(color[1]) - right.reverse() - return [''.join('\033[%dm' % n for n in left), ''.join('\033[%dm' % n for n in right)] - -class AnsiConsole(Console): - def __init__(self, output=None, styles=[]): - Console.__init__(self, output, styles) - - def write(self, s): - if self.styles: - with self.render(mix_styles(self.styles)): - self.output.write(s) - self.output.flush() - else: - self.output.write(s) - self.output.flush() - - def render(self, code): - return Render(self.output, code) - diff --git a/xunlei-lixian/lixian_colors_win32.py b/xunlei-lixian/lixian_colors_win32.py deleted file mode 100644 index 0e7e780..0000000 --- a/xunlei-lixian/lixian_colors_win32.py +++ /dev/null @@ -1,201 +0,0 @@ - -__all__ = ['WinConsole'] - -from lixian_colors_console import Console - -import ctypes -from ctypes import windll, byref, Structure -from ctypes.wintypes import SHORT, WORD - -import sys - -INVALID_HANDLE_VALUE = -1 -STD_OUTPUT_HANDLE = -11 -STD_ERROR_HANDLE = -12 - -class COORD(Structure): - _fields_ = (('X', SHORT), - ('Y', SHORT),) - -class SMALL_RECT(Structure): - _fields_ = (('Left', SHORT), - ('Top', SHORT), - ('Right', SHORT), - ('Bottom', SHORT),) - -class CONSOLE_SCREEN_BUFFER_INFO(Structure): - _fields_ = (('dwSize', COORD), - ('dwCursorPosition', COORD), - ('wAttributes', WORD), - ('srWindow', SMALL_RECT), - ('dwMaximumWindowSize', COORD),) - - -def GetWinError(): - code = ctypes.GetLastError() - message = ctypes.FormatError(code) - return '[Error %s] %s' % (code, message) - -def GetStdHandle(handle): - h = windll.kernel32.GetStdHandle(handle) - if h == INVALID_HANDLE_VALUE: - raise OSError(GetWinError()) - return h - -def GetConsoleScreenBufferInfo(handle): - info = CONSOLE_SCREEN_BUFFER_INFO() - if not windll.kernel32.GetConsoleScreenBufferInfo(handle, byref(info)): - raise OSError(GetWinError()) - return info - -def SetConsoleTextAttribute(handle, attributes): - if not windll.Kernel32.SetConsoleTextAttribute(handle, attributes): - raise OSError(GetWinError()) - - -FOREGROUND_BLUE = 0x0001 -FOREGROUND_GREEN = 0x0002 -FOREGROUND_RED = 0x0004 -FOREGROUND_INTENSITY = 0x0008 -BACKGROUND_BLUE = 0x0010 -BACKGROUND_GREEN = 0x0020 -BACKGROUND_RED = 0x0040 -BACKGROUND_INTENSITY = 0x0080 -COMMON_LVB_LEADING_BYTE = 0x0100 -COMMON_LVB_TRAILING_BYTE = 0x0200 -COMMON_LVB_GRID_HORIZONTAL = 0x0400 -COMMON_LVB_GRID_LVERTICAL = 0x0800 -COMMON_LVB_GRID_RVERTICAL = 0x1000 -COMMON_LVB_REVERSE_VIDEO = 0x4000 -COMMON_LVB_UNDERSCORE = 0x8000 - -colors = { - 'black' : 0b000, - 'blue' : 0b001, - 'green' : 0b010, - 'red' : 0b100, - 'cyan' : 0b011, - 'yellow' : 0b110, - 'purple' : 0b101, - 'magenta': 0b101, - 'white' : 0b111, -} - -def mix_styles(styles, attributes): - fg_color = -1 - bg_color = -1 - fg_bright = -1 - bg_bright = -1 - reverse = -1 - underscore = -1 - for style in styles: - if style == 0: - # reset mode - raise NotImplementedError() - elif style == 1: - # foreground bright on - fg_bright = 1 - elif style == 2: - # both bright off - fg_bright = 0 - bg_bright = 0 - elif style == 4 or style == 'underline': - # Underscore - underscore = 1 - elif style == 5: - # background bright on - bg_bright = 1 - elif style == 7 or style == 'inverse': - # Reverse foreground and background attributes. - reverse = 1 - elif style == 21 or style == 22: - # foreground bright off - fg_bright = 0 - elif style == 24: - # Underscore: no - underscore = 0 - elif style == 25: - # background bright off - bg_bright = 0 - elif style == 27: - # Reverse: no - reverse = 0 - elif 30 <= style <= 37: - # set foreground color - fg_color = style - 30 - elif style == 39: - # default text color - fg_color = 7 - fg_bright = 0 - elif 40 <= style <= 47: - # set background color - bg_color = style - 40 - elif style == 49: - # default background color - bg_color = 0 - elif 90 <= style <= 97: - # set bold foreground color - fg_bright = 1 - fg_color = style - 90 - elif 100 <= style <= 107: - # set bold background color - bg_bright = 1 - bg_color = style - 100 - elif style == 'bold': - fg_bright = 1 - elif style in colors: - fg_color = colors[style] - - if fg_color != -1: - attributes &= ~ 0b111 - attributes |= fg_color - if fg_bright != -1: - attributes &= ~ 0b1000 - attributes |= fg_bright << 3 - if bg_color != -1: - attributes &= ~ 0b1110000 - attributes |= bg_color << 4 - if bg_bright != -1: - attributes &= ~ 0b10000000 - attributes |= bg_bright << 7 - if reverse != -1: - attributes &= ~ COMMON_LVB_REVERSE_VIDEO - attributes |= reverse << 14 - # XXX: COMMON_LVB_REVERSE_VIDEO doesn't work... - if reverse: - attributes = (attributes & ~(0b11111111 | COMMON_LVB_REVERSE_VIDEO)) | ((attributes & 0b11110000) >> 4) | ((attributes & 0b1111) << 4) - if underscore != -1: - attributes &= ~ COMMON_LVB_UNDERSCORE - attributes |= underscore << 15 - - return attributes - -class Render: - def __init__(self, handle, default, attributes): - self.handle = handle - self.default = default - self.attributes = attributes - def __enter__(self): - SetConsoleTextAttribute(self.handle, self.attributes) - def __exit__(self, type, value, traceback): - SetConsoleTextAttribute(self.handle, self.default) - -class WinConsole(Console): - def __init__(self, output=None, styles=[], handle=STD_OUTPUT_HANDLE): - Console.__init__(self, output, styles) - self.handle = GetStdHandle(handle) - self.default = GetConsoleScreenBufferInfo(self.handle).wAttributes - - def write(self, s): - if self.styles: - with self.render(mix_styles(self.styles, self.default)): - self.output.write(s) - self.output.flush() - else: - self.output.write(s) - self.output.flush() - - def render(self, attributes): - return Render(self.handle, self.default, attributes) - - diff --git a/xunlei-lixian/lixian_commands/__init__.py b/xunlei-lixian/lixian_commands/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/xunlei-lixian/lixian_commands/add.py b/xunlei-lixian/lixian_commands/add.py deleted file mode 100644 index b7d152a..0000000 --- a/xunlei-lixian/lixian_commands/add.py +++ /dev/null @@ -1,27 +0,0 @@ - -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_config import get_config -import lixian_help -import lixian_query - -@command_line_parser(help=lixian_help.add) -@with_parser(parse_login) -@with_parser(parse_colors) -@with_parser(parse_logging) -@with_parser(parse_size) -@command_line_value('limit', default=get_config('limit')) -@command_line_value('page-size', default=get_config('page-size')) -@command_line_value('input', alias='i') -@command_line_option('torrent', alias='bt') -def add_task(args): - assert len(args) or args.input - client = create_client(args) - tasks = lixian_query.find_tasks_to_download(client, args) - print 'All tasks added. Checking status...' - columns = ['id', 'status', 'name'] - if get_config('n'): - columns.insert(0, 'n') - if args.size: - columns.append('size') - output_tasks(tasks, columns, args) diff --git a/xunlei-lixian/lixian_commands/config.py b/xunlei-lixian/lixian_commands/config.py deleted file mode 100644 index 8c5822b..0000000 --- a/xunlei-lixian/lixian_commands/config.py +++ /dev/null @@ -1,36 +0,0 @@ - - -from lixian import encypt_password -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_config import * -import lixian_help -from getpass import getpass - -@command_line_parser(help=lixian_help.config) -@command_line_option('print') -@command_line_option('delete') -def lx_config(args): - if args.delete: - assert len(args) == 1 - delete_config(args[0]) - elif args['print'] or not len(args): - if len(args): - assert len(args) == 1 - print get_config(args[0]) - else: - print 'Loading', global_config.path, '...\n' - print source_config() - print global_config - else: - assert len(args) in (1, 2) - if args[0] == 'password': - if len(args) == 1 or args[1] == '-': - password = getpass('Password: ') - else: - password = args[1] - print 'Saving password (encrypted) to', global_config.path - put_config('password', encypt_password(password)) - else: - print 'Saving configuration to', global_config.path - put_config(*args) diff --git a/xunlei-lixian/lixian_commands/delete.py b/xunlei-lixian/lixian_commands/delete.py deleted file mode 100644 index 3e74719..0000000 --- a/xunlei-lixian/lixian_commands/delete.py +++ /dev/null @@ -1,38 +0,0 @@ - -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_config import get_config -from lixian_encoding import default_encoding -from lixian_colors import colors -import lixian_help -import lixian_query - -@command_line_parser(help=lixian_help.delete) -@with_parser(parse_login) -@with_parser(parse_colors) -@with_parser(parse_logging) -@command_line_option('i') -@command_line_option('all') -@command_line_option('failed') -@command_line_value('limit', default=get_config('limit')) -@command_line_value('page-size', default=get_config('page-size')) -def delete_task(args): - client = create_client(args) - to_delete = lixian_query.search_tasks(client, args) - if not to_delete: - print 'Nothing to delete' - return - with colors(args.colors).red.bold(): - print "Below files are going to be deleted:" - for x in to_delete: - print x['name'].encode(default_encoding) - if args.i: - yes_or_no = raw_input('Are your sure to delete them from Xunlei cloud? (y/n) ') - while yes_or_no.lower() not in ('y', 'yes', 'n', 'no'): - yes_or_no = raw_input('yes or no? ') - if yes_or_no.lower() in ('y', 'yes'): - pass - elif yes_or_no.lower() in ('n', 'no'): - print 'Deletion abort per user request.' - return - client.delete_tasks(to_delete) diff --git a/xunlei-lixian/lixian_commands/download.py b/xunlei-lixian/lixian_commands/download.py deleted file mode 100644 index 1e2127c..0000000 --- a/xunlei-lixian/lixian_commands/download.py +++ /dev/null @@ -1,325 +0,0 @@ - -import lixian_download_tools -import lixian_nodes -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_config import * -from lixian_encoding import default_encoding -from lixian_colors import colors -import lixian_help -import lixian_query -import lixian_hash -import lixian_hash_bt -import lixian_hash_ed2k - -import os -import os.path -import re - -def ensure_dir_exists(dirname): - if dirname and not os.path.exists(dirname): - try: - os.makedirs(dirname) - except os.error: - if not os.path.exists(dirname): - raise - -def escape_filename(name): - amp = re.compile(r'&(amp;)+', flags=re.I) - name = re.sub(amp, '&', name) - name = re.sub(r'[\\/:*?"<>|]', '-', name) - return name - -def safe_encode_native_path(path): - return path.encode(default_encoding).decode(default_encoding).replace('?', '-').encode(default_encoding) - -def verify_basic_hash(path, task): - if os.path.getsize(path) != task['size']: - print 'hash error: incorrect file size (%s != %s)' % (os.path.getsize(path), task['size']) - return False - return lixian_hash.verify_dcid(path, task['dcid']) - -def verify_hash(path, task): - if verify_basic_hash(path, task): - if task['type'] == 'ed2k': - return lixian_hash_ed2k.verify_ed2k_link(path, task['original_url']) - else: - return True - -def verify_mini_hash(path, task): - return os.path.exists(path) and os.path.getsize(path) == task['size'] and lixian_hash.verify_dcid(path, task['dcid']) - -def verify_mini_bt_hash(dirname, files): - for f in files: - name = f['name'].encode(default_encoding) - path = os.path.join(dirname, *name.split('\\')) - if not verify_mini_hash(path, f): - return False - return True - -def download_file(client, path, task, options): - download_tool = lixian_download_tools.get_tool(options['tool']) - - resuming = options.get('resuming') - overwrite = options.get('overwrite') - mini_hash = options.get('mini_hash') - no_hash = options.get('no_hash') - - url = str(task['xunlei_url']) - if options['node']: - if options['node'] == 'best' or options['node'] == 'fastest': - from lixian_util import parse_size - if task['size'] >= parse_size(options['node_detection_threshold']): - url = lixian_nodes.use_fastest_node(url, options['vod_nodes'], client.get_gdriveid()) - elif options['node'] == 'fast': - from lixian_util import parse_size - if task['size'] >= parse_size(options['node_detection_threshold']): - url = lixian_nodes.use_fast_node(url, options['vod_nodes'], parse_size(options['node_detection_acceptable']), client.get_gdriveid()) - else: - url = lixian_nodes.switch_node(url, options['node'], client.get_gdriveid()) - - def download1(download, path): - if not os.path.exists(path): - download() - elif not resuming: - if overwrite: - download() - else: - raise Exception('%s already exists. Please try --continue or --overwrite' % path) - else: - if download.finished(): - pass - else: - download() - - def download1_checked(client, url, path, size): - download = download_tool(client=client, url=url, path=path, size=size, resuming=resuming) - checked = 0 - while checked < 10: - download1(download, path) - if download.finished(): - break - else: - checked += 1 - assert os.path.getsize(path) == size, 'incorrect downloaded file size (%s != %s)' % (os.path.getsize(path), size) - - def download2(client, url, path, task): - size = task['size'] - if mini_hash and resuming and verify_mini_hash(path, task): - return - download1_checked(client, url, path, size) - verify = verify_basic_hash if no_hash else verify_hash - if not verify(path, task): - with colors(options.get('colors')).yellow(): - print 'hash error, redownloading...' - os.rename(path, path + '.error') - download1_checked(client, url, path, size) - if not verify(path, task): - raise Exception('hash check failed') - - download2(client, url, path, task) - - -def download_single_task(client, task, options): - output = options.get('output') - output = output and os.path.expanduser(output) - output_dir = options.get('output_dir') - output_dir = output_dir and os.path.expanduser(output_dir) - delete = options.get('delete') - resuming = options.get('resuming') - overwrite = options.get('overwrite') - mini_hash = options.get('mini_hash') - no_hash = options.get('no_hash') - no_bt_dir = options.get('no_bt_dir') - save_torrent_file = options.get('save_torrent_file') - - assert client.get_gdriveid() - if task['status_text'] != 'completed': - if 'files' not in task: - with colors(options.get('colors')).yellow(): - print 'skip task %s as the status is %s' % (task['name'].encode(default_encoding), task['status_text']) - return - - if output: - output_path = output - output_dir = os.path.dirname(output) - output_name = os.path.basename(output) - else: - output_name = safe_encode_native_path(escape_filename(task['name'])) - output_dir = output_dir or '.' - output_path = os.path.join(output_dir, output_name) - - if task['type'] == 'bt': - files, skipped, single_file = lixian_query.expand_bt_sub_tasks(task) - if single_file: - dirname = output_dir - else: - if no_bt_dir: - output_path = os.path.dirname(output_path) - dirname = output_path - assert dirname # dirname must be non-empty, otherwise dirname + os.path.sep + ... might be dangerous - ensure_dir_exists(dirname) - for t in skipped: - with colors(options.get('colors')).yellow(): - print 'skip task %s/%s (%s) as the status is %s' % (str(t['id']), t['index'], t['name'].encode(default_encoding), t['status_text']) - if mini_hash and resuming and verify_mini_bt_hash(dirname, files): - print task['name'].encode(default_encoding), 'is already done' - if delete and 'files' not in task: - client.delete_task(task) - return - if not single_file: - with colors(options.get('colors')).green(): - print output_name + '/' - for f in files: - name = f['name'] - if f['status_text'] != 'completed': - print 'Skipped %s file %s ...' % (f['status_text'], name.encode(default_encoding)) - continue - if not single_file: - print name.encode(default_encoding), '...' - else: - with colors(options.get('colors')).green(): - print name.encode(default_encoding), '...' - # XXX: if file name is escaped, hashing bt won't get correct file - splitted_path = map(escape_filename, name.split('\\')) - name = safe_encode_native_path(os.path.join(*splitted_path)) - path = dirname + os.path.sep + name # fix issue #82 - if splitted_path[:-1]: - subdir = safe_encode_native_path(os.path.join(*splitted_path[:-1])) - subdir = dirname + os.path.sep + subdir # fix issue #82 - ensure_dir_exists(subdir) - download_file(client, path, f, options) - if save_torrent_file: - info_hash = str(task['bt_hash']) - if single_file: - torrent = os.path.join(dirname, escape_filename(task['name']).encode(default_encoding) + '.torrent') - else: - torrent = os.path.join(dirname, info_hash + '.torrent') - if os.path.exists(torrent): - pass - else: - content = client.get_torrent_file_by_info_hash(info_hash) - with open(torrent, 'wb') as ouput_stream: - ouput_stream.write(content) - if not no_hash: - torrent_file = client.get_torrent_file(task) - print 'Hashing bt ...' - from lixian_progress import SimpleProgressBar - bar = SimpleProgressBar() - file_set = [f['name'].encode('utf-8').split('\\') for f in files] if 'files' in task else None - verified = lixian_hash_bt.verify_bt(output_path, lixian_hash_bt.bdecode(torrent_file)['info'], file_set=file_set, progress_callback=bar.update) - bar.done() - if not verified: - # note that we don't delete bt download folder if hash failed - raise Exception('bt hash check failed') - else: - ensure_dir_exists(output_dir) - - with colors(options.get('colors')).green(): - print output_name, '...' - download_file(client, output_path, task, options) - - if delete and 'files' not in task: - client.delete_task(task) - -def download_multiple_tasks(client, tasks, options): - for task in tasks: - download_single_task(client, task, options) - skipped = filter(lambda t: t['status_text'] != 'completed', tasks) - if skipped: - with colors(options.get('colors')).yellow(): - print "Below tasks were skipped as they were not ready:" - for task in skipped: - print task['id'], task['status_text'], task['name'].encode(default_encoding) - -@command_line_parser(help=lixian_help.download) -@with_parser(parse_login) -@with_parser(parse_colors) -@with_parser(parse_logging) -@command_line_value('tool', default=get_config('tool', 'wget')) -@command_line_value('input', alias='i') -@command_line_value('output', alias='o') -@command_line_value('output-dir', default=get_config('output-dir')) -@command_line_option('torrent', alias='bt') -@command_line_option('all') -@command_line_value('category') -@command_line_value('limit', default=get_config('limit')) -@command_line_value('page-size', default=get_config('page-size')) -@command_line_option('delete', default=get_config('delete')) -@command_line_option('continue', alias='c', default=get_config('continue')) -@command_line_option('overwrite') -@command_line_option('mini-hash', default=get_config('mini-hash')) -@command_line_option('hash', default=get_config('hash', True)) -@command_line_option('bt-dir', default=True) -@command_line_option('save-torrent-file') -@command_line_option('watch') -@command_line_option('watch-present') -@command_line_value('watch-interval', default=get_config('watch-interval', '3m')) -@command_line_value('node', default=get_config('node')) -@command_line_value('node-detection-threshold', default=get_config('node-detection-threshold', '100M')) -@command_line_value('node-detection-acceptable', default=get_config('node-detection-acceptable', '1M')) -@command_line_value('vod-nodes', default=get_config('vod-nodes', lixian_nodes.VOD_RANGE)) -def download_task(args): - assert len(args) or args.input or args.all or args.category, 'Not enough arguments' - lixian_download_tools.get_tool(args.tool) # check tool - download_args = {'tool': args.tool, - 'output': args.output, - 'output_dir': args.output_dir, - 'delete': args.delete, - 'resuming': args._args['continue'], - 'overwrite': args.overwrite, - 'mini_hash': args.mini_hash, - 'no_hash': not args.hash, - 'no_bt_dir': not args.bt_dir, - 'save_torrent_file': args.save_torrent_file, - 'node': args.node, - 'node_detection_threshold': args.node_detection_threshold, - 'node_detection_acceptable': args.node_detection_acceptable, - 'vod_nodes': args.vod_nodes, - 'colors': args.colors} - client = create_client(args) - query = lixian_query.build_query(client, args) - query.query_once() - - def sleep(n): - assert isinstance(n, (int, basestring)), repr(n) - import time - if isinstance(n, basestring): - n, u = re.match(r'^(\d+)([smh])?$', n.lower()).groups() - n = int(n) * {None: 1, 's': 1, 'm': 60, 'h': 3600}[u] - time.sleep(n) - - if args.watch_present: - assert not args.output, 'not supported with watch option yet' - tasks = query.pull_completed() - while True: - if tasks: - download_multiple_tasks(client, tasks, download_args) - if not query.download_jobs: - break - if not tasks: - sleep(args.watch_interval) - query.refresh_status() - tasks = query.pull_completed() - - elif args.watch: - assert not args.output, 'not supported with watch option yet' - tasks = query.pull_completed() - while True: - if tasks: - download_multiple_tasks(client, tasks, download_args) - if (not query.download_jobs) and (not query.queries): - break - if not tasks: - sleep(args.watch_interval) - query.refresh_status() - query.query_search() - tasks = query.pull_completed() - - else: - tasks = query.peek_download_jobs() - if args.output: - assert len(tasks) == 1 - download_single_task(client, tasks[0], download_args) - else: - download_multiple_tasks(client, tasks, download_args) diff --git a/xunlei-lixian/lixian_commands/help.py b/xunlei-lixian/lixian_commands/help.py deleted file mode 100644 index 230d8b7..0000000 --- a/xunlei-lixian/lixian_commands/help.py +++ /dev/null @@ -1,12 +0,0 @@ - -from lixian_commands.util import * -import lixian_help - -def lx_help(args): - if len(args) == 1: - helper = getattr(lixian_help, args[0].lower(), lixian_help.help) - usage(helper) - elif len(args) == 0: - usage(lixian_help.welcome_help) - else: - usage(lixian_help.help) diff --git a/xunlei-lixian/lixian_commands/info.py b/xunlei-lixian/lixian_commands/info.py deleted file mode 100644 index aa1cc09..0000000 --- a/xunlei-lixian/lixian_commands/info.py +++ /dev/null @@ -1,19 +0,0 @@ - - -from lixian import XunleiClient -from lixian_commands.util import * -from lixian_cli_parser import * -import lixian_help - -@command_line_parser(help=lixian_help.info) -@with_parser(parse_login) -@command_line_option('id', alias='i') -def lixian_info(args): - client = XunleiClient(args.username, args.password, args.cookies, login=False) - if args.id: - print client.get_username() - else: - print 'id:', client.get_username() - print 'internalid:', client.get_userid() - print 'gdriveid:', client.get_gdriveid() or '' - diff --git a/xunlei-lixian/lixian_commands/list.py b/xunlei-lixian/lixian_commands/list.py deleted file mode 100644 index 9ad2cb6..0000000 --- a/xunlei-lixian/lixian_commands/list.py +++ /dev/null @@ -1,57 +0,0 @@ - -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_config import get_config -import lixian_help -import lixian_query -import re - -@command_line_parser(help=lixian_help.list) -@with_parser(parse_login) -@with_parser(parse_colors) -@with_parser(parse_logging) -@with_parser(parse_size) -@command_line_option('all', default=True) -@command_line_option('completed') -@command_line_option('failed') -@command_line_option('deleted') -@command_line_option('expired') -@command_line_value('category') -@command_line_value('limit', default=get_config('limit')) -@command_line_value('page-size', default=get_config('page-size')) -@command_line_option('id', default=get_config('id', True)) -@command_line_option('name', default=True) -@command_line_option('status', default=True) -@command_line_option('dcid') -@command_line_option('gcid') -@command_line_option('original-url') -@command_line_option('download-url') -@command_line_option('speed') -@command_line_option('progress') -@command_line_option('date') -@command_line_option('n', default=get_config('n')) -def list_task(args): - - parent_ids = [a[:-1] for a in args if re.match(r'^#?\d+/$', a)] - if parent_ids and not all(re.match(r'^#?\d+/$', a) for a in args): - raise NotImplementedError("Can't mix 'id/' with others") - assert len(parent_ids) <= 1, "sub-tasks listing only supports single task id" - ids = [a[:-1] if re.match(r'^#?\d+/$', a) else a for a in args] - - client = create_client(args) - if parent_ids: - args[0] = args[0][:-1] - tasks = lixian_query.search_tasks(client, args) - assert len(tasks) == 1 - tasks = client.list_bt(tasks[0]) - #tasks = client.list_bt(client.get_task_by_id(parent_ids[0])) - tasks.sort(key=lambda x: int(x['index'])) - else: - tasks = lixian_query.search_tasks(client, args) - if len(args) == 1 and re.match(r'\d+/', args[0]) and len(tasks) == 1 and 'files' in tasks[0]: - parent_ids = [tasks[0]['id']] - tasks = tasks[0]['files'] - columns = ['n', 'id', 'name', 'status', 'size', 'progress', 'speed', 'date', 'dcid', 'gcid', 'original-url', 'download-url'] - columns = filter(lambda k: getattr(args, k), columns) - - output_tasks(tasks, columns, args, not parent_ids) diff --git a/xunlei-lixian/lixian_commands/login.py b/xunlei-lixian/lixian_commands/login.py deleted file mode 100644 index 4b16b9c..0000000 --- a/xunlei-lixian/lixian_commands/login.py +++ /dev/null @@ -1,42 +0,0 @@ - - -from lixian import XunleiClient -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_config import get_config -import lixian_help -from getpass import getpass - -@command_line_parser(help=lixian_help.login) -@with_parser(parse_login) -@with_parser(parse_logging) -def login(args): - if args.cookies == '-': - args._args['cookies'] = None - if len(args) < 1: - args.username = args.username or XunleiClient(cookie_path=args.cookies, login=False).get_username() or get_config('username') or raw_input('ID: ') - args.password = args.password or get_config('password') or getpass('Password: ') - elif len(args) == 1: - args.username = args.username or XunleiClient(cookie_path=args.cookies, login=False).get_username() or get_config('username') - args.password = args[0] - if args.password == '-': - args.password = getpass('Password: ') - elif len(args) == 2: - args.username, args.password = list(args) - if args.password == '-': - args.password = getpass('Password: ') - elif len(args) == 3: - args.username, args.password, args.cookies = list(args) - if args.password == '-': - args.password = getpass('Password: ') - elif len(args) > 3: - raise RuntimeError('Too many arguments') - if not args.username: - raise RuntimeError("What's your name?") - if args.cookies: - print 'Saving login session to', args.cookies - else: - print 'Testing login without saving session' - import lixian_verification_code - verification_code_reader = lixian_verification_code.default_verification_code_reader(args) - XunleiClient(args.username, args.password, args.cookies, login=True, verification_code_reader=verification_code_reader) diff --git a/xunlei-lixian/lixian_commands/logout.py b/xunlei-lixian/lixian_commands/logout.py deleted file mode 100644 index ce6dc14..0000000 --- a/xunlei-lixian/lixian_commands/logout.py +++ /dev/null @@ -1,18 +0,0 @@ - -from lixian import XunleiClient -from lixian_commands.util import * -from lixian_cli_parser import * -import lixian_config -import lixian_help - -@command_line_parser(help=lixian_help.logout) -@with_parser(parse_logging) -@command_line_value('cookies', default=lixian_config.LIXIAN_DEFAULT_COOKIES) -def logout(args): - if len(args): - raise RuntimeError('Too many arguments') - print 'logging out from', args.cookies - assert args.cookies - client = XunleiClient(cookie_path=args.cookies, login=False) - client.logout() - diff --git a/xunlei-lixian/lixian_commands/pause.py b/xunlei-lixian/lixian_commands/pause.py deleted file mode 100644 index 6d6ac5c..0000000 --- a/xunlei-lixian/lixian_commands/pause.py +++ /dev/null @@ -1,23 +0,0 @@ - -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_config import get_config -from lixian_encoding import default_encoding -import lixian_help -import lixian_query - -@command_line_parser(help=lixian_help.pause) -@with_parser(parse_login) -@with_parser(parse_colors) -@with_parser(parse_logging) -@command_line_option('i') -@command_line_option('all') -@command_line_value('limit', default=get_config('limit')) -@command_line_value('page-size', default=get_config('page-size')) -def pause_task(args): - client = create_client(args) - to_pause = lixian_query.search_tasks(client, args) - print "Below files are going to be paused:" - for x in to_pause: - print x['name'].encode(default_encoding) - client.pause_tasks(to_pause) diff --git a/xunlei-lixian/lixian_commands/readd.py b/xunlei-lixian/lixian_commands/readd.py deleted file mode 100644 index 1fb1ce5..0000000 --- a/xunlei-lixian/lixian_commands/readd.py +++ /dev/null @@ -1,40 +0,0 @@ - -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_encoding import default_encoding -import lixian_help -import lixian_query - -@command_line_parser(help=lixian_help.readd) -@with_parser(parse_login) -@with_parser(parse_logging) -@command_line_option('deleted') -@command_line_option('expired') -@command_line_option('all') -def readd_task(args): - if args.deleted: - status = 'deleted' - elif args.expired: - status = 'expired' - else: - raise NotImplementedError('Please use --expired or --deleted') - client = create_client(args) - if status == 'expired' and args.all: - return client.readd_all_expired_tasks() - to_readd = lixian_query.search_tasks(client, args) - non_bt = [] - bt = [] - if not to_readd: - return - print "Below files are going to be re-added:" - for x in to_readd: - print x['name'].encode(default_encoding) - if x['type'] == 'bt': - bt.append((x['bt_hash'], x['id'])) - else: - non_bt.append((x['original_url'], x['id'])) - if non_bt: - urls, ids = zip(*non_bt) - client.add_batch_tasks(urls, ids) - for hash, id in bt: - client.add_torrent_task_by_info_hash2(hash, id) diff --git a/xunlei-lixian/lixian_commands/rename.py b/xunlei-lixian/lixian_commands/rename.py deleted file mode 100644 index ed5e020..0000000 --- a/xunlei-lixian/lixian_commands/rename.py +++ /dev/null @@ -1,19 +0,0 @@ - -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_encoding import from_native -import lixian_help -import re -import sys - -@command_line_parser(help=lixian_help.rename) -@with_parser(parse_login) -@with_parser(parse_logging) -def rename_task(args): - if len(args) != 2 or not re.match(r'\d+$', args[0]): - usage(lixian_help.rename, 'Incorrect arguments') - sys.exit(1) - client = create_client(args) - taskid, new_name = args - task = client.get_task_by_id(taskid) - client.rename_task(task, from_native(new_name)) diff --git a/xunlei-lixian/lixian_commands/restart.py b/xunlei-lixian/lixian_commands/restart.py deleted file mode 100644 index 920054e..0000000 --- a/xunlei-lixian/lixian_commands/restart.py +++ /dev/null @@ -1,23 +0,0 @@ - -from lixian_commands.util import * -from lixian_cli_parser import * -from lixian_config import get_config -from lixian_encoding import default_encoding -import lixian_help -import lixian_query - -@command_line_parser(help=lixian_help.restart) -@with_parser(parse_login) -@with_parser(parse_colors) -@with_parser(parse_logging) -@command_line_option('i') -@command_line_option('all') -@command_line_value('limit', default=get_config('limit')) -@command_line_value('page-size', default=get_config('page-size')) -def restart_task(args): - client = create_client(args) - to_restart = lixian_query.search_tasks(client, args) - print "Below files are going to be restarted:" - for x in to_restart: - print x['name'].encode(default_encoding) - client.restart_tasks(to_restart) diff --git a/xunlei-lixian/lixian_commands/util.py b/xunlei-lixian/lixian_commands/util.py deleted file mode 100644 index e100cbc..0000000 --- a/xunlei-lixian/lixian_commands/util.py +++ /dev/null @@ -1,112 +0,0 @@ - -__all__ = ['parse_login', 'parse_colors', 'parse_logging', 'parse_size', 'create_client', 'output_tasks', 'usage'] - -from lixian_cli_parser import * -from lixian_config import get_config -from lixian_config import LIXIAN_DEFAULT_COOKIES -from lixian_encoding import default_encoding, to_native -from lixian_colors import colors -from getpass import getpass -import lixian_help - -@command_line_value('username', default=get_config('username')) -@command_line_value('password', default=get_config('password')) -@command_line_value('cookies', default=LIXIAN_DEFAULT_COOKIES) -@command_line_value('verification-code-path', default=get_config('verification-code-path')) -def parse_login(args): - if args.password == '-': - args.password = getpass('Password: ') - if args.cookies == '-': - args._args['cookies'] = None - return args - -@command_line_option('colors', default=get_config('colors', True)) -def parse_colors(args): - pass - -@command_line_value('log-level', default=get_config('log-level')) -@command_line_value('log-path', default=get_config('log-path')) -@command_line_option('debug') -@command_line_option('trace') -def parse_logging(args): - path = args.log_path - level = args.log_level - if args.trace: - level = 'trace' - elif args.debug: - level = 'debug' - if path or level: - import lixian_logging - level = level or 'info' - lixian_logging.init_logger(use_colors=args.colors, level=level, path=path) - logger = lixian_logging.get_logger() - import lixian - # inject logger to lixian (this makes lixian.py zero-dependency) - lixian.logger = logger - -@command_line_option('size', default=get_config('size')) -@command_line_option('format-size', default=get_config('format-size')) -def parse_size(args): - pass - -def create_client(args): - from lixian import XunleiClient - import lixian_verification_code - verification_code_reader = lixian_verification_code.default_verification_code_reader(args) - client = XunleiClient(args.username, args.password, args.cookies, verification_code_reader=verification_code_reader) - if args.page_size: - client.page_size = int(args.page_size) - return client - -def output_tasks(tasks, columns, args, top=True): - for i, t in enumerate(tasks): - status_colors = { - 'waiting': 'yellow', - 'downloading': 'magenta', - 'completed':'green', - 'pending':'cyan', - 'failed':'red', - } - c = status_colors[t['status_text']] - with colors(args.colors).ansi(c)(): - for k in columns: - if k == 'n': - if top: - print '#%d' % t['#'], - elif k == 'id': - print t.get('index', t['id']), - elif k == 'name': - print t['name'].encode(default_encoding), - elif k == 'status': - with colors(args.colors).bold(): - print t['status_text'], - elif k == 'size': - if args.format_size: - from lixian_util import format_size - print format_size(t['size']), - else: - print t['size'], - elif k == 'progress': - print t['progress'], - elif k == 'speed': - print t['speed'], - elif k == 'date': - print t['date'], - elif k == 'dcid': - print t['dcid'], - elif k == 'gcid': - print t['gcid'], - elif k == 'original-url': - print t['original_url'], - elif k == 'download-url': - print t['xunlei_url'], - else: - raise NotImplementedError(k) - print - -def usage(doc=lixian_help.usage, message=None): - if hasattr(doc, '__call__'): - doc = doc() - if message: - print to_native(message) - print to_native(doc).strip() diff --git a/xunlei-lixian/lixian_config.py b/xunlei-lixian/lixian_config.py deleted file mode 100644 index 2e0da6c..0000000 --- a/xunlei-lixian/lixian_config.py +++ /dev/null @@ -1,86 +0,0 @@ - -import os -import os.path - -def get_config_path(filename): - if os.path.exists(filename): - return filename - import sys - local_path = os.path.join(sys.path[0], filename) - if os.path.exists(local_path): - return local_path - user_home = os.getenv('USERPROFILE') or os.getenv('HOME') - lixian_home = os.getenv('LIXIAN_HOME') or user_home - return os.path.join(lixian_home, filename) - -LIXIAN_DEFAULT_CONFIG = get_config_path('.xunlei.lixian.config') -LIXIAN_DEFAULT_COOKIES = get_config_path('.xunlei.lixian.cookies') - -def load_config(path): - values = {} - if os.path.exists(path): - with open(path) as x: - for line in x.readlines(): - line = line.strip() - if line: - if line.startswith('--'): - line = line.lstrip('-') - if line.startswith('no-'): - values[line[3:]] = False - elif '=' in line: - k, v = line.split('=', 1) - values[k] = v - else: - values[line] = True - else: - raise NotImplementedError(line) - return values - -def dump_config(path, values): - with open(path, 'w') as x: - for k in values: - v = values[k] - if v is True: - x.write('--%s\n'%k) - elif v is False: - x.write('--no-%s\n'%k) - else: - x.write('--%s=%s\n'%(k, v)) - -class Config: - def __init__(self, path=LIXIAN_DEFAULT_CONFIG): - self.path = path - self.values = load_config(path) - def put(self, k, v=True): - self.values[k] = v - dump_config(self.path, self.values) - def get(self, k, v=None): - return self.values.get(k, v) - def delete(self, k): - if k in self.values: - del self.values[k] - dump_config(self.path, self.values) - def source(self): - if os.path.exists(self.path): - with open(self.path) as x: - return x.read() - def __str__(self): - return '' % self.values - -global_config = Config() - -def put_config(k, v=True): - if k.startswith('no-') and v is True: - k = k[3:] - v = False - global_config.put(k, v) - -def get_config(k, v=None): - return global_config.get(k, v) - -def delete_config(k): - return global_config.delete(k) - -def source_config(): - return global_config.source() - diff --git a/xunlei-lixian/lixian_download_asyn.py b/xunlei-lixian/lixian_download_asyn.py deleted file mode 100644 index a06db85..0000000 --- a/xunlei-lixian/lixian_download_asyn.py +++ /dev/null @@ -1,358 +0,0 @@ - -import asyncore -import asynchat -import socket -import re -#from cStringIO import StringIO -from time import time, sleep -import sys -import os - -#asynchat.async_chat.ac_out_buffer_size = 1024*1024 - -class http_client(asynchat.async_chat): - - def __init__(self, url, headers=None, start_from=0): - asynchat.async_chat.__init__(self) - - self.args = {'headers': headers, 'start_from': start_from} - - m = re.match(r'http://([^/:]+)(?::(\d+))?(/.*)?$', url) - assert m, 'Invalid url: %s' % url - host, port, path = m.groups() - port = int(port or 80) - path = path or '/' - - def resolve_host(host): - try: - return socket.gethostbyname(host) - except: - pass - host_ip = resolve_host(host) - if not host_ip: - self.log_error("host can't be resolved: " + host) - self.size = None - return - if host_ip == '180.168.41.175': - # fuck shanghai dian DNS - self.log_error('gethostbyname failed') - self.size = None - return - - - request_headers = {'host': host, 'connection': 'close'} - if start_from: - request_headers['RANGE'] = 'bytes=%d-' % start_from - if headers: - request_headers.update(headers) - headers = request_headers - self.request = 'GET %s HTTP/1.1\r\n%s\r\n\r\n' % (path, '\r\n'.join('%s: %s' % (k, headers[k]) for k in headers)) - self.op = 'GET' - - self.headers = {} # for response headers - - #self.buffer = StringIO() - self.buffer = [] - self.buffer_size = 0 - self.cache_size = 1024*1024 - self.size = None - self.completed = 0 - self.set_terminator("\r\n\r\n") - self.reading_headers = True - - self.create_socket(socket.AF_INET, socket.SOCK_STREAM) - try: - self.connect((host, port)) - except: - self.close() - self.log_error('connect_failed') - - def handle_connect(self): - self.start_time = time() - self.push(self.request) - - def handle_close(self): - asynchat.async_chat.handle_close(self) - self.flush_data() - if self.reading_headers: - self.log_error('incomplete http response') - return - self.handle_status_update(self.size, self.completed, force_update=True) - self.handle_speed_update(self.completed, self.start_time, force_update=True) - if self.size is not None and self.completed < self.size: - self.log_error('incomplete download') - - def handle_connection_error(self): - self.handle_error() - - def handle_error(self): - self.close() - self.flush_data() - error_message = sys.exc_info()[1] - self.log_error('there is some error: %s' % error_message) - #raise - - def collect_incoming_data(self, data): - if self.reading_headers: - #self.buffer.write(data) - self.buffer.append(data) - self.buffer_size += len(data) - return - elif self.cache_size: - #self.buffer.write(data) - self.buffer.append(data) - self.buffer_size += len(data) - #if self.buffer.tell() > self.cache_size: - if self.buffer_size > self.cache_size: - #self.handle_data(self.buffer.getvalue()) - self.handle_data(''.join(self.buffer)) - #self.buffer.truncate(0) - #self.buffer.clear() - del self.buffer[:] - self.buffer_size = 0 - else: - self.handle_data(data) - - self.completed += len(data) - self.handle_status_update(self.size, self.completed) - self.handle_speed_update(self.completed, self.start_time) - if self.size == self.completed: - self.close() - self.flush_data() - self.handle_status_update(self.size, self.completed, force_update=True) - self.handle_speed_update(self.completed, self.start_time, force_update=True) - - def handle_data(self, data): - print len(data) - pass - - def flush_data(self): - #if self.buffer.tell(): - if self.buffer_size: - #self.handle_data(self.buffer.getvalue()) - self.handle_data(''.join(self.buffer)) - #self.buffer.truncate(0) - del self.buffer[:] - self.buffer_size = 0 - - def parse_headers(self, header): - lines = header.split('\r\n') - status_line = lines.pop(0) - #print status_line - protocal, status_code, status_text = re.match(r'^HTTP/([\d.]+) (\d+) (.+)$', status_line).groups() - status_code = int(status_code) - self.status_code = status_code - self.status_text = status_text - #headers = dict(h.split(': ', 1) for h in lines) - for k, v in (h.split(': ', 1) for h in lines): - self.headers[k.lower()] = v - - if status_code in (200, 206): - pass - elif status_code == 302: - return self.handle_http_relocate(self.headers['location']) - else: - return self.handle_http_status_error() - - self.size = self.headers.get('content-length', None) - if self.size is not None: - self.size = int(self.size) - self.handle_http_headers() - - def found_terminator(self): - if self.reading_headers: - self.reading_headers = False - #self.parse_headers("".join(self.buffer.getvalue())) - self.parse_headers("".join(self.buffer)) - #self.buffer.truncate(0) - del self.buffer[:] - self.buffer_size = 0 - self.set_terminator(None) - else: - raise NotImplementedError() - - def handle_http_headers(self): - pass - - def handle_http_status_error(self): - self.close() - - def handle_http_relocate(self, location): - self.close() - relocate_times = getattr(self, 'relocate_times', 0) - max_relocate_times = getattr(self, 'max_relocate_times', 2) - if relocate_times >= max_relocate_times: - raise Exception('too many relocate times') - new_client = self.__class__(location, **self.args) - new_client.relocate_times = relocate_times + 1 - new_client.max_relocate_times = max_relocate_times - self.next_client = new_client - - def handle_status_update(self, total, completed, force_update=False): - pass - - def handle_speed_update(self, completed, start_time, force_update=False): - pass - - def log_error(self, message): - print 'log_error', message - self.error_message = message - -class ProgressBar: - def __init__(self, total=0): - self.total = total - self.completed = 0 - self.start = time() - self.speed = 0 - self.bar_width = 0 - self.displayed = False - def update(self): - self.displayed = True - bar_size = 40 - if self.total: - percent = self.completed * 100.0 / self.total - if percent > 100: - percent = 100.0 - dots = int(bar_size * percent / 100) - plus = percent / 100 * bar_size - dots - if plus > 0.8: - plus = '=' - elif plus > 0.4: - plus = '-' - else: - plus = '' - bar = '=' * dots + plus - percent = int(percent) - else: - percent = 0 - bar = '-' - speed = self.speed - if speed < 1000: - speed = '%sB/s' % int(speed) - elif speed < 1000*10: - speed = '%.1fK/s' % (speed/1000.0) - elif speed < 1000*1000: - speed = '%dK/s' % int(speed/1000) - elif speed < 1000*1000*100: - speed = '%.1fM/s' % (speed/1000.0/1000.0) - else: - speed = '%dM/s' % int(speed/1000/1000) - seconds = time() - self.start - if seconds < 10: - seconds = '%.1fs' % seconds - elif seconds < 60: - seconds = '%ds' % int(seconds) - elif seconds < 60*60: - seconds = '%dm%ds' % (int(seconds/60), int(seconds)%60) - elif seconds < 60*60*24: - seconds = '%dh%dm%ds' % (int(seconds)/60/60, (int(seconds)/60)%60, int(seconds)%60) - else: - seconds = int(seconds) - days = seconds/60/60/24 - seconds -= days*60*60*24 - hours = seconds/60/60 - seconds -= hours*60*60 - minutes = seconds/60 - seconds -= minutes*60 - seconds = '%dd%dh%dm%ds' % (days, hours, minutes, seconds) - completed = ','.join((x[::-1] for x in reversed(re.findall('..?.?', str(self.completed)[::-1])))) - bar = '{0:>3}%[{1:<40}] {2:<12} {3:>4} in {4:>6s}'.format(percent, bar, completed, speed, seconds) - new_bar_width = len(bar) - bar = bar.ljust(self.bar_width) - self.bar_width = new_bar_width - sys.stdout.write('\r'+bar) - sys.stdout.flush() - def update_status(self, total, completed): - self.total = total - self.completed = completed - self.update() - def update_speed(self, start, speed): - self.start = start - self.speed = speed - self.update() - def done(self): - if self.displayed: - print - self.displayed = False - -def download(url, path, headers=None, resuming=False): - class download_client(http_client): - def __init__(self, url, headers=headers, start_from=0): - self.output = None - self.bar = ProgressBar() - http_client.__init__(self, url, headers=headers, start_from=start_from) - self.start_from = start_from - self.last_status_time = time() - self.last_speed_time = time() - self.last_size = 0 - self.path = path - def handle_close(self): - http_client.handle_close(self) - if self.output: - self.output.close() - self.output = None - def handle_http_status_error(self): - http_client.handle_http_status_error(self) - self.log_error('http status error: %s, %s' % (self.status_code, self.status_text)) - def handle_data(self, data): - if not self.output: - if self.start_from: - self.output = open(path, 'ab') - else: - self.output = open(path, 'wb') - self.output.write(data) - def handle_status_update(self, total, completed, force_update=False): - if total is None: - return - if time() - self.last_status_time > 1 or force_update: - #print '%.02f' % (completed*100.0/total) - self.bar.update_status(total+start_from, completed+start_from) - self.last_status_time = time() - def handle_speed_update(self, completed, start_time, force_update=False): - now = time() - period = now - self.last_speed_time - if period > 1 or force_update: - #print '%.02f, %.02f' % ((completed-self.last_size)/period, completed/(now-start_time)) - self.bar.update_speed(start_time, (completed-self.last_size)/period) - self.last_speed_time = time() - self.last_size = completed - def log_error(self, message): - self.bar.done() - http_client.log_error(self, message) - def __del__(self): # XXX: sometimes handle_close() is not called, don't know why... - #http_client.__del__(self) - if self.output: - self.output.close() - self.output = None - - max_retry_times = 25 - retry_times = 0 - start_from = 0 - if resuming and os.path.exists(path): - start_from = os.path.getsize(path) - # TODO: fix status bar for resuming - while True: - client = download_client(url, start_from=start_from) - asyncore.loop() - while hasattr(client, 'next_client'): - client = client.next_client - client.bar.done() - if getattr(client, 'error_message', None): - retry_times += 1 - if retry_times >= max_retry_times: - raise Exception(client.error_message) - if client.size and client.completed: - start_from = os.path.getsize(path) - print 'retry', retry_times - sleep(retry_times) - else: - break - - -def main(): - url, path = sys.argv[1:] - download(url, path) - -if __name__ == '__main__': - main() - diff --git a/xunlei-lixian/lixian_download_tools.py b/xunlei-lixian/lixian_download_tools.py deleted file mode 100644 index 4d1185b..0000000 --- a/xunlei-lixian/lixian_download_tools.py +++ /dev/null @@ -1,128 +0,0 @@ - -__all__ = ['download_tool', 'get_tool'] - -from lixian_config import * -import subprocess -import urllib2 -import os.path - -download_tools = {} - -def download_tool(name): - def register(tool): - download_tools[name] = tool_adaptor(tool) - return tool - return register - -class DownloadToolAdaptor: - def __init__(self, tool, **kwargs): - self.tool = tool - self.client = kwargs['client'] - self.url = kwargs['url'] - self.path = kwargs['path'] - self.resuming = kwargs.get('resuming') - self.size = kwargs['size'] - def finished(self): - assert os.path.getsize(self.path) <= self.size, 'existing file (%s) bigger than expected (%s)' % (os.path.getsize(self.path), self.size) - return os.path.getsize(self.path) == self.size - def __call__(self): - self.tool(self.client, self.url, self.path, self.resuming) - -def tool_adaptor(tool): - import types - if type(tool) == types.FunctionType: - def adaptor(**kwargs): - return DownloadToolAdaptor(tool, **kwargs) - return adaptor - else: - return tool - - -def check_bin(bin): - import distutils.spawn - assert distutils.spawn.find_executable(bin), "Can't find %s" % bin - -@download_tool('urllib2') -def urllib2_download(client, download_url, filename, resuming=False): - '''In the case you don't even have wget...''' - assert not resuming - print 'Downloading', download_url, 'to', filename, '...' - request = urllib2.Request(download_url, headers={'Cookie': 'gdriveid='+client.get_gdriveid()}) - response = urllib2.urlopen(request) - import shutil - with open(filename, 'wb') as output: - shutil.copyfileobj(response, output) - -@download_tool('asyn') -def asyn_download(client, download_url, filename, resuming=False): - import lixian_download_asyn - lixian_download_asyn.download(download_url, filename, headers={'Cookie': 'gdriveid='+str(client.get_gdriveid())}, resuming=resuming) - -@download_tool('wget') -def wget_download(client, download_url, filename, resuming=False): - gdriveid = str(client.get_gdriveid()) - wget_opts = ['wget', '--header=Cookie: gdriveid='+gdriveid, download_url, '-O', filename] - if resuming: - wget_opts.append('-c') - wget_opts.extend(get_config('wget-opts', '').split()) - check_bin(wget_opts[0]) - exit_code = subprocess.call(wget_opts) - if exit_code != 0: - raise Exception('wget exited abnormally') - -@download_tool('curl') -def curl_download(client, download_url, filename, resuming=False): - gdriveid = str(client.get_gdriveid()) - curl_opts = ['curl', '-L', download_url, '--cookie', 'gdriveid='+gdriveid, '--output', filename] - if resuming: - curl_opts += ['--continue-at', '-'] - curl_opts.extend(get_config('curl-opts', '').split()) - check_bin(curl_opts[0]) - exit_code = subprocess.call(curl_opts) - if exit_code != 0: - raise Exception('curl exited abnormally') - -@download_tool('aria2') -@download_tool('aria2c') -class Aria2DownloadTool: - def __init__(self, **kwargs): - self.gdriveid = str(kwargs['client'].get_gdriveid()) - self.url = kwargs['url'] - self.path = kwargs['path'] - self.size = kwargs['size'] - self.resuming = kwargs.get('resuming') - def finished(self): - assert os.path.getsize(self.path) <= self.size, 'existing file (%s) bigger than expected (%s)' % (os.path.getsize(self.path), self.size) - return os.path.getsize(self.path) == self.size and not os.path.exists(self.path + '.aria2') - def __call__(self): - gdriveid = self.gdriveid - download_url = self.url - path = self.path - resuming = self.resuming - dir = os.path.dirname(path) - filename = os.path.basename(path) - aria2_opts = ['aria2c', '--header=Cookie: gdriveid='+gdriveid, download_url, '--out', filename, '--file-allocation=none'] - if dir: - aria2_opts.extend(('--dir', dir)) - if resuming: - aria2_opts.append('-c') - aria2_opts.extend(get_config('aria2-opts', '').split()) - check_bin(aria2_opts[0]) - exit_code = subprocess.call(aria2_opts) - if exit_code != 0: - raise Exception('aria2c exited abnormally') - -@download_tool('axel') -def axel_download(client, download_url, path, resuming=False): - gdriveid = str(client.get_gdriveid()) - axel_opts = ['axel', '--header=Cookie: gdriveid='+gdriveid, download_url, '--output', path] - axel_opts.extend(get_config('axel-opts', '').split()) - check_bin(axel_opts[0]) - exit_code = subprocess.call(axel_opts) - if exit_code != 0: - raise Exception('axel exited abnormally') - -def get_tool(name): - return download_tools[name] - - diff --git a/xunlei-lixian/lixian_encoding.py b/xunlei-lixian/lixian_encoding.py deleted file mode 100644 index 33930d2..0000000 --- a/xunlei-lixian/lixian_encoding.py +++ /dev/null @@ -1,27 +0,0 @@ - -from lixian_config import get_config -import sys - -default_encoding = get_config('encoding', sys.getfilesystemencoding()) -if default_encoding is None or default_encoding.lower() == 'ascii': - default_encoding = 'utf-8' - - -def to_native(s): - if type(s) == unicode: - return s.encode(default_encoding) - else: - return s - -def from_native(s): - if type(s) == str: - return s.decode(default_encoding) - else: - return s - -def try_native_to_utf_8(url): - try: - return url.decode(default_encoding).encode('utf-8') - except: - return url - diff --git a/xunlei-lixian/lixian_filter_expr.py b/xunlei-lixian/lixian_filter_expr.py deleted file mode 100644 index 1a9fe9f..0000000 --- a/xunlei-lixian/lixian_filter_expr.py +++ /dev/null @@ -1,69 +0,0 @@ - -__all__ = ['filter_expr'] - -import re - -def get_name(x): - assert isinstance(x, (basestring, dict)) - if type(x) == dict: - return x['name'] - else: - return x - -def filter_expr1(links, p): - if not links: - return links - if re.match(r'^\[[^][]+\]$', p): - matched = [] - for p in re.split(r'\s*,\s*', p[1:-1]): - assert re.match(r'^\d+(-\d+)?|\.\w+$', p), p - if re.match(r'^\d+$', p): - i = int(p) - matched.append((i, links[i])) - elif '-' in p: - start, end = p.split('-') - if not start: - start = 0 - if not end: - end = len(links) - 1 - start = int(start) - end = int(end) - assert 0 <= start < len(links) - assert 0 <= end < len(links) - if start <= end: - matched += list(enumerate(links))[start:end+1] - else: - matched += reversed(list(enumerate(links))[end:start+1]) - elif p.startswith('.'): - matched += filter(lambda (i, x): get_name(x).lower().endswith(p.lower()), enumerate(links)) - else: - raise NotImplementedError(p) - indexes = [] - for i, _ in matched: - if i not in indexes: - indexes.append(i) - return [links[x] for x in indexes] - elif re.match(r'^\d+$', p): - n = int(p) - if 0 <= n < len(links): - return [links[int(p)]] - else: - return filter(lambda x: re.search(p, get_name(x), re.I), links) - elif p == '*': - return links - elif re.match(r'\.\w+$', p): - return filter(lambda x: get_name(x).lower().endswith(p.lower()), links) - else: - import lixian_plugins.filters - filter_results = lixian_plugins.filters.filter_things(links, p) - if filter_results is None: - return filter(lambda x: re.search(p, get_name(x), re.I), links) - else: - return filter_results - -def filter_expr(links, expr): - for p in expr.split('/'): - links = filter_expr1(links, p) - return links - - diff --git a/xunlei-lixian/lixian_hash.py b/xunlei-lixian/lixian_hash.py deleted file mode 100755 index bae5205..0000000 --- a/xunlei-lixian/lixian_hash.py +++ /dev/null @@ -1,90 +0,0 @@ -#!/usr/bin/env python - -import hashlib -import lixian_hash_ed2k -import lixian_hash_bt -import os - -def lib_hash_file(h, path): - with open(path, 'rb') as stream: - while True: - bytes = stream.read(1024*1024) - if not bytes: - break - h.update(bytes) - return h.hexdigest() - -def sha1_hash_file(path): - return lib_hash_file(hashlib.sha1(), path) - -def verify_sha1(path, sha1): - return sha1_hash_file(path).lower() == sha1.lower() - -def md5_hash_file(path): - return lib_hash_file(hashlib.md5(), path) - -def verify_md5(path, md5): - return md5_hash_file(path).lower() == md5.lower() - -def md4_hash_file(path): - return lib_hash_file(hashlib.new('md4'), path) - -def verify_md4(path, md4): - return md4_hash_file(path).lower() == md4.lower() - -def dcid_hash_file(path): - h = hashlib.sha1() - size = os.path.getsize(path) - with open(path, 'rb') as stream: - if size < 0xF000: - h.update(stream.read()) - else: - h.update(stream.read(0x5000)) - stream.seek(size/3) - h.update(stream.read(0x5000)) - stream.seek(size-0x5000) - h.update(stream.read(0x5000)) - return h.hexdigest() - -def verify_dcid(path, dcid): - return dcid_hash_file(path).lower() == dcid.lower() - -def main(args): - option = args.pop(0) - def verify_bt(f, t): - from lixian_progress import SimpleProgressBar - bar = SimpleProgressBar() - result = lixian_hash_bt.verify_bt_file(t, f, progress_callback=bar.update) - bar.done() - return result - if option.startswith('--verify'): - hash_fun = {'--verify-sha1':verify_sha1, - '--verify-md5':verify_md5, - '--verify-md4':verify_md4, - '--verify-dcid':verify_dcid, - '--verify-ed2k':lixian_hash_ed2k.verify_ed2k_link, - '--verify-bt': verify_bt, - }[option] - assert len(args) == 2 - hash, path = args - if hash_fun(path, hash): - print 'looks good...' - else: - print 'failed...' - else: - hash_fun = {'--sha1':sha1_hash_file, - '--md5':md5_hash_file, - '--md4':md4_hash_file, - '--dcid':dcid_hash_file, - '--ed2k':lixian_hash_ed2k.generate_ed2k_link, - '--info-hash':lixian_hash_bt.info_hash, - }[option] - for f in args: - h = hash_fun(f) - print '%s *%s' % (h, f) - -if __name__ == '__main__': - import sys - args = sys.argv[1:] - main(args) - diff --git a/xunlei-lixian/lixian_hash_bt.py b/xunlei-lixian/lixian_hash_bt.py deleted file mode 100644 index 3a026be..0000000 --- a/xunlei-lixian/lixian_hash_bt.py +++ /dev/null @@ -1,249 +0,0 @@ - -import os.path -import sys -import hashlib -from cStringIO import StringIO -import re - -from lixian_encoding import default_encoding - -def magnet_to_infohash(magnet): - import re - import base64 - m = re.match(r'magnet:\?xt=urn:btih:(\w+)', magnet) - assert m, magnet - code = m.group(1) - if re.match(r'^[a-zA-Z0-9]{40}$', code): - return code.decode('hex') - else: - return base64.b32decode(code) - -class decoder: - def __init__(self, bytes): - self.bytes = bytes - self.i = 0 - def decode_value(self): - x = self.bytes[self.i] - if x.isdigit(): - return self.decode_string() - self.i += 1 - if x == 'd': - v = {} - while self.peek() != 'e': - k = self.decode_string() - v[k] = self.decode_value() - self.i += 1 - return v - elif x == 'l': - v = [] - while self.peek() != 'e': - v.append(self.decode_value()) - self.i += 1 - return v - elif x == 'i': - return self.decode_int() - else: - raise NotImplementedError(x) - def decode_string(self): - i = self.bytes.index(':', self.i) - n = int(self.bytes[self.i:i]) - s = self.bytes[i+1:i+1+n] - self.i = i + 1 + n - return s - def decode_int(self): - e = self.bytes.index('e', self.i) - n = int(self.bytes[self.i:e]) - self.i = e + 1 - return n - def peek(self): - return self.bytes[self.i] - -class encoder: - def __init__(self, stream): - self.stream = stream - def encode(self, v): - if type(v) == str: - self.stream.write(str(len(v))) - self.stream.write(':') - self.stream.write(v) - elif type(v) == dict: - self.stream.write('d') - for k in sorted(v): - self.encode(k) - self.encode(v[k]) - self.stream.write('e') - elif type(v) == list: - self.stream.write('l') - for x in v: - self.encode(x) - self.stream.write('e') - elif type(v) in (int, long): - self.stream.write('i') - self.stream.write(str(v)) - self.stream.write('e') - else: - raise NotImplementedError(type(v)) - -def bdecode(bytes): - return decoder(bytes).decode_value() - -def bencode(v): - from cStringIO import StringIO - stream = StringIO() - encoder(stream).encode(v) - return stream.getvalue() - -def assert_content(content): - assert re.match(r'd\d+:', content), 'Probably not a valid content file [%s...]' % repr(content[:17]) - -def info_hash_from_content(content): - assert_content(content) - return hashlib.sha1(bencode(bdecode(content)['info'])).hexdigest() - -def info_hash(path): - if not path.lower().endswith('.torrent'): - print '[WARN] Is it really a .torrent file? '+path - if os.path.getsize(path) > 3*1000*1000: - raise NotImplementedError('Torrent file too big') - with open(path, 'rb') as stream: - return info_hash_from_content(stream.read()) - -def encode_path(path): - return path.decode('utf-8').encode(default_encoding) - -class sha1_reader: - def __init__(self, pieces, progress_callback=None): - assert pieces - assert len(pieces) % 20 == 0 - self.total = len(pieces)/20 - self.processed = 0 - self.stream = StringIO(pieces) - self.progress_callback = progress_callback - def next_sha1(self): - self.processed += 1 - if self.progress_callback: - self.progress_callback(float(self.processed)/self.total) - return self.stream.read(20) - -def sha1_update_stream(sha1, stream, n): - while n > 0: - readn = min(n, 1024*1024) - bytes = stream.read(readn) - assert len(bytes) == readn - n -= readn - sha1.update(bytes) - assert n == 0 - -def verify_bt_single_file(path, info, progress_callback=None): - # TODO: check md5sum if available - if os.path.getsize(path) != info['length']: - return False - piece_length = info['piece length'] - assert piece_length > 0 - sha1_stream = sha1_reader(info['pieces'], progress_callback=progress_callback) - size = info['length'] - with open(path, 'rb') as stream: - while size > 0: - n = min(size, piece_length) - size -= n - sha1sum = hashlib.sha1() - sha1_update_stream(sha1sum, stream, n) - if sha1sum.digest() != sha1_stream.next_sha1(): - return False - assert size == 0 - assert stream.read(1) == '' - assert sha1_stream.next_sha1() == '' - return True - -def verify_bt_multiple(folder, info, file_set=None, progress_callback=None): - # TODO: check md5sum if available - piece_length = info['piece length'] - assert piece_length > 0 - - path_encoding = info.get('encoding', 'utf-8') - files = [] - for x in info['files']: - if 'path.utf-8' in x: - unicode_path = [p.decode('utf-8') for p in x['path.utf-8']] - else: - unicode_path = [p.decode(path_encoding) for p in x['path']] - native_path = [p.encode(default_encoding) for p in unicode_path] - utf8_path = [p.encode('utf-8') for p in unicode_path] - files.append({'path':os.path.join(folder, apply(os.path.join, native_path)), 'length':x['length'], 'file':utf8_path}) - - sha1_stream = sha1_reader(info['pieces'], progress_callback=progress_callback) - sha1sum = hashlib.sha1() - - piece_left = piece_length - complete_piece = True - - while files: - f = files.pop(0) - path = f['path'] - size = f['length'] - if os.path.exists(path) and ((not file_set) or (f['file'] in file_set)): - if os.path.getsize(path) != size: - return False - if size <= piece_left: - with open(path, 'rb') as stream: - sha1_update_stream(sha1sum, stream, size) - assert stream.read(1) == '' - piece_left -= size - if not piece_left: - if sha1sum.digest() != sha1_stream.next_sha1() and complete_piece: - return False - complete_piece = True - sha1sum = hashlib.sha1() - piece_left = piece_length - else: - with open(path, 'rb') as stream: - while size >= piece_left: - size -= piece_left - sha1_update_stream(sha1sum, stream, piece_left) - if sha1sum.digest() != sha1_stream.next_sha1() and complete_piece: - return False - complete_piece = True - sha1sum = hashlib.sha1() - piece_left = piece_length - if size: - sha1_update_stream(sha1sum, stream, size) - piece_left -= size - else: - if size: - while size >= piece_left: - size -= piece_left - sha1_stream.next_sha1() - sha1sum = hashlib.sha1() - piece_left = piece_length - if size: - complete_piece = False - piece_left -= size - else: - complete_piece = True - - if piece_left < piece_length: - if complete_piece: - if sha1sum.digest() != sha1_stream.next_sha1(): - return False - else: - sha1_stream.next_sha1() - assert sha1_stream.next_sha1() == '' - - return True - -def verify_bt(path, info, file_set=None, progress_callback=None): - if not os.path.exists(path): - raise Exception("File doesn't exist: %s" % path) - if 'files' not in info: - if os.path.isfile(path): - return verify_bt_single_file(path, info, progress_callback=progress_callback) - else: - path = os.path.join(path, encode_path(info['name'])) - return verify_bt_single_file(path, info, progress_callback=progress_callback) - else: - return verify_bt_multiple(path, info, file_set=file_set, progress_callback=progress_callback) - -def verify_bt_file(path, torrent_path, file_set=None, progress_callback=None): - with open(torrent_path, 'rb') as x: - return verify_bt(path, bdecode(x.read())['info'], file_set, progress_callback) - diff --git a/xunlei-lixian/lixian_hash_ed2k.py b/xunlei-lixian/lixian_hash_ed2k.py deleted file mode 100644 index 0d88552..0000000 --- a/xunlei-lixian/lixian_hash_ed2k.py +++ /dev/null @@ -1,79 +0,0 @@ - -import hashlib - -chunk_size = 9728000 -buffer_size = 1024*1024 - -def md4(): - return hashlib.new('md4') - -def hash_stream(stream): - total_md4 = None - while True: - chunk_md4 = md4() - chunk_left = chunk_size - while chunk_left: - n = min(chunk_left, buffer_size) - part = stream.read(n) - chunk_md4.update(part) - if len(part) < n: - if total_md4: - total_md4.update(chunk_md4.digest()) - return total_md4.hexdigest() - else: - return chunk_md4.hexdigest() - chunk_left -= n - if total_md4 is None: - total_md4 = md4() - total_md4.update(chunk_md4.digest()) - raise NotImplementedError() - -def hash_string(s): - from cStringIO import StringIO - return hash_stream(StringIO(s)) - -def hash_file(path): - with open(path, 'rb') as stream: - return hash_stream(stream) - -def parse_ed2k_link(link): - import re, urllib - ed2k_re = r'ed2k://\|file\|([^|]*)\|(\d+)\|([a-fA-F0-9]{32})\|' - m = re.match(ed2k_re, link) or re.match(ed2k_re, urllib.unquote(link)) - if not m: - raise Exception('not an acceptable ed2k link: '+link) - name, file_size, hash_hex = m.groups() - from lixian_url import unquote_url - return unquote_url(name), hash_hex.lower(), int(file_size) - -def parse_ed2k_id(link): - return parse_ed2k_link(link)[1:] - -def parse_ed2k_file(link): - return parse_ed2k_link(link)[0] - -def verify_ed2k_link(path, link): - hash_hex, file_size = parse_ed2k_id(link) - import os.path - if os.path.getsize(path) != file_size: - return False - return hash_file(path).lower() == hash_hex.lower() - -def generate_ed2k_link(path): - import sys, os.path, urllib - filename = os.path.basename(path) - encoding = sys.getfilesystemencoding() - if encoding.lower() != 'ascii': - filename = filename.decode(encoding).encode('utf-8') - return 'ed2k://|file|%s|%d|%s|/' % (urllib.quote(filename), os.path.getsize(path), hash_file(path)) - -def test_md4(): - assert hash_string("") == '31d6cfe0d16ae931b73c59d7e0c089c0' - assert hash_string("a") == 'bde52cb31de33e46245e05fbdbd6fb24' - assert hash_string("abc") == 'a448017aaf21d8525fc10ae87aa6729d' - assert hash_string("message digest") == 'd9130a8164549fe818874806e1c7014b' - assert hash_string("abcdefghijklmnopqrstuvwxyz") == 'd79e1c308aa5bbcdeea8ed63df412da9' - assert hash_string("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789") == '043f8582f241db351ce627e153e7f0e4' - assert hash_string("12345678901234567890123456789012345678901234567890123456789012345678901234567890") == 'e33b4ddc9c38f2199c3e7b164fcc0536' - - diff --git a/xunlei-lixian/lixian_help.py b/xunlei-lixian/lixian_help.py deleted file mode 100644 index 1d50424..0000000 --- a/xunlei-lixian/lixian_help.py +++ /dev/null @@ -1,300 +0,0 @@ - -basic_commands = [ - ('help', "try this help..."), - ('login', "login Xunlei cloud"), - ('download', "download tasks from Xunlei cloud"), - ('list', "list tasks on Xunlei cloud"), - ('add', "add tasks to Xunlei cloud"), - ('delete', "delete tasks from Xunlei cloud"), - ('pause', "pause tasks on Xunlei cloud"), - ('restart', "restart tasks on Xunlei cloud"), - ('rename', "rename task"), - ('readd', "re-add tasks"), - ('config', "save configuration so you don't have to repeat it"), - ('info', "print user id, internal user id, and gdriveid"), - ('logout', "logout from Xunlei cloud"), -] - -def join_commands(commands): - n = max(len(x[0]) for x in commands) - n = max(n, 10) - return ''.join(' %%-%ds %%s\n' % n % (k, h) for (k, h) in commands) - -basic_usage = '''python lixian_cli.py [] - -Basic commands: -''' + join_commands(basic_commands) - -extended_usage = '' - -# lx -def usage(): - return basic_usage + ''' -Use 'python lixian_cli.py help' for details. -Use 'python lixian_cli.py help ' for more information on a specific command. -Check https://github.com/iambus/xunlei-lixian for detailed (and Chinese) doc.''' - -# lx xxx -# lx help help -help_help = '''Get helps: - python lixian_cli.py help help - python lixian_cli.py help examples - python lixian_cli.py help readme - python lixian_cli.py help ''' - -# lx xxx -# lx help help -help = help_help - -# lx help -# lx -h -def welcome_help(): - return '''Python script for Xunlei cloud. - -Basic usage: -''' + basic_usage + extended_usage + '\n' + help_help - -def examples(): - return '''python lixian_cli.py login "Your Xunlei account" "Your password" -python lixian_cli.py login "Your password" -python lixian_cli.py login - -python lixian_cli.py config username "Your Xunlei account" -python lixian_cli.py config password "Your password" - -python lixian_cli.py list -python lixian_cli.py list --completed -python lixian_cli.py list --completed --name --original-url --download-url --no-status --no-id -python lixian_cli.py list --deleted -python lixian_cli.py list --expired -python lixian_cli.py list id1 id2 -python lixian_cli.py list zip rar -python lixian_cli.py list 2012.04.04 2012.04.05 - -python lixian_cli.py download task-id -python lixian_cli.py download ed2k-url -python lixian_cli.py download --tool=wget ed2k-url -python lixian_cli.py download --tool=asyn ed2k-url -python lixian_cli.py download ed2k-url --output "file to save" -python lixian_cli.py download id1 id2 id3 -python lixian_cli.py download url1 url2 url3 -python lixian_cli.py download --input download-urls-file -python lixian_cli.py download --input download-urls-file --delete -python lixian_cli.py download --input download-urls-file --output-dir root-dir-to-save-files -python lixian_cli.py download bt://torrent-info-hash -python lixian_cli.py download 1.torrent -python lixian_cli.py download torrent-info-hash -python lixian_cli.py download --bt http://xxx/xxx.torrent -python lixian_cli.py download bt-task-id/file-id -python lixian_cli.py download --all -python lixian_cli.py download mkv -python lixian_cli.py download 2012.04.04 -python lixian_cli.py download 0 1 2 -python lixian_cli.py download 0-2 - -python lixian_cli.py add url -python lixian_cli.py add 1.torrent -python lixian_cli.py add torrent-info-hash -python lixian_cli.py add --bt http://xxx/xxx.torrent - -python lixian_cli.py delete task-id -python lixian_cli.py delete url -python lixian_cli.py delete file-name-on-cloud-to-delete - -python lixian_cli.py pause id - -python lixian_cli.py restart id - -python lixian_cli.py rename id name - -python lixian_cli.py logout - -Please check https://github.com/iambus/xunlei-lixian for detailed (and Chinese) doc. -''' - -def readme(): - import sys - import os.path - doc = os.path.join(sys.path[0], 'README.md') - with open(doc) as txt: - return txt.read().decode('utf-8') - - -login = '''python lixian_cli.py login - -login Xunlei cloud - -Examples: - python lixian_cli.py login "Your Xunlei account" "Your password" - python lixian_cli.py login "Your password" - python lixian_cli.py login -''' - -download = '''python lixian_cli.py download [options] [id|url]... - -download tasks from Xunlei cloud - -Options: - --input=[file] -i Download URLs found in file. - --output=[file] -o Download task to file. - --output-dir=[dir] Download task to dir. - --tool=[wget|asyn|aria2|curl] Choose download tool. - Default: wget - --continue -c Continue downloading a partially downloaded file. - Default: false. - --overwrite Overwrite partially downloaded file. - Default: false. - --delete Delete task from Xunlei cloud after download is finished. - Default: false. - --torrent --bt Treat URLs as torrent files - Default: false. - --all Download all tasks. This option will be ignored if specific download URLs or task ids can be found. - Default: false. - --hash When this option is false (--no-hash), never do full hash, but a minimal hash will be performed (supposed to be very fast). - Default: true. - --mini-hash If the target file already exists, and the file size is complete, do a minimal hash (instead of full hash, which would be much more expensive). This is useful when you are resuming a batch download, in this case the previously downloaded and verified files won't be re-verified. - Default: false. - -Examples: - python lixian_cli.py download task-id - python lixian_cli.py download ed2k-url - python lixian_cli.py download --tool=wget ed2k-url - python lixian_cli.py download --tool=asyn ed2k-url - python lixian_cli.py download ed2k-url --output "file to save" - python lixian_cli.py download id1 id2 id3 - python lixian_cli.py download url1 url2 url3 - python lixian_cli.py download --input download-urls-file - python lixian_cli.py download --input download-urls-file --delete - python lixian_cli.py download --input download-urls-file --output-dir root-dir-to-save-files - python lixian_cli.py download bt://torrent-info-hash - python lixian_cli.py download 1.torrent - python lixian_cli.py download torrent-info-hash - python lixian_cli.py download --bt http://xxx/xxx.torrent - python lixian_cli.py download bt-task-id/file-id - python lixian_cli.py download --all - python lixian_cli.py download mkv - python lixian_cli.py download 2012.04.04 - python lixian_cli.py download 0 1 2 - python lixian_cli.py download 0-2 -''' - -list = '''python lixian_cli.py list - -list tasks on Xunlei cloud - -Options: - --completed Print only completed tasks. Default: no - --deleted Print only deleted tasks. Default: no - --expired Print only expired tasks. Default: no - --[no]-n Print task sequence number. Default: no - --[no]-id Print task id. Default: yes - --[no]-name Print task name. Default: yes - --[no]-status Print task status. Default: yes - --[no]-size Print task size. Default: no - --[no]-progress Print task progress (in percent). Default: no - --[no]-speed Print task speed. Default: no - --[no]-date Print the date task added. Default: no - --[no]-original-url Print the original URL. Default: no - --[no]-download-url Print the download URL used to download from Xunlei cloud. Default: no - --[no]-format-size Print file size in human readable format. Default: no - --[no]-colors Colorful output. Default: yes - -Examples: - python lixian_cli.py list - python lixian_cli.py list id - python lixian_cli.py list bt-task-id/ - python lixian_cli.py list --completed - python lixian_cli.py list --completed --name --original-url --download-url --no-status --no-id - python lixian_cli.py list --deleted - python lixian_cli.py list --expired - python lixian_cli.py list id1 id2 - python lixian_cli.py list zip rar - python lixian_cli.py list 2012.04.04 2012.04.05 -''' - -add = '''python lixian_cli.py add [options] url... - -add tasks to Xunlei cloud - -Options: - --input=[file] Download URLs found in file. - --torrent --bt Treat all arguments as torrent files (e.g. local torrent file, torrent http url, torrent info hash) - Default: false. - -Examples: - python lixian_cli.py add url - python lixian_cli.py add 1.torrent - python lixian_cli.py add torrent-info-hash - python lixian_cli.py add --bt http://xxx/xxx.torrent -''' - -delete = '''python lixian_cli.py delete [options] [id|url|filename|keyword|date]... - -delete tasks from Xunlei cloud - -Options: - -i prompt before delete - --all delete all tasks if there are multiple matches - -Examples: - python lixian_cli.py delete task-id - python lixian_cli.py delete url - python lixian_cli.py delete file-name-on-cloud-to-delete -''' - -pause = '''python lixian_cli.py pause [options] [id|url|filename|keyword|date]... - -pause tasks on Xunlei cloud - -Options: - -i prompt before pausing tasks - --all pause all tasks if there are multiple matches -''' - -restart = '''python lixian_cli.py restart [id|url|filename|keyword|date]... - -restart tasks on Xunlei cloud - -Options: - -i prompt before restart - --all restart all tasks if there are multiple matches -''' - -rename = '''python lixian_cli.py rename task-id task-name - -rename task -''' - -readd = '''python lixian_cli.py readd [--deleted|--expired] task-id... - -re-add deleted/expired tasks - -Options: - --deleted re-add deleted tasks - --expired re-add expired tasks -''' - -config = '''python lixian_cli.py config key [value] - -save configuration so you don't have to repeat it - -Examples: - python lixian_cli.py config username "your xunlei id" - python lixian_cli.py config password "your xunlei password" - python lixian_cli.py config continue -''' - -info = '''python lixian_cli.py info - -print user id, internal user id, and gdriveid - -Options: - --id -i print user id only -''' - -logout = '''python lixian_cli.py logout - -logout from Xunlei cloud -''' - - diff --git a/xunlei-lixian/lixian_logging.py b/xunlei-lixian/lixian_logging.py deleted file mode 100644 index 2ad85e7..0000000 --- a/xunlei-lixian/lixian_logging.py +++ /dev/null @@ -1,91 +0,0 @@ - -__all__ = ['init_logger', 'get_logger'] - -import logging - -INFO = logging.INFO -DEBUG = logging.DEBUG -TRACE = 1 - -def file_logger(path, level): - import os.path - path = os.path.expanduser(path) - - logger = logging.getLogger('lixian') - logger.setLevel(min(level, DEBUG)) # if file log is enabled, always log debug message - - handler = logging.FileHandler(filename=path, ) - handler.setFormatter(logging.Formatter('%(asctime)s %(message)s')) - - logger.addHandler(handler) - - return logger - -class ConsoleLogger: - def __init__(self, level=INFO): - self.level = level - def stdout(self, message): - print message - def info(self, message): - if self.level <= INFO: - print message - def debug(self, message): - if self.level <= DEBUG: - print message - def trace(self, message): - pass - -class FileLogger: - def __init__(self, path, level=INFO, file_level=None, console_level=None): - console_level = console_level or level - file_level = file_level or level - self.console = ConsoleLogger(console_level) - self.logger = file_logger(path, file_level) - def stdout(self, message): - self.console.stdout(message) - def info(self, message): - self.console.info(message) - self.logger.info(message) - def debug(self, message): - self.console.debug(message) - self.logger.debug(message) - def trace(self, message): - self.logger.log(level=TRACE, msg=message) - -default_logger = None - -def init_logger(use_colors=True, level=INFO, path=None): - global default_logger - if not default_logger: - if isinstance(level, int): - assert level in (INFO, DEBUG, TRACE) - console_level = level - file_level = level - elif isinstance(level, basestring): - level = level.lower() - if level in ('info', 'debug', 'trace'): - level = {'info': INFO, 'debug': DEBUG, 'trace': TRACE}[level] - console_level = level - file_level = level - else: - console_level = INFO - file_level = DEBUG - for level in level.split(','): - device, level = level.split(':') - if device == 'console': - console_level = {'info': INFO, 'debug': DEBUG, 'trace': TRACE}[level] - elif device == 'file': - file_level = {'info': INFO, 'debug': DEBUG, 'trace': TRACE}[level] - else: - raise NotImplementedError('Invalid logging level: ' + device) - else: - raise NotImplementedError(type(level)) - if path: - default_logger = FileLogger(path, console_level=console_level, file_level=file_level) - else: - default_logger = ConsoleLogger(console_level) - -def get_logger(): - init_logger() - return default_logger - diff --git a/xunlei-lixian/lixian_nodes.py b/xunlei-lixian/lixian_nodes.py deleted file mode 100644 index e25ef00..0000000 --- a/xunlei-lixian/lixian_nodes.py +++ /dev/null @@ -1,149 +0,0 @@ - -import lixian_logging - -import urllib2 -import re - -VOD_RANGE = '0-50' - -def resolve_node_url(url, gdriveid, timeout=60): - request = urllib2.Request(url, headers={'Cookie': 'gdriveid=' + gdriveid}) - response = urllib2.urlopen(request, timeout=timeout) - response.close() - return response.geturl() - -def switch_node_in_url(node_url, node): - return re.sub(r'(http://)(vod\d+)(\.t\d+\.lixian\.vip\.xunlei\.com)', r'\1%s\3' % node, node_url) - - -def switch_node(url, node, gdriveid): - assert re.match(r'^vod\d+$', node) - logger = lixian_logging.get_logger() - logger.debug('Download URL: ' + url) - try: - url = resolve_node_url(url, gdriveid, timeout=60) - logger.debug('Resolved URL: ' + url) - except: - import traceback - logger.debug(traceback.format_exc()) - return url - url = switch_node_in_url(url, node) - logger.debug('Switch to node URL: ' + url) - return url - -def test_response_speed(response, max_size, max_duration): - import time - current_duration = 0 - current_size = 0 - start = time.clock() - while current_duration < max_duration and current_size < max_size: - data = response.read(max_size - current_size) - if not data: - # print "End of file" - break - current_size += len(data) - end = time.clock() - current_duration = end - start - if current_size < 1024: - raise Exception("Sample too small: %d" % current_size) - return current_size / current_duration, current_size, current_duration - - -def get_node_url_speed(url, gdriveid): - request = urllib2.Request(url, headers={'Cookie': 'gdriveid=' + gdriveid}) - response = urllib2.urlopen(request, timeout=3) - speed, size, duration = test_response_speed(response, 2*1000*1000, 3) - response.close() - return speed - - -def parse_vod_nodes(vod_nodes): - if vod_nodes == 'all' or not vod_nodes: - vod_nodes = VOD_RANGE - nodes = [] - # remove duplicate nodes - seen = set() - def add(node): - if node not in seen: - nodes.append(node) - seen.add(node) - for expr in re.split(r'\s*,\s*', vod_nodes): - if re.match(r'^\d+-\d+$', expr): - start, end = map(int, expr.split('-')) - if start <= end: - for i in range(start, end + 1): - add("vod%d" % i) - else: - for i in range(start, end - 1, -1): - add("vod%d" % i) - elif re.match(r'^\d+$', expr): - add('vod'+expr) - else: - raise Exception("Invalid vod expr: " + expr) - return nodes - -def get_best_node_url_from(node_url, nodes, gdriveid): - best = None - best_speed = 0 - logger = lixian_logging.get_logger() - for node in nodes: - url = switch_node_in_url(node_url, node) - try: - speed = get_node_url_speed(url, gdriveid) - logger.debug("%s speed: %s" % (node, speed)) - if speed > best_speed: - best_speed = speed - best = url - except Exception, e: - logger.debug("%s error: %s" % (node, e)) - return best - -def get_good_node_url_from(node_url, nodes, acceptable_speed, gdriveid): - best = None - best_speed = 0 - logger = lixian_logging.get_logger() - for node in nodes: - url = switch_node_in_url(node_url, node) - try: - speed = get_node_url_speed(url, gdriveid) - logger.debug("%s speed: %s" % (node, speed)) - if speed > acceptable_speed: - return url - elif speed > best_speed: - best_speed = speed - best = url - except Exception, e: - logger.debug("%s error: %s" % (node, e)) - return best - -def use_node_by_policy(url, vod_nodes, gdriveid, policy): - nodes = parse_vod_nodes(vod_nodes) - assert nodes - logger = lixian_logging.get_logger() - logger.debug('Download URL: ' + url) - try: - node_url = resolve_node_url(url, gdriveid, timeout=60) - logger.debug('Resolved URL: ' + node_url) - except: - import traceback - logger.debug(traceback.format_exc()) - return url - default_node = re.match(r'http://(vod\d+)\.', node_url).group(1) - if default_node not in nodes: - nodes.insert(0, default_node) - chosen = policy(node_url, nodes, gdriveid) - if chosen: - logger.debug('Switch to URL: ' + chosen) - return chosen - else: - return node_url - - -def use_fastest_node(url, vod_nodes, gdriveid): - return use_node_by_policy(url, vod_nodes, gdriveid, get_best_node_url_from) - -def use_fast_node(url, vod_nodes, acceptable_speed, gdriveid): - def policy(url, vod_nodes, gdriveid): - return get_good_node_url_from(url, vod_nodes, acceptable_speed, gdriveid) - return use_node_by_policy(url, vod_nodes, gdriveid, policy) - diff --git a/xunlei-lixian/lixian_plugins/__init__.py b/xunlei-lixian/lixian_plugins/__init__.py deleted file mode 100644 index 9d2ee87..0000000 --- a/xunlei-lixian/lixian_plugins/__init__.py +++ /dev/null @@ -1,20 +0,0 @@ - -def load_plugins_at(dir): - import os - import os.path - import re - home = os.path.dirname(os.path.dirname(__file__)) - plugin_dir = os.path.join(home, dir.replace('.', '/')) - plugins = os.listdir(plugin_dir) - plugins = [re.sub(r'\.py$', '', p) for p in plugins if re.match(r'^[a-zA-Z]\w*\.py$', p)] - for p in plugins: - __import__(dir + '.' + p) - -def load_plugins(): - load_plugins_at('lixian_plugins.commands') - load_plugins_at('lixian_plugins.queries') - load_plugins_at('lixian_plugins.filters') - load_plugins_at('lixian_plugins.parsers') - load_plugins_at('lixian_plugins') - -load_plugins() diff --git a/xunlei-lixian/lixian_plugins/api/__init__.py b/xunlei-lixian/lixian_plugins/api/__init__.py deleted file mode 100644 index ad710dd..0000000 --- a/xunlei-lixian/lixian_plugins/api/__init__.py +++ /dev/null @@ -1,68 +0,0 @@ - -__all__ = ['command', 'register_alias', - 'user_query', 'extract_info_hash_from_url', 'download_torrent_from_url', - 'task_filter', 'name_filter', - 'page_parser'] - -################################################## -# commands -################################################## - -from lixian_plugins.commands import command - -################################################## -# commands -################################################## - -from lixian_alias import register_alias - -################################################## -# queries -################################################## - -from lixian_query import user_query - -def extract_info_hash_from_url(regexp): - import lixian_queries - import re - @user_query - def processor(base, x): - m = re.match(regexp, x) - if m: - return lixian_queries.BtHashQuery(base, m.group(1)) - -def download_torrent_from_url(regexp): - import lixian_queries - import re - @user_query - def processor(base, x): - if re.match(regexp, x): - return lixian_queries.bt_url_processor(base, x) - -################################################## -# filters -################################################## - -from lixian_plugins.filters import task_filter -from lixian_plugins.filters import name_filter - -################################################## -# parsers -################################################## - -def page_parser(pattern): - def f(extend_links): - import lixian_plugins.parsers - patterns = pattern if type(pattern) is list else [pattern] - for p in patterns: - lixian_plugins.parsers.register_parser(p, extend_links) - return f - - -################################################## -# download tools -################################################## - -from lixian_download_tools import download_tool - - diff --git a/xunlei-lixian/lixian_plugins/commands/__init__.py b/xunlei-lixian/lixian_plugins/commands/__init__.py deleted file mode 100644 index 894b084..0000000 --- a/xunlei-lixian/lixian_plugins/commands/__init__.py +++ /dev/null @@ -1,36 +0,0 @@ - -__all__ = [] - -commands = {} - -extended_commands = [] - -def update_helps(commands): - if commands: - import lixian_help - lixian_help.extended_usage = '''\nExtended commands: -''' + lixian_help.join_commands([(x[0], x[1]) for x in commands]) - - for name, usage, doc in commands: - setattr(lixian_help, name, doc) - -def register_command(command): - extended_commands.append(command) - global commands - commands = dict((x.command_name, x) for x in extended_commands) - update_helps(sorted((x.command_name, x.command_usage, x.command_help) for x in extended_commands)) - - -def command(name='', usage='', help=''): - def as_command(f): - assert usage, 'missing command usage: ' + f.func_name - f.command_name = name or f.func_name.replace('_', '-') - f.command_usage = usage - f.command_help = help or f.func_doc - import textwrap - if f.command_help: - f.command_help = textwrap.dedent(f.command_help) - register_command(f) - return f - return as_command - diff --git a/xunlei-lixian/lixian_plugins/commands/aria2.py b/xunlei-lixian/lixian_plugins/commands/aria2.py deleted file mode 100644 index 5f5e28f..0000000 --- a/xunlei-lixian/lixian_plugins/commands/aria2.py +++ /dev/null @@ -1,91 +0,0 @@ - -from lixian_plugins.api import command - -from lixian_config import * -from lixian_encoding import default_encoding -from lixian_cli_parser import command_line_parser -from lixian_cli_parser import with_parser -from lixian_cli_parser import command_line_option, command_line_value -from lixian_commands.util import parse_login, create_client - -def export_aria2_conf(args): - client = create_client(args) - import lixian_query - tasks = lixian_query.search_tasks(client, args) - files = [] - for task in tasks: - if task['type'] == 'bt': - subs, skipped, single_file = lixian_query.expand_bt_sub_tasks(task) - if not subs: - continue - if single_file: - files.append((subs[0]['xunlei_url'], subs[0]['name'], None)) - else: - for f in subs: - files.append((f['xunlei_url'], f['name'], task['name'])) - else: - files.append((task['xunlei_url'], task['name'], None)) - output = '' - for url, name, dir in files: - if type(url) == unicode: - url = url.encode(default_encoding) - output += url + '\n' - output += ' out=' + name.encode(default_encoding) + '\n' - if dir: - output += ' dir=' + dir.encode(default_encoding) + '\n' - output += ' header=Cookie: gdriveid=' + client.get_gdriveid() + '\n' - return output - -@command(usage='export task download urls as aria2 format') -@command_line_parser() -@with_parser(parse_login) -@command_line_option('all') -def export_aria2(args): - ''' - usage: lx export-aria2 [id|name]... - ''' - print export_aria2_conf(args) - -def download_aria2_stdin(aria2_conf, j): - aria2_opts = ['aria2c', '-i', '-', '-j', j] - aria2_opts.extend(get_config('aria2-opts', '').split()) - from subprocess import Popen, PIPE - sub = Popen(aria2_opts, stdin=PIPE, bufsize=1, shell=True) - sub.communicate(aria2_conf) - sub.stdin.close() - exit_code = sub.wait() - if exit_code != 0: - raise Exception('aria2c exited abnormaly') - -def download_aria2_temp(aria2_conf, j): - import tempfile - temp = tempfile.NamedTemporaryFile('w', delete=False) - temp.file.write(aria2_conf) - temp.file.close() - try: - aria2_opts = ['aria2c', '-i', temp.name, '-j', j] - aria2_opts.extend(get_config('aria2-opts', '').split()) - import subprocess - exit_code = subprocess.call(aria2_opts) - finally: - import os - os.unlink(temp.name) - if exit_code != 0: - raise Exception('aria2c exited abnormaly') - -@command(usage='concurrently download tasks in aria2') -@command_line_parser() -@with_parser(parse_login) -@command_line_option('all') -@command_line_value('max-concurrent-downloads', alias='j', default=get_config('aria2-j', '5')) -def download_aria2(args): - ''' - usage: lx download-aria2 -j 5 [id|name]... - ''' - aria2_conf = export_aria2_conf(args) - import platform - if platform.system() == 'Windows': - download_aria2_temp(aria2_conf, args.max_concurrent_downloads) - else: - download_aria2_stdin(aria2_conf, args.max_concurrent_downloads) - diff --git a/xunlei-lixian/lixian_plugins/commands/decode_url.py b/xunlei-lixian/lixian_plugins/commands/decode_url.py deleted file mode 100644 index f83c06c..0000000 --- a/xunlei-lixian/lixian_plugins/commands/decode_url.py +++ /dev/null @@ -1,12 +0,0 @@ - -from lixian_plugins.api import command - -@command(usage='convert thunder:// (and more) to normal url') -def decode_url(args): - ''' - usage: lx decode-url thunder://... - ''' - from lixian_url import url_unmask - for x in args: - print url_unmask(x) - diff --git a/xunlei-lixian/lixian_plugins/commands/diagnostics.py b/xunlei-lixian/lixian_plugins/commands/diagnostics.py deleted file mode 100644 index 5ff7294..0000000 --- a/xunlei-lixian/lixian_plugins/commands/diagnostics.py +++ /dev/null @@ -1,16 +0,0 @@ - -from lixian_plugins.api import command - -@command(name='diagnostics', usage='print helpful information for diagnostics') -def lx_diagnostics(args): - ''' - usage: lx diagnostics - ''' - from lixian_encoding import default_encoding - print 'default_encoding ->', default_encoding - import sys - print 'sys.getdefaultencoding() ->', sys.getdefaultencoding() - print 'sys.getfilesystemencoding() ->', sys.getfilesystemencoding() - print r"print u'\u4e2d\u6587'.encode('utf-8') ->", u'\u4e2d\u6587'.encode('utf-8') - print r"print u'\u4e2d\u6587'.encode('gbk') ->", u'\u4e2d\u6587'.encode('gbk') - diff --git a/xunlei-lixian/lixian_plugins/commands/echo.py b/xunlei-lixian/lixian_plugins/commands/echo.py deleted file mode 100644 index d4abf9d..0000000 --- a/xunlei-lixian/lixian_plugins/commands/echo.py +++ /dev/null @@ -1,11 +0,0 @@ - -from lixian_plugins.api import command - -@command(usage='echo arguments') -def echo(args): - ''' - lx echo ... - ''' - import lixian_cli_parser - print ' '.join(lixian_cli_parser.expand_command_line(args)) - diff --git a/xunlei-lixian/lixian_plugins/commands/export_download_urls.py b/xunlei-lixian/lixian_plugins/commands/export_download_urls.py deleted file mode 100644 index 8c3dff2..0000000 --- a/xunlei-lixian/lixian_plugins/commands/export_download_urls.py +++ /dev/null @@ -1,37 +0,0 @@ - -from lixian_plugins.api import command - - -from lixian_cli_parser import command_line_parser -from lixian_cli_parser import with_parser -from lixian_cli_parser import command_line_option, command_line_value -from lixian_commands.util import parse_login, create_client - -@command(usage='export task download urls') -@command_line_parser() -@with_parser(parse_login) -@command_line_option('all') -@command_line_value('category') -def export_download_urls(args): - ''' - usage: lx export-download-urls [id|name]... - ''' - assert len(args) or args.all or args.category, 'Not enough arguments' - client = create_client(args) - import lixian_query - tasks = lixian_query.search_tasks(client, args) - urls = [] - for task in tasks: - if task['type'] == 'bt': - subs, skipped, single_file = lixian_query.expand_bt_sub_tasks(task) - if not subs: - continue - if single_file: - urls.append((subs[0]['xunlei_url'], subs[0]['name'], None)) - else: - for f in subs: - urls.append((f['xunlei_url'], f['name'], task['name'])) - else: - urls.append((task['xunlei_url'], task['name'], None)) - for url, _, _ in urls: - print url diff --git a/xunlei-lixian/lixian_plugins/commands/extend_links.py b/xunlei-lixian/lixian_plugins/commands/extend_links.py deleted file mode 100644 index c62ddf7..0000000 --- a/xunlei-lixian/lixian_plugins/commands/extend_links.py +++ /dev/null @@ -1,26 +0,0 @@ - -from lixian_plugins.api import command - -@command(usage='parse links') -def extend_links(args): - ''' - usage: lx extend-links http://kuai.xunlei.com/d/... http://www.verycd.com/topics/... - - parse and print links from pages - - lx extend-links urls... - lx extend-links --name urls... - ''' - - from lixian_cli_parser import parse_command_line - from lixian_encoding import default_encoding - - args = parse_command_line(args, [], ['name']) - import lixian_plugins.parsers - if args.name: - for x in lixian_plugins.parsers.extend_links_name(args): - print x.encode(default_encoding) - else: - for x in lixian_plugins.parsers.extend_links(args): - print x - diff --git a/xunlei-lixian/lixian_plugins/commands/get_torrent.py b/xunlei-lixian/lixian_plugins/commands/get_torrent.py deleted file mode 100644 index ed2c202..0000000 --- a/xunlei-lixian/lixian_plugins/commands/get_torrent.py +++ /dev/null @@ -1,44 +0,0 @@ - -from lixian_plugins.api import command - -from lixian_cli_parser import command_line_parser, command_line_option -from lixian_cli_parser import with_parser -from lixian_cli import parse_login -from lixian_commands.util import create_client - -@command(name='get-torrent', usage='get .torrent by task id or info hash') -@command_line_parser() -@with_parser(parse_login) -@command_line_option('rename', default=True) -def get_torrent(args): - ''' - usage: lx get-torrent [info-hash|task-id]... - ''' - client = create_client(args) - for id in args: - id = id.lower() - import re - if re.match(r'[a-fA-F0-9]{40}$', id): - torrent = client.get_torrent_file_by_info_hash(id) - elif re.match(r'\d+$', id): - import lixian_query - task = lixian_query.get_task_by_id(client, id) - id = task['bt_hash'] - id = id.lower() - torrent = client.get_torrent_file_by_info_hash(id) - else: - raise NotImplementedError() - if args.rename: - import lixian_hash_bt - from lixian_encoding import default_encoding - info = lixian_hash_bt.bdecode(torrent)['info'] - name = info['name'].decode(info.get('encoding', 'utf-8')).encode(default_encoding) - import re - name = re.sub(r'[\\/:*?"<>|]', '-', name) - else: - name = id - path = name + '.torrent' - print path - with open(path, 'wb') as output: - output.write(torrent) - diff --git a/xunlei-lixian/lixian_plugins/commands/hash.py b/xunlei-lixian/lixian_plugins/commands/hash.py deleted file mode 100644 index 0a089ef..0000000 --- a/xunlei-lixian/lixian_plugins/commands/hash.py +++ /dev/null @@ -1,27 +0,0 @@ - -from lixian_plugins.api import command - -@command(name='hash', usage='compute hashes') -def print_hash(args): - ''' - lx hash --sha1 file... - lx hash --md5 file... - lx hash --md4 file... - lx hash --dcid file... - lx hash --ed2k file... - lx hash --info-hash xxx.torrent... - lx hash --verify-sha1 file hash - lx hash --verify-md5 file hash - lx hash --verify-md4 file hash - lx hash --verify-dcid file hash - lx hash --verify-ed2k file ed2k://... - lx hash --verify-bt file xxx.torrent - ''' - #assert len(args) == 1 - import lixian_hash - #import lixian_hash_ed2k - #print 'ed2k:', lixian_hash_ed2k.hash_file(args[0]) - #print 'dcid:', lixian_hash.dcid_hash_file(args[0]) - import lixian_cli_parser - lixian_hash.main(lixian_cli_parser.expand_command_line(args)) - diff --git a/xunlei-lixian/lixian_plugins/commands/kuai.py b/xunlei-lixian/lixian_plugins/commands/kuai.py deleted file mode 100644 index ad75d83..0000000 --- a/xunlei-lixian/lixian_plugins/commands/kuai.py +++ /dev/null @@ -1,16 +0,0 @@ - -from lixian_plugins.api import command - -@command(usage='parse links from kuai.xunlei.com') -def kuai(args): - ''' - usage: lx kuai http://kuai.xunlei.com/d/xxx... - - Note that you can simply use: - lx add http://kuai.xunlei.com/d/xxx... - or: - lx download http://kuai.xunlei.com/d/xxx... - ''' - import lixian_kuai - lixian_kuai.main(args) - diff --git a/xunlei-lixian/lixian_plugins/commands/list_torrent.py b/xunlei-lixian/lixian_plugins/commands/list_torrent.py deleted file mode 100644 index 44876bc..0000000 --- a/xunlei-lixian/lixian_plugins/commands/list_torrent.py +++ /dev/null @@ -1,65 +0,0 @@ - -from lixian_plugins.api import command - -from lixian_cli_parser import parse_command_line -from lixian_config import get_config -from lixian_encoding import default_encoding - -def b_encoding(b): - if 'encoding' in b: - return b['encoding'] - if 'codepage' in b: - return 'cp' + str(b['codepage']) - return 'utf-8' - -def b_name(info, encoding='utf-8'): - if 'name.utf-8' in info: - return info['name.utf-8'].decode('utf-8') - return info['name'].decode(encoding) - -def b_path(f, encoding='utf-8'): - if 'path.utf-8' in f: - return [p.decode('utf-8') for p in f['path.utf-8']] - return [p.decode(encoding) for p in f['path']] - -@command(usage='list files in local .torrent') -def list_torrent(args): - ''' - usage: lx list-torrent [--size] xxx.torrent... - ''' - args = parse_command_line(args, [], ['size'], default={'size':get_config('size')}) - torrents = args - if not torrents: - from glob import glob - torrents = glob('*.torrent') - if not torrents: - raise Exception('No .torrent file found') - for p in torrents: - with open(p, 'rb') as stream: - from lixian_hash_bt import bdecode - b = bdecode(stream.read()) - encoding = b_encoding(b) - info = b['info'] - from lixian_util import format_size - if args.size: - size = sum(f['length'] for f in info['files']) if 'files' in info else info['length'] - print '*', b_name(info, encoding).encode(default_encoding), format_size(size) - else: - print '*', b_name(info, encoding).encode(default_encoding) - if 'files' in info: - for f in info['files']: - if f['path'][0].startswith('_____padding_file_'): - continue - path = '/'.join(b_path(f, encoding)).encode(default_encoding) - if args.size: - print '%s (%s)' % (path, format_size(f['length'])) - else: - print path - else: - path = b_name(info, encoding).encode(default_encoding) - if args.size: - from lixian_util import format_size - print '%s (%s)' % (path, format_size(info['length'])) - else: - print path - diff --git a/xunlei-lixian/lixian_plugins/commands/speed_test.py b/xunlei-lixian/lixian_plugins/commands/speed_test.py deleted file mode 100644 index fa3062d..0000000 --- a/xunlei-lixian/lixian_plugins/commands/speed_test.py +++ /dev/null @@ -1,96 +0,0 @@ -from lixian_plugins.api import command - - -from lixian_cli_parser import command_line_parser -from lixian_cli_parser import with_parser -from lixian_cli_parser import command_line_option, command_line_value -from lixian_commands.util import parse_login, parse_colors, create_client -from lixian_config import get_config - -from lixian_encoding import default_encoding -from lixian_colors import colors - -import lixian_nodes - -@command(usage='test download speed from multiple vod nodes') -@command_line_parser() -@with_parser(parse_login) -@with_parser(parse_colors) -@command_line_value('vod-nodes', default=get_config('vod-nodes', lixian_nodes.VOD_RANGE)) -def speed_test(args): - ''' - usage: lx speed_test [--vod-nodes=0-50] [id|name] - ''' - assert len(args) - client = create_client(args) - import lixian_query - tasks = lixian_query.search_tasks(client, args) - if not tasks: - raise Exception('No task found') - task = tasks[0] - urls = [] - if task['type'] == 'bt': - subs, skipped, single_file = lixian_query.expand_bt_sub_tasks(task) - if not subs: - raise Exception('No files found') - subs = [f for f in subs if f['size'] > 1000*1000] or subs # skip files with length < 1M - if single_file: - urls.append((subs[0]['xunlei_url'], subs[0]['name'], None)) - else: - for f in subs: - urls.append((f['xunlei_url'], f['name'], task['name'])) - else: - urls.append((task['xunlei_url'], task['name'], None)) - url, filename, dirname = urls[0] - name = dirname + '/' + filename if dirname else filename - test_file(client, url, name, args) - -def test_file(client, url, name, options): - with colors(options.colors).cyan(): - print name.encode(default_encoding) - # print 'File:', name.encode(default_encoding) - # print 'Address:', url - node_url = lixian_nodes.resolve_node_url(url, client.get_gdriveid(), timeout=3) - # print 'Node:', node_url - test_nodes(node_url, client.get_gdriveid(), options) - -def test_nodes(node_url, gdriveid, options): - nodes = lixian_nodes.parse_vod_nodes(options.vod_nodes) - best = None - best_speed = 0 - for node in nodes: - # print 'Node:', node - url = lixian_nodes.switch_node_in_url(node_url, node) - try: - speed = lixian_nodes.get_node_url_speed(url, gdriveid) - if best_speed < speed: - best = node - best_speed = speed - kb = int(speed/1000) - # print 'Speed: %dKB/s' % kb, '.' * (kb /100) - show_node_speed(node, kb, options) - except Exception, e: - show_node_error(node, e, options) - if best: - with colors(options.colors).green(): - print best, - print "is the fastest node!" - -def show_node_speed(node, kb, options): - node = "%-5s " % node - speed = '%dKB/s' % kb - bar = '.' * (kb /100) - whitespaces = ' ' * (79 - len(node) - len(bar) - len(speed)) - if kb >= 1000: - with colors(options.colors).green(): - # print node + bar + whitespaces + speed - with colors(options.colors).bold(): - print node[:-1], - print bar + whitespaces + speed - else: - print node + bar + whitespaces + speed - -def show_node_error(node, e, options): - with colors(options.colors).red(): - print "%-5s %s" % (node, e) - diff --git a/xunlei-lixian/lixian_plugins/filters/__init__.py b/xunlei-lixian/lixian_plugins/filters/__init__.py deleted file mode 100644 index de170b3..0000000 --- a/xunlei-lixian/lixian_plugins/filters/__init__.py +++ /dev/null @@ -1,67 +0,0 @@ - -import re - -name_filters = {} -task_filters = {} - -def find_matcher(keyword, filters): - for p in filters: - if re.search(p, keyword): - return filters[p] - -def has_task_filter(keyword): - return bool(find_matcher(keyword, task_filters)) - -def filter_tasks_with_matcher(tasks, keyword, (mode, m)): - if mode == 'single': - return filter(lambda x: m(keyword, x), tasks) - elif mode == 'batch': - return m(keyword, tasks) - else: - raise NotImplementedError(mode) - -def filter_tasks(tasks, keyword): - m = find_matcher(keyword, task_filters) - if m: - return filter_tasks_with_matcher(tasks, keyword, m) - -def filter_things(things, keyword): - if not things: - # XXX: neither None or things should be OK - return things - assert len(set(map(type, things))) == 1 - filters = task_filters if type(things[0]) == dict else name_filters - m = find_matcher(keyword, filters) - if m: - return filter_tasks_with_matcher(things, keyword, m) - -def define_task_filter(pattern, matcher, batch=False): - task_filters[pattern] = ('batch' if batch else 'single', matcher) - -def define_name_filter(pattern, matcher): - name_filters[pattern] = ('single', matcher) - task_filters[pattern] = ('single', lambda k, x: matcher(k, x['name'])) - -def task_filter(pattern=None, protocol=None, batch=False): - assert bool(pattern) ^ bool(protocol) - def define_filter(matcher): - if pattern: - define_task_filter(pattern, matcher, batch) - else: - assert re.match(r'^[\w-]+$', protocol), protocol - define_task_filter(r'^%s:' % protocol, lambda k, x: matcher(re.sub(r'^[\w-]+:', '', k), x), batch) - return matcher - return define_filter - -def name_filter(pattern=None, protocol=None): - # FIXME: duplicate code - assert bool(pattern) ^ bool(protocol) - def define_filter(matcher): - if pattern: - define_name_filter(pattern, matcher) - else: - assert re.match(r'^\w+$', protocol), protocol - define_name_filter(r'^%s:' % protocol, lambda k, x: matcher(re.sub(r'^\w+:', '', k), x)) - return matcher - return define_filter - diff --git a/xunlei-lixian/lixian_plugins/filters/date.py b/xunlei-lixian/lixian_plugins/filters/date.py deleted file mode 100644 index be9f476..0000000 --- a/xunlei-lixian/lixian_plugins/filters/date.py +++ /dev/null @@ -1,7 +0,0 @@ - -from lixian_plugins.api import task_filter - -@task_filter(pattern=r'^\d{4}[-.]\d{2}[-.]\d{2}$') -def filter_by_date(keyword, task): - return task['date'] == keyword.replace('-', '.') - diff --git a/xunlei-lixian/lixian_plugins/filters/name.py b/xunlei-lixian/lixian_plugins/filters/name.py deleted file mode 100644 index 803ee37..0000000 --- a/xunlei-lixian/lixian_plugins/filters/name.py +++ /dev/null @@ -1,7 +0,0 @@ - -from lixian_plugins.api import name_filter - -@name_filter(protocol='name') -def filter_by_raw_text(keyword, name): - return keyword.lower() in name.lower() - diff --git a/xunlei-lixian/lixian_plugins/filters/raw.py b/xunlei-lixian/lixian_plugins/filters/raw.py deleted file mode 100644 index 863d32a..0000000 --- a/xunlei-lixian/lixian_plugins/filters/raw.py +++ /dev/null @@ -1,7 +0,0 @@ - -from lixian_plugins.api import name_filter - -@name_filter(protocol='raw') -def filter_by_raw_text(keyword, name): - return keyword.lower() in name.lower() - diff --git a/xunlei-lixian/lixian_plugins/filters/regexp.py b/xunlei-lixian/lixian_plugins/filters/regexp.py deleted file mode 100644 index f9fd722..0000000 --- a/xunlei-lixian/lixian_plugins/filters/regexp.py +++ /dev/null @@ -1,8 +0,0 @@ - -from lixian_plugins.api import name_filter - -import re - -@name_filter(protocol='regexp') -def filter_by_regexp(keyword, name): - return re.search(keyword, name) diff --git a/xunlei-lixian/lixian_plugins/filters/size.py b/xunlei-lixian/lixian_plugins/filters/size.py deleted file mode 100644 index c08d516..0000000 --- a/xunlei-lixian/lixian_plugins/filters/size.py +++ /dev/null @@ -1,23 +0,0 @@ - -from lixian_plugins.api import task_filter - -import re - -@task_filter(protocol='size') -def filter_by_size(keyword, task): - ''' - Example: - lx download size:10m- - lx download size:1G+ - lx download 0/size:1g- - ''' - m = re.match(r'^([<>])?(\d+(?:\.\d+)?)([GM])?([+-])?$', keyword, flags=re.I) - assert m, keyword - less_or_great, n, u, less_or_more = m.groups() - assert bool(less_or_great) ^ bool(less_or_more), 'must bt size, size-, or size+' - size = float(n) * {None: 1, 'G': 1000**3, 'g': 1000**3, 'M': 1000**2, 'm': 1000**2}[u] - if less_or_great == '<' or less_or_more == '-': - return task['size'] < size - else: - return task['size'] > size - diff --git a/xunlei-lixian/lixian_plugins/filters/sort.py b/xunlei-lixian/lixian_plugins/filters/sort.py deleted file mode 100644 index fce958f..0000000 --- a/xunlei-lixian/lixian_plugins/filters/sort.py +++ /dev/null @@ -1,11 +0,0 @@ - -from lixian_plugins.api import task_filter - -@task_filter(protocol='sort', batch=True) -def sort_by_name(keyword, tasks): - ''' - Example: - lx list sort: - lx download 0/sort:/[0-1] - ''' - return sorted(tasks, key=lambda x: x['name']) diff --git a/xunlei-lixian/lixian_plugins/filters/total_size.py b/xunlei-lixian/lixian_plugins/filters/total_size.py deleted file mode 100644 index f224cbc..0000000 --- a/xunlei-lixian/lixian_plugins/filters/total_size.py +++ /dev/null @@ -1,27 +0,0 @@ - -from lixian_plugins.api import task_filter - -import re - -@task_filter(protocol='total-size', batch=True) -def fetch_by_total_size(keyword, tasks): - ''' - Example: - lx download total_size:1g - lx download 0/total_size:1g - lx list total_size:1g - ''' - m = re.match(r'^(\d+(?:\.\d+)?)([GM])?$', keyword, flags=re.I) - assert m, keyword - n, u = m.groups() - limit = float(n) * {None: 1, 'G': 1000**3, 'g': 1000**3, 'M': 1000**2, 'm': 1000**2}[u] - total = 0 - results = [] - for t in tasks: - total += t['size'] - if total <= limit: - results.append(t) - else: - return results - return results - diff --git a/xunlei-lixian/lixian_plugins/parsers/__init__.py b/xunlei-lixian/lixian_plugins/parsers/__init__.py deleted file mode 100644 index 092d947..0000000 --- a/xunlei-lixian/lixian_plugins/parsers/__init__.py +++ /dev/null @@ -1,69 +0,0 @@ - -import re - -page_parsers = {} - -def register_parser(site, extend_link): - page_parsers[site] = extend_link - - -def in_site(url, site): - if url.startswith(site): - return True - if '*' in site: - import fnmatch - p = fnmatch.translate(site) - return re.match(p, url) - -def find_parser(link): - for p in page_parsers: - if in_site(link, p): - return page_parsers[p] - - -def to_name(x): - if type(x) == dict: - return x['name'] - else: - return x - -def to_url(x): - if type(x) == dict: - return x['url'] - else: - return x - -def parse_pattern(link): - m = re.search(r'[^:]//', link) - if m: - u = link[:m.start()+1] - p = link[m.start()+3:] - assert '//' not in p, link - if p.endswith('/'): - u += '/' - p = p[:-1] - return u, p - -def try_to_extend_link(link): - parser = find_parser(link) - if parser: - x = parse_pattern(link) - if x: - links = parser(x[0]) - import lixian_filter_expr - return lixian_filter_expr.filter_expr(links, x[1]) - else: - return parser(link) - -def extend_link(link): - return try_to_extend_link(link) or [link] - -def extend_links_rich(links): - return sum(map(extend_link, links), []) - -def extend_links(links): - return map(to_url, extend_links_rich(links)) - -def extend_links_name(links): - return map(to_name, extend_links_rich(links)) - diff --git a/xunlei-lixian/lixian_plugins/parsers/icili.py b/xunlei-lixian/lixian_plugins/parsers/icili.py deleted file mode 100644 index 8ba801a..0000000 --- a/xunlei-lixian/lixian_plugins/parsers/icili.py +++ /dev/null @@ -1,19 +0,0 @@ - -from lixian_plugins.api import page_parser - -import urllib2 -import re - -def icili_links(url): - assert url.startswith('http://www.icili.com/emule/download/'), url - html = urllib2.urlopen(url).read() - table = re.search(r'.*?
    ', html, flags=re.S).group() - links = re.findall(r'value="(ed2k://[^"]+)"', table) - return links - -@page_parser('http://www.icili.com/emule/download/') -def extend_link(url): - links = icili_links(url) - from lixian_hash_ed2k import parse_ed2k_file - return [{'url':x, 'name':parse_ed2k_file(x)} for x in links] - diff --git a/xunlei-lixian/lixian_plugins/parsers/kuai.py b/xunlei-lixian/lixian_plugins/parsers/kuai.py deleted file mode 100644 index 7ed5219..0000000 --- a/xunlei-lixian/lixian_plugins/parsers/kuai.py +++ /dev/null @@ -1,52 +0,0 @@ - -from lixian_plugins.api import page_parser - -import urllib -import re - -def generate_lixian_url(info): - print info['url'] - info = dict(info) - info['namehex'] = '0102' - info['fid'] = re.search(r'fid=([^&]+)', info['url']).group(1) - info['tid'] = re.search(r'tid=([^&]+)', info['url']).group(1) - info['internalid'] = '111' - info['taskid'] = 'xxx' - return 'http://gdl.lixian.vip.xunlei.com/download?fid=%(fid)s&mid=666&threshold=150&tid=%(tid)s&srcid=4&verno=1&g=%(gcid)s&scn=t16&i=%(gcid)s&t=1&ui=%(internalid)s&ti=%(taskid)s&s=%(size)s&m=0&n=%(namehex)s' % info - -def parse_link(html): - attrs = dict(re.findall(r'(\w+)="([^"]+)"', html)) - if 'file_url' not in attrs: - return - keys = {'url': 'file_url', 'name':'file_name', 'size':'file_size', 'gcid':'gcid', 'cid':'cid', 'gcid_resid':'gcid_resid'} - info = {} - for k in keys: - info[k] = attrs[keys[k]] - #info['name'] = urllib.unquote(info['name']) - return info - -@page_parser('http://kuai.xunlei.com/d/') -def kuai_links(url): - assert url.startswith('http://kuai.xunlei.com/d/'), url - html = urllib.urlopen(url).read().decode('utf-8') - #return re.findall(r'file_url="([^"]+)"', html) - #return map(parse_link, re.findall(r'', html, flags=re.S)) - return filter(bool, map(parse_link, re.findall(r'.*?', html, flags=re.S))) - -extend_link = kuai_links - -def main(args): - from lixian_cli_parser import parse_command_line - args = parse_command_line(args, [], ['name']) - for x in args: - for v in kuai_links(x): - if args.name: - print v['name'] - else: - print v['url'] - - -if __name__ == '__main__': - import sys - main(sys.argv[1:]) - diff --git a/xunlei-lixian/lixian_plugins/parsers/qjwm.py b/xunlei-lixian/lixian_plugins/parsers/qjwm.py deleted file mode 100644 index 174f78e..0000000 --- a/xunlei-lixian/lixian_plugins/parsers/qjwm.py +++ /dev/null @@ -1,22 +0,0 @@ - -from lixian_plugins.api import page_parser - -import urllib2 -import re - -def qjwm_link(url): - assert re.match(r'http://.*\.qjwm\.com/down(load)?_\d+.html', url) - url = url.replace('/down_', '/download_') - html = urllib2.urlopen(url).read() - m = re.search(r'var thunder_url = "([^"]+)";', html) - if m: - url = m.group(1) - url = url.decode('gbk') - return url - - -@page_parser('http://*.qjwm.com/*') -def extend_link(url): - url = qjwm_link(url) - return url and [url] or [] - diff --git a/xunlei-lixian/lixian_plugins/parsers/simplecd.py b/xunlei-lixian/lixian_plugins/parsers/simplecd.py deleted file mode 100644 index d494655..0000000 --- a/xunlei-lixian/lixian_plugins/parsers/simplecd.py +++ /dev/null @@ -1,30 +0,0 @@ - -from lixian_plugins.api import page_parser - -import urllib2 -import re - - -def simplecd_links(url): - m = re.match(r'(http://(?:www\.)?s[ia]mplecd\.\w+/)(id|entry)/', url) - assert m, url - site = m.group(1) - html = urllib2.urlopen(url).read() - ids = re.findall(r'value="(\w+)"\s+name="selectemule"', html) - form = '&'.join('rid=' + id for id in ids) - q = 'mode=copy&' + form - html = urllib2.urlopen(site + 'download/?' + q).read() - table = re.search(r'', html, flags=re.S).group() - links = re.findall(r'ed2k://[^\s<>]+', table) - import lixian_url - return map(lixian_url.normalize_unicode_link, links) - -@page_parser(['http://simplecd.*/', - 'http://www.simplecd.*/', - 'http://samplecd.*/', - 'http://www.samplecd.*/']) -def extend_link(url): - links = simplecd_links(url) - from lixian_hash_ed2k import parse_ed2k_file - return [{'url':x, 'name':parse_ed2k_file(x)} for x in links] - diff --git a/xunlei-lixian/lixian_plugins/parsers/verycd.py b/xunlei-lixian/lixian_plugins/parsers/verycd.py deleted file mode 100644 index ab50696..0000000 --- a/xunlei-lixian/lixian_plugins/parsers/verycd.py +++ /dev/null @@ -1,21 +0,0 @@ - -from lixian_plugins.api import page_parser - -import urllib2 -import re - -def parse_links(html): - html = re.search(r'.*?', html, re.S).group() - links = re.findall(r'value="([^"]+)"', html) - return [x for x in links if x.startswith('ed2k://')] - -def verycd_links(url): - assert url.startswith('http://www.verycd.com/topics/'), url - return parse_links(urllib2.urlopen(url).read()) - -@page_parser('http://www.verycd.com/topics/') -def extend_link(url): - links = verycd_links(url) - from lixian_hash_ed2k import parse_ed2k_file - return [{'url':x, 'name':parse_ed2k_file(x)} for x in links] - diff --git a/xunlei-lixian/lixian_plugins/queries/__init__.py b/xunlei-lixian/lixian_plugins/queries/__init__.py deleted file mode 100644 index 139597f..0000000 --- a/xunlei-lixian/lixian_plugins/queries/__init__.py +++ /dev/null @@ -1,2 +0,0 @@ - - diff --git a/xunlei-lixian/lixian_plugins/queries/torrentz.py b/xunlei-lixian/lixian_plugins/queries/torrentz.py deleted file mode 100644 index be74ab2..0000000 --- a/xunlei-lixian/lixian_plugins/queries/torrentz.py +++ /dev/null @@ -1,5 +0,0 @@ - -from lixian_plugins.api import extract_info_hash_from_url - -extract_info_hash_from_url(r'^http://torrentz.eu/([0-9a-f]{40})$') - diff --git a/xunlei-lixian/lixian_progress.py b/xunlei-lixian/lixian_progress.py deleted file mode 100644 index f72ec27..0000000 --- a/xunlei-lixian/lixian_progress.py +++ /dev/null @@ -1,30 +0,0 @@ - -import sys - -class SimpleProgressBar: - def __init__(self): - self.displayed = False - def update(self, percent): - self.displayed = True - bar_size = 40 - percent *= 100.0 - if percent > 100: - percent = 100.0 - dots = int(bar_size * percent / 100) - plus = percent / 100 * bar_size - dots - if plus > 0.8: - plus = '=' - elif plus > 0.4: - plus = '-' - else: - plus = '' - percent = int(percent) - bar = '=' * dots + plus - bar = '{0:>3}%[{1:<40}]'.format(percent, bar) - sys.stdout.write('\r'+bar) - sys.stdout.flush() - def done(self): - if self.displayed: - print - self.displayed = False - diff --git a/xunlei-lixian/lixian_queries.py b/xunlei-lixian/lixian_queries.py deleted file mode 100644 index aa86ad1..0000000 --- a/xunlei-lixian/lixian_queries.py +++ /dev/null @@ -1,326 +0,0 @@ - -from lixian_query import ExactQuery -from lixian_query import SearchQuery -from lixian_query import query -from lixian_query import bt_query - -import lixian_hash_bt -import lixian_url -import lixian_encoding - -import re - -################################################## -# queries -################################################## - -class SingleTaskQuery(ExactQuery): - def __init__(self, base, t): - super(SingleTaskQuery, self).__init__(base) - self.id = t['id'] - - def query_once(self): - return [self.base.get_task_by_id(self.id)] - - def query_search(self): - t = self.base.find_task_by_id(self.id) - return [t] if t else [] - - -@query(priority=1) -@bt_query(priority=1) -def single_id_processor(base, x): - if not re.match(r'^\d+/?$', x): - return - n = x.rstrip('/') - t = base.find_task_by_id(n) - if t: - return SingleTaskQuery(base, t) - -################################################## - -class MultipleTasksQuery(ExactQuery): - def __init__(self, base, tasks): - super(MultipleTasksQuery, self).__init__(base) - self.tasks = tasks - - def query_once(self): - return map(self.base.get_task_by_id, (t['id'] for t in self.tasks)) - - def query_search(self): - return filter(bool, map(self.base.find_task_by_id, (t['id'] for t in self.tasks))) - -@query(priority=1) -@bt_query(priority=1) -def range_id_processor(base, x): - m = re.match(r'^(\d+)-(\d+)$', x) - if not m: - return - begin = int(m.group(1)) - end = int(m.group(2)) - tasks = base.get_tasks() - if begin <= end: - found = filter(lambda x: begin <= x['#'] <= end, tasks) - else: - found = reversed(filter(lambda x: end <= x['#'] <= begin, tasks)) - if found: - return MultipleTasksQuery(base, found) - -################################################## - -class SubTaskQuery(ExactQuery): - def __init__(self, base, t, subs): - super(SubTaskQuery, self).__init__(base) - self.task = t - self.subs = subs - - def query_once(self): - task = dict(self.base.get_task_by_id(self.task['id'])) - files = self.base.get_files(task) - task['files'] = self.subs - return [task] - - def query_search(self): - task = self.base.find_task_by_id(self.task['id']) - if not task: - return [] - task = dict(task) - files = self.base.get_files(task) - task['files'] = self.subs - return [task] - -@query(priority=2) -@bt_query(priority=2) -def sub_id_processor(base, x): - x = lixian_encoding.from_native(x) - - m = re.match(r'^(\d+)/(.+)$', x) - if not m: - return - task_id, sub_id = m.groups() - task = base.find_task_by_id(task_id) - if not task: - return - - assert task['type'] == 'bt', 'task %s is not a bt task' % lixian_encoding.to_native(task['name']) - files = base.get_files(task) - import lixian_filter_expr - files = lixian_filter_expr.filter_expr(files, sub_id) - subs = [x for x in files] - return SubTaskQuery(base, task, subs) - -################################################## - -class BtHashQuery(ExactQuery): - def __init__(self, base, x): - super(BtHashQuery, self).__init__(base) - self.hash = re.match(r'^(?:bt://)?([0-9a-f]{40})$', x, flags=re.I).group(1).lower() - self.task = self.base.find_task_by_hash(self.hash) - - def prepare(self): - if not self.task: - self.base.add_bt_task_by_hash(self.hash) - - def query_once(self): - t = self.base.find_task_by_hash(self.hash) - assert t, 'Task not found: bt://' + self.hash - return [t] - - def query_search(self): - t = self.base.find_task_by_hash(self.hash) - return [t] if t else [] - -@query(priority=1) -@bt_query(priority=1) -def bt_hash_processor(base, x): - if re.match(r'^(bt://)?[0-9a-f]{40}$', x, flags=re.I): - return BtHashQuery(base, x) - -################################################## - -class LocalBtQuery(ExactQuery): - def __init__(self, base, x): - super(LocalBtQuery, self).__init__(base) - self.path = x - self.hash = lixian_hash_bt.info_hash(self.path) - self.task = self.base.find_task_by_hash(self.hash) - with open(self.path, 'rb') as stream: - self.torrent = stream.read() - - def prepare(self): - if not self.task: - self.base.add_bt_task_by_content(self.torrent, self.path) - - def query_once(self): - t = self.base.find_task_by_hash(self.hash) - assert t, 'Task not found: bt://' + self.hash - return [t] - - def query_search(self): - t = self.base.find_task_by_hash(self.hash) - return [t] if t else [] - -@query(priority=1) -@bt_query(priority=1) -def local_bt_processor(base, x): - import os.path - if x.lower().endswith('.torrent') and os.path.exists(x): - return LocalBtQuery(base, x) - -################################################## - -class MagnetQuery(ExactQuery): - def __init__(self, base, x): - super(MagnetQuery, self).__init__(base) - self.url = x - self.hash = lixian_hash_bt.magnet_to_infohash(x).encode('hex').lower() - self.task = self.base.find_task_by_hash(self.hash) - - def prepare(self): - if not self.task: - self.base.add_magnet_task(self.url) - - def query_once(self): - t = self.base.find_task_by_hash(self.hash) - assert t, 'Task not found: bt://' + self.hash - return [t] - - def query_search(self): - t = self.base.find_task_by_hash(self.hash) - return [t] if t else [] - -@query(priority=4) -@bt_query(priority=4) -def magnet_processor(base, url): - if re.match(r'magnet:', url): - return MagnetQuery(base, url) - -################################################## - -class BatchUrlsQuery(ExactQuery): - def __init__(self, base, urls): - super(BatchUrlsQuery, self).__init__(base) - self.urls = urls - - def prepare(self): - for url in self.urls: - if not self.base.find_task_by_url(url): - self.base.add_url_task(url) - - def query_once(self): - return map(self.base.get_task_by_url, self.urls) - - def query_search(self): - return filter(bool, map(self.base.find_task_by_url, self.urls)) - -@query(priority=6) -@bt_query(priority=6) -def url_extend_processor(base, url): - import lixian_plugins.parsers - extended = lixian_plugins.parsers.try_to_extend_link(url) - if extended: - extended = map(lixian_plugins.parsers.to_url, extended) - return BatchUrlsQuery(base, extended) - -################################################## - -class UrlQuery(ExactQuery): - def __init__(self, base, x): - super(UrlQuery, self).__init__(base) - self.url = lixian_url.url_unmask(x) - self.task = self.base.find_task_by_url(self.url) - - def prepare(self): - if not self.task: - self.base.add_url_task(self.url) - - def query_once(self): - t = self.base.find_task_by_url(self.url) - assert t, 'Task not found: ' + self.url - return [t] - - def query_search(self): - t = self.base.find_task_by_url(self.url) - return [t] if t else [] - -@query(priority=7) -def url_processor(base, url): - if re.match(r'\w+://', url): - return UrlQuery(base, url) - -################################################## - -class BtUrlQuery(ExactQuery): - def __init__(self, base, url, torrent): - super(BtUrlQuery, self).__init__(base) - self.url = url - self.torrent = torrent - self.hash = lixian_hash_bt.info_hash_from_content(self.torrent) - self.task = self.base.find_task_by_hash(self.hash) - - def prepare(self): - if not self.task: - self.base.add_bt_task_by_content(self.torrent, self.url) - - def query_once(self): - t = self.base.find_task_by_hash(self.hash) - assert t, 'Task not found: bt://' + self.hash - return [t] - - def query_search(self): - t = self.base.find_task_by_hash(self.hash) - return [t] if t else [] - -@bt_query(priority=7) -def bt_url_processor(base, url): - if not re.match(r'http://', url): - return - print 'Downloading torrent file from', url - import urllib2 - response = urllib2.urlopen(url, timeout=60) - torrent = response.read() - if response.info().get('Content-Encoding') == 'gzip': - def ungzip(s): - from StringIO import StringIO - import gzip - buffer = StringIO(s) - f = gzip.GzipFile(fileobj=buffer) - return f.read() - torrent = ungzip(torrent) - return BtUrlQuery(base, url, torrent) - -################################################## - -class FilterQuery(SearchQuery): - def __init__(self, base, x): - super(FilterQuery, self).__init__(base) - self.keyword = x - - def query_search(self): - import lixian_plugins.filters - tasks = lixian_plugins.filters.filter_tasks(self.base.get_tasks(), self.keyword) - assert tasks is not None - return tasks - -@query(priority=8) -@bt_query(priority=8) -def filter_processor(base, x): - import lixian_plugins.filters - if lixian_plugins.filters.has_task_filter(x): - return FilterQuery(base, x) - -################################################## - -class DefaultQuery(SearchQuery): - def __init__(self, base, x): - super(DefaultQuery, self).__init__(base) - self.text = lixian_encoding.from_native(x) - - def query_search(self): - return filter(lambda t: t['name'].lower().find(self.text.lower()) != -1, self.base.get_tasks()) - -@query(priority=9) -@bt_query(priority=9) -def default_processor(base, x): - return DefaultQuery(base, x) - diff --git a/xunlei-lixian/lixian_query.py b/xunlei-lixian/lixian_query.py deleted file mode 100644 index e1a27cc..0000000 --- a/xunlei-lixian/lixian_query.py +++ /dev/null @@ -1,476 +0,0 @@ - -__all__ = ['query', 'bt_query', 'user_query', 'Query', 'ExactQuery', 'SearchQuery', - 'build_query', 'find_tasks_to_download', 'search_tasks', 'expand_bt_sub_tasks'] - -import lixian_hash_bt -import lixian_hash_ed2k -import lixian_encoding - - -def link_normalize(url): - from lixian_url import url_unmask, normalize_unicode_link - url = url_unmask(url) - if url.startswith('magnet:'): - return 'bt://'+lixian_hash_bt.magnet_to_infohash(url).encode('hex') - elif url.startswith('ed2k://'): - return lixian_hash_ed2k.parse_ed2k_id(url) - elif url.startswith('bt://'): - return url.lower() - elif url.startswith('http://') or url.startswith('ftp://'): - return normalize_unicode_link(url) - return url - -def link_equals(x1, x2): - return link_normalize(x1) == link_normalize(x2) - - -class TaskBase(object): - def __init__(self, client, list_tasks, limit=None): - self.client = client - self.fetch_tasks_unlimited = list_tasks - self.limit = limit - - self.queries = [] - - self.tasks = None - self.files = {} - - self.commit_jobs = [[], []] - - self.download_jobs = [] - - def fetch_tasks(self): - if self.limit: - with self.client.attr(limit=self.limit): - return self.fetch_tasks_unlimited() - else: - return self.fetch_tasks_unlimited() - - def register_queries(self, queries): - self.queries += queries - - def unregister_query(self, query): - self.queries.remove(query) - - def get_tasks(self): - if self.tasks is None: - self.tasks = self.fetch_tasks() - return self.tasks - - def refresh_tasks(self): - self.tasks = self.fetch_tasks() - return self.tasks - - def get_files(self, task): - assert isinstance(task, dict), task - id = task['id'] - if id in self.files: - return self.files[id] - self.files[id] = self.client.list_bt(task) - return self.files[id] - - def find_task_by_id(self, id): - assert isinstance(id, basestring), repr(id) - for t in self.get_tasks(): - if t['id'] == str(id) or t['#'] == int(id): - return t - - def get_task_by_id(self, id): - t = self.find_task_by_id(id) - if not t: - raise Exception('No task found for id '+id) - return t - - def find_task_by_hash(self, hash): - for t in self.get_tasks(): - if t['type'] == 'bt' and t['bt_hash'].lower() == hash: - return t - - def find_task_by_url(self, url): - for t in self.get_tasks(): - if link_equals(t['original_url'], url): - return t - - def get_task_by_url(self, url): - t = self.find_task_by_url(url) - if not t: - raise Exception('No task found for ' + lixian_encoding.to_native(url)) - return t - - def add_url_task(self, url): - self.commit_jobs[0].append(url) - - def add_bt_task_by_hash(self, hash): - self.commit_jobs[1].append(['hash', hash]) - - def add_bt_task_by_content(self, content, name): - self.commit_jobs[1].append(['content', (content, name)]) - - def add_magnet_task(self, hash): - self.commit_jobs[1].append(['magnet', hash]) - - def commit(self): - urls, bts = self.commit_jobs - if urls: - self.client.add_batch_tasks(map(lixian_encoding.try_native_to_utf_8, urls)) - for bt_type, value in bts: - if bt_type == 'hash': - print 'Adding bt task', value # TODO: print the thing user inputs (may be not hash) - self.client.add_torrent_task_by_info_hash(value) - elif bt_type == 'content': - content, name = value - print 'Adding bt task', name - self.client.add_torrent_task_by_content(content) - elif bt_type == 'magnet': - print 'Adding magnet task', value # TODO: print the thing user inputs (may be not hash) - self.client.add_task(value) - else: - raise NotImplementedError(bt_type) - self.commit_jobs = [[], []] - self.refresh_tasks() - - def prepare(self): - # prepare actions (e.g. add tasks) - for query in self.queries: - query.prepare() - # commit and refresh task list - self.commit() - - def query_complete(self): - for query in list(self.queries): - query.query_complete() - - def merge_results(self): - tasks = merge_tasks(self.download_jobs) - for t in tasks: - if t['type'] == 'bt': - # XXX: a dirty trick to cache requests - t['base'] = self - self.download_jobs = tasks - - def query_once(self): - self.prepare() - # merge results - for query in self.queries: - self.download_jobs += query.query_once() - self.query_complete() - self.merge_results() - - def query_search(self): - for query in self.queries: - self.download_jobs += query.query_search() - self.merge_results() - - def peek_download_jobs(self): - return self.download_jobs - - def pull_completed(self): - completed = [] - waiting = [] - for t in self.download_jobs: - if t['status_text'] == 'completed': - completed.append(t) - elif t['type'] != 'bt': - waiting.append(t) - elif 'files' not in t: - waiting.append(t) - else: - i_completed = [] - i_waiting = [] - for f in t['files']: - if f['status_text'] == 'completed': - i_completed.append(f) - else: - i_waiting.append(f) - if i_completed: - tt = dict(t) - tt['files'] = i_completed - completed.append(tt) - if i_waiting: - tt = dict(t) - tt['files'] = i_waiting - waiting.append(tt) - self.download_jobs = waiting - return completed - - def refresh_status(self): - self.refresh_tasks() - self.files = {} - tasks = [] - for old_task in self.download_jobs: - new_task = dict(self.get_task_by_id(old_task['id'])) - if 'files' in old_task: - files = self.get_files(new_task) - new_task['files'] = [files[f['index']] for f in old_task['files']] - tasks.append(new_task) - self.download_jobs = tasks - -class Query(object): - def __init__(self, base): - self.bind(base) - - def bind(self, base): - self.base = base - self.client = base.client - return self - - def unregister(self): - self.base.unregister_query(self) - - def prepare(self): - pass - - def query_once(self): - raise NotImplementedError() - - def query_complete(self): - raise NotImplementedError() - - def query_search(self): - raise NotImplementedError() - -class ExactQuery(Query): - def __init__(self, base): - super(ExactQuery, self).__init__(base) - - def query_once(self): - raise NotImplementedError() - - def query_complete(self): - self.unregister() - - def query_search(self): - raise NotImplementedError() - -class SearchQuery(Query): - def __init__(self, base): - super(SearchQuery, self).__init__(base) - - def query_once(self): - return self.query_search() - - def query_complete(self): - pass - - def query_search(self): - raise NotImplementedError() - -################################################## -# register -################################################## - -processors = [] - -bt_processors = [] - -# 0 -# 1 -- builtin -- most -# 2 -- subs -- 0/[0-9] -# 4 -- magnet -# 5 -- user -# 6 -- extend url -# 7 -- plain url, bt url -# 8 -- filter -# 9 -- default -- text search - -def query(priority): - assert isinstance(priority, (int, float)) - def register(processor): - processors.append((priority, processor)) - return processor - return register - -def bt_query(priority): - assert isinstance(priority, (int, float)) - def register(processor): - bt_processors.append((priority, processor)) - return processor - return register - -def user_query(processor): - return query(priority=5)(processor) - -def load_default_queries(): - import lixian_queries - - -################################################## -# query -################################################## - -def to_list_tasks(client, args): - if args.category: - return lambda: client.read_all_tasks_by_category(args.category) - elif args.deleted: - return client.read_all_deleted - elif args.expired: - return client.read_all_expired - elif args.completed: - return client.read_all_tasks - elif args.failed: - return client.read_all_tasks - elif args.all: - return client.read_all_tasks - else: - return client.read_all_tasks - -def to_query(base, arg, processors): - for _, process in sorted(processors): - q = process(base, arg) - if q: - return q - raise NotImplementedError('No proper query process found for: ' + arg) - -def merge_files(files1, files2): - ids = [] - files = [] - for f in files1 + files2: - if f['id'] not in ids: - files.append(f) - ids.append(f['id']) - return files - -def merge_tasks(tasks): - result_tasks = [] - task_mapping = {} - for task in tasks: - assert type(task) == dict, repr(type) - id = task['id'] - assert 'index' not in task - if id in task_mapping: - if 'files' in task and 'files' in task_mapping[id]: - task_mapping[id]['files'] = merge_files(task_mapping[id]['files'], task['files']) - else: - if 'files' in task: - t = dict(task) - result_tasks.append(t) - task_mapping[id] = t - else: - result_tasks.append(task) - task_mapping[id] = task - return result_tasks - -class AllQuery(SearchQuery): - def __init__(self, base): - super(AllQuery, self).__init__(base) - def query_search(self): - return self.base.get_tasks() - -class CompletedQuery(SearchQuery): - def __init__(self, base): - super(CompletedQuery, self).__init__(base) - def query_search(self): - return filter(lambda x: x['status_text'] == 'completed', self.base.get_tasks()) - -class FailedQuery(SearchQuery): - def __init__(self, base): - super(FailedQuery, self).__init__(base) - def query_search(self): - return filter(lambda x: x['status_text'] == 'failed', self.base.get_tasks()) - -class NoneQuery(SearchQuery): - def __init__(self, base): - super(NoneQuery, self).__init__(base) - def query_search(self): - return [] - -def default_query(options): - if options.category: - return AllQuery - elif options.deleted: - return AllQuery - elif options.expired: - return AllQuery - elif options.completed: - return CompletedQuery - elif options.failed: - return FailedQuery - elif options.all: - return AllQuery - else: - return NoneQuery - -def parse_queries(base, args): - return [to_query(base, arg, bt_processors if args.torrent else processors) for arg in args] or [default_query(args)(base)] - -def parse_limit(args): - limit = args.limit - if limit: - limit = int(limit) - ids = [] - for x in args: - import re - if re.match(r'^\d+$', x): - ids.append(int(x)) - elif re.match(r'^(\d+)/', x): - ids.append(int(x.split('/')[0])) - elif re.match(r'^(\d+)-(\d+)$', x): - ids.extend(map(int, x.split('-'))) - else: - return limit - if ids and limit: - return min(max(ids)+1, limit) - elif ids: - return max(ids)+1 - else: - return limit - -def build_query(client, args): - if args.input: - import fileinput - args._left.extend(line.strip() for line in fileinput.input(args.input) if line.strip()) - load_default_queries() # IMPORTANT: init default queries - limit = parse_limit(args) - base = TaskBase(client, to_list_tasks(client, args), limit) - base.register_queries(parse_queries(base, args)) - return base - -################################################## -# compatible APIs -################################################## - -def find_tasks_to_download(client, args): - base = build_query(client, args) - base.query_once() - return base.peek_download_jobs() - -def search_tasks(client, args): - base = build_query(client, args) - base.query_search() - return base.peek_download_jobs() - -def expand_bt_sub_tasks(task): - files = task['base'].get_files(task) # XXX: a dirty trick to cache requests - not_ready = [] - single_file = False - if len(files) == 1 and files[0]['name'] == task['name']: - single_file = True - if 'files' in task: - ordered_files = [] - for t in task['files']: - assert isinstance(t, dict) - if t['status_text'] != 'completed': - not_ready.append(t) - else: - ordered_files.append(t) - files = ordered_files - return files, not_ready, single_file - - -################################################## -# simple helpers -################################################## - -def get_task_by_id(client, id): - base = TaskBase(client, client.read_all_tasks) - return base.get_task_by_id(id) - -def get_task_by_any(client, arg): - import lixian_cli_parser - tasks = search_tasks(client, lixian_cli_parser.parse_command_line([arg])) - if not tasks: - raise LookupError(arg) - if len(tasks) > 1: - raise LookupError('Too many results for ' + arg) - return tasks[0] - diff --git a/xunlei-lixian/lixian_url.py b/xunlei-lixian/lixian_url.py deleted file mode 100644 index 0f3d685..0000000 --- a/xunlei-lixian/lixian_url.py +++ /dev/null @@ -1,75 +0,0 @@ - -import base64 -import urllib - -def xunlei_url_encode(url): - return 'thunder://'+base64.encodestring('AA'+url+'ZZ').replace('\n', '') - -def xunlei_url_decode(url): - assert url.startswith('thunder://') - url = base64.decodestring(url[10:]) - assert url.startswith('AA') and url.endswith('ZZ') - return url[2:-2] - -def flashget_url_encode(url): - return 'Flashget://'+base64.encodestring('[FLASHGET]'+url+'[FLASHGET]').replace('\n', '') - -def flashget_url_decode(url): - assert url.startswith('Flashget://') - url = base64.decodestring(url[11:]) - assert url.startswith('[FLASHGET]') and url.endswith('[FLASHGET]') - return url.replace('[FLASHGET]', '') - -def flashgetx_url_decode(url): - assert url.startswith('flashgetx://|mhts|') - name, size, hash, end = url.split('|')[2:] - assert end == '/' - return 'ed2k://|file|'+base64.decodestring(name)+'|'+size+'|'+hash+'/' - -def qqdl_url_encode(url): - return 'qqdl://' + base64.encodestring(url).replace('\n', '') - -def qqdl_url_decode(url): - assert url.startswith('qqdl://') - return base64.decodestring(url[7:]) - -def url_unmask(url): - if url.startswith('thunder://'): - return normalize_unicode_link(xunlei_url_decode(url)) - elif url.startswith('Flashget://'): - return flashget_url_decode(url) - elif url.startswith('flashgetx://'): - return flashgetx_url_decode(url) - elif url.startswith('qqdl://'): - return qqdl_url_decode(url) - else: - return url - -def normalize_unicode_link(url): - import re - def escape_unicode(m): - c = m.group() - if ord(c) < 0x80: - return c - else: - return urllib.quote(c.encode('utf-8')) - def escape_str(m): - c = m.group() - if ord(c) < 0x80: - return c - else: - return urllib.quote(c) - if type(url) == unicode: - return re.sub(r'.', escape_unicode, url) - else: - return re.sub(r'.', escape_str, url) - -def unquote_url(x): - x = urllib.unquote(x) - if type(x) != str: - return x - try: - return x.decode('utf-8') - except UnicodeDecodeError: - return x.decode('gbk') # can't decode in utf-8 and gbk - diff --git a/xunlei-lixian/lixian_util.py b/xunlei-lixian/lixian_util.py deleted file mode 100644 index 7d58612..0000000 --- a/xunlei-lixian/lixian_util.py +++ /dev/null @@ -1,29 +0,0 @@ - -__all__ = [] - -import re - -def format_1d(n): - return re.sub(r'\.0*$', '', '%.1f' % n) - -def format_size(n): - if n < 1000: - return '%sB' % n - elif n < 1000**2: - return '%sK' % format_1d(n/1000.) - elif n < 1000**3: - return '%sM' % format_1d(n/1000.**2) - elif n < 1000**4: - return '%sG' % format_1d(n/1000.**3) - - -def parse_size(size): - size = str(size) - if re.match('^\d+$', size): - return int(size) - m = re.match(r'^(\d+(?:\.\d+)?)(K|M|G)B?$', size, flags=re.I) - if not m: - raise Exception("Invalid size format: %s" % size) - return int(float(m.group(1)) * {'K': 1000, 'M': 1000*1000, 'G': 1000*1000*1000}[m.group(2).upper()]) - - diff --git a/xunlei-lixian/lixian_verification_code.py b/xunlei-lixian/lixian_verification_code.py deleted file mode 100644 index c6d9345..0000000 --- a/xunlei-lixian/lixian_verification_code.py +++ /dev/null @@ -1,13 +0,0 @@ - -def file_path_verification_code_reader(path): - def reader(image): - with open(path, 'wb') as output: - output.write(image) - print 'Verification code picture is saved to %s, please open it manually and enter what you see.' % path - code = raw_input('Verification code: ') - return code - return reader - -def default_verification_code_reader(args): - if args.verification_code_path: - return file_path_verification_code_reader(args.verification_code_path) diff --git a/xunlei-lixian/tests/123.txt b/xunlei-lixian/tests/123.txt deleted file mode 100644 index d800886..0000000 --- a/xunlei-lixian/tests/123.txt +++ /dev/null @@ -1 +0,0 @@ -123 \ No newline at end of file diff --git a/xunlei-lixian/tests/123456.txt b/xunlei-lixian/tests/123456.txt deleted file mode 100644 index 4632e06..0000000 --- a/xunlei-lixian/tests/123456.txt +++ /dev/null @@ -1 +0,0 @@ -123456 \ No newline at end of file diff --git a/xunlei-lixian/tests/The-quick-brown-fox-jumps-over-the-lazy-dog.txt b/xunlei-lixian/tests/The-quick-brown-fox-jumps-over-the-lazy-dog.txt deleted file mode 100644 index ff3bb63..0000000 --- a/xunlei-lixian/tests/The-quick-brown-fox-jumps-over-the-lazy-dog.txt +++ /dev/null @@ -1 +0,0 @@ -The quick brown fox jumps over the lazy dog \ No newline at end of file diff --git a/xunlei-lixian/tests/a.txt b/xunlei-lixian/tests/a.txt deleted file mode 100644 index 2e65efe..0000000 --- a/xunlei-lixian/tests/a.txt +++ /dev/null @@ -1 +0,0 @@ -a \ No newline at end of file diff --git a/xunlei-lixian/tests/abc.txt b/xunlei-lixian/tests/abc.txt deleted file mode 100644 index f2ba8f8..0000000 --- a/xunlei-lixian/tests/abc.txt +++ /dev/null @@ -1 +0,0 @@ -abc \ No newline at end of file diff --git a/xunlei-lixian/tests/empty.txt b/xunlei-lixian/tests/empty.txt deleted file mode 100644 index e69de29..0000000

    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 Z-adngtF;4qMF7@ea?|2Q)iu|r{|{W1*p~nR diff --git a/node_modules/iced-coffee-script/media/rotate4.png b/node_modules/iced-coffee-script/media/rotate4.png deleted file mode 100644 index 2c9aaff4b69140e132efcf9947dc38e969ad0348..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 149621 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 diff --git a/node_modules/iced-coffee-script/media/rotate5.png b/node_modules/iced-coffee-script/media/rotate5.png deleted file mode 100644 index 2f1da94d79aa88770ac635296c847653cef91b18..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 74346 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