sproutit-narwhal 0.1.106

Sign up to get free protection for your applications and to get access to all the features.
Files changed (291) hide show
  1. data/DISTRIBUTION.yml +15 -0
  2. data/README.md +86 -0
  3. data/Rakefile +349 -0
  4. data/VERSION.yml +7 -0
  5. data/bin/activate +50 -0
  6. data/bin/activate.bash +50 -0
  7. data/bin/activate.cmd +3 -0
  8. data/bin/js +67 -0
  9. data/bin/json +2 -0
  10. data/bin/narwhal +67 -0
  11. data/bin/narwhal.cmd +29 -0
  12. data/bin/sea +45 -0
  13. data/bin/sea.cmd +25 -0
  14. data/bin/tusk +2 -0
  15. data/bin/tusk.cmd +5 -0
  16. data/catalog.json +902 -0
  17. data/docs/available-packages.md +32 -0
  18. data/docs/browser-api-plan.md +290 -0
  19. data/docs/browser-api.md +153 -0
  20. data/docs/download.md +25 -0
  21. data/docs/engines.md +32 -0
  22. data/docs/json-tool.md +121 -0
  23. data/docs/lib/binary.wiki +242 -0
  24. data/docs/lib/file.wiki +325 -0
  25. data/docs/lib/os/popen.md +70 -0
  26. data/docs/modules.md +38 -0
  27. data/docs/narwhal.md +487 -0
  28. data/docs/packages-howto.md +32 -0
  29. data/docs/packages.md +30 -0
  30. data/docs/posts/2009-07-29-hello-0.1.md +19 -0
  31. data/docs/quick-start.md +69 -0
  32. data/docs/sea.md +49 -0
  33. data/engines/browser/lib/binary.js +2 -0
  34. data/engines/browser/lib/reactor.js +21 -0
  35. data/engines/browser/lib/system.js +3 -0
  36. data/engines/default/lib/array.js +164 -0
  37. data/engines/default/lib/binary-engine.js +53 -0
  38. data/engines/default/lib/binary.js +755 -0
  39. data/engines/default/lib/date.js +8 -0
  40. data/engines/default/lib/file-engine.js +119 -0
  41. data/engines/default/lib/function.js +119 -0
  42. data/engines/default/lib/global.js +11 -0
  43. data/engines/default/lib/io-engine.js +26 -0
  44. data/engines/default/lib/json.js +488 -0
  45. data/engines/default/lib/object.js +69 -0
  46. data/engines/default/lib/os-engine.js +3 -0
  47. data/engines/default/lib/reactor.js +12 -0
  48. data/engines/default/lib/string.js +84 -0
  49. data/engines/default/lib/system.js +20 -0
  50. data/engines/default/lib/worker.js +133 -0
  51. data/engines/jsc/README.md +18 -0
  52. data/engines/jsc/bootstrap.js +53 -0
  53. data/engines/jsc/deps/http-parser/LICENSE +77 -0
  54. data/engines/jsc/deps/http-parser/README.md +145 -0
  55. data/engines/jsc/deps/http-parser/http_parser.c +6087 -0
  56. data/engines/jsc/deps/http-parser/http_parser.h +141 -0
  57. data/engines/jsc/deps/http-parser/http_parser.rl +500 -0
  58. data/engines/jsc/deps/http-parser/test.c +858 -0
  59. data/engines/jsc/include/binary-engine.h +11 -0
  60. data/engines/jsc/include/io-engine.h +23 -0
  61. data/engines/jsc/include/narwhal.h +427 -0
  62. data/engines/jsc/lib/file-engine.js +31 -0
  63. data/engines/jsc/lib/http.js +1 -0
  64. data/engines/jsc/lib/io-engine.js +202 -0
  65. data/engines/jsc/lib/os-engine.js +25 -0
  66. data/engines/jsc/lib/system.js +18 -0
  67. data/engines/jsc/lib/zip.js +1 -0
  68. data/engines/jsc/narwhal-jsc.c +273 -0
  69. data/engines/jsc/narwhal.c +29 -0
  70. data/engines/jsc/package.json +8 -0
  71. data/engines/jsc/src/binary-engine.cc +290 -0
  72. data/engines/jsc/src/file-engine.cc +405 -0
  73. data/engines/jsc/src/io-engine.cc +423 -0
  74. data/engines/jsc/src/jack/handler/jill.cc +710 -0
  75. data/engines/jsc/src/os-engine.cc +210 -0
  76. data/engines/rhino/bin/narwhal-rhino +68 -0
  77. data/engines/rhino/bin/narwhal-rhino.cmd +34 -0
  78. data/engines/rhino/bootstrap.js +119 -0
  79. data/engines/rhino/jars/jline.jar +0 -0
  80. data/engines/rhino/jars/jna.jar +0 -0
  81. data/engines/rhino/jars/js.jar +0 -0
  82. data/engines/rhino/lib/binary-engine.js +83 -0
  83. data/engines/rhino/lib/concurrency.js +6 -0
  84. data/engines/rhino/lib/event-queue.js +18 -0
  85. data/engines/rhino/lib/file-engine.js +216 -0
  86. data/engines/rhino/lib/http-client-engine.js +90 -0
  87. data/engines/rhino/lib/http-engine.js +10 -0
  88. data/engines/rhino/lib/io-engine.js +347 -0
  89. data/engines/rhino/lib/md5-engine.js +40 -0
  90. data/engines/rhino/lib/os-engine.js +150 -0
  91. data/engines/rhino/lib/packages-engine.js +71 -0
  92. data/engines/rhino/lib/sandbox-engine.js +70 -0
  93. data/engines/rhino/lib/system.js +38 -0
  94. data/engines/rhino/lib/worker-engine.js +23 -0
  95. data/engines/rhino/lib/zip.js +78 -0
  96. data/engines/rhino/package.json +4 -0
  97. data/engines/secure/lib/file.js +6 -0
  98. data/engines/secure/lib/system.js +6 -0
  99. data/engines/template/bin/narwhal-engine-name +32 -0
  100. data/engines/template/bootstrap.js +40 -0
  101. data/engines/template/lib/file-engine.js +118 -0
  102. data/engines/template/lib/system.js +17 -0
  103. data/examples/browser-deployment-jackconfig.js +35 -0
  104. data/examples/fibonacci-worker.js +35 -0
  105. data/examples/fibonacci.js +19 -0
  106. data/examples/hello +2 -0
  107. data/examples/narwhal +3 -0
  108. data/examples/not-quite-a-quine.js +1 -0
  109. data/extconf.rb +44 -0
  110. data/gem_bin/narwhal +5 -0
  111. data/gem_bin/sea +4 -0
  112. data/gem_bin/tusk +4 -0
  113. data/lib/args.js +849 -0
  114. data/lib/base16.js +16 -0
  115. data/lib/base64.js +120 -0
  116. data/lib/codec/base64.js +8 -0
  117. data/lib/crc32.js +60 -0
  118. data/lib/file-bootstrap.js +187 -0
  119. data/lib/file.js +659 -0
  120. data/lib/hash.js +28 -0
  121. data/lib/hashp.js +65 -0
  122. data/lib/html.js +16 -0
  123. data/lib/http-client.js +134 -0
  124. data/lib/http.js +17 -0
  125. data/lib/io.js +98 -0
  126. data/lib/jsmin.js +315 -0
  127. data/lib/jsonpath.js +89 -0
  128. data/lib/logger.js +55 -0
  129. data/lib/md4.js +146 -0
  130. data/lib/md5.js +164 -0
  131. data/lib/mime.js +166 -0
  132. data/lib/narwhal.js +102 -0
  133. data/lib/narwhal/client.js +261 -0
  134. data/lib/narwhal/compile.js +99 -0
  135. data/lib/narwhal/env.js +140 -0
  136. data/lib/narwhal/inline.js +106 -0
  137. data/lib/narwhal/json.js +324 -0
  138. data/lib/narwhal/json.md +178 -0
  139. data/lib/narwhal/repl.js +96 -0
  140. data/lib/narwhal/server-test.js +6 -0
  141. data/lib/narwhal/server.js +270 -0
  142. data/lib/narwhal/tusk.js +170 -0
  143. data/lib/narwhal/tusk/bin.js +13 -0
  144. data/lib/narwhal/tusk/bundle.js +0 -0
  145. data/lib/narwhal/tusk/catalog.js +22 -0
  146. data/lib/narwhal/tusk/clone.js +66 -0
  147. data/lib/narwhal/tusk/consolidate.js +25 -0
  148. data/lib/narwhal/tusk/create-catalog.js +80 -0
  149. data/lib/narwhal/tusk/engine.js +42 -0
  150. data/lib/narwhal/tusk/freeze.js +0 -0
  151. data/lib/narwhal/tusk/init.js +56 -0
  152. data/lib/narwhal/tusk/install.js +288 -0
  153. data/lib/narwhal/tusk/list.js +20 -0
  154. data/lib/narwhal/tusk/orphans.js +0 -0
  155. data/lib/narwhal/tusk/reheat.js +15 -0
  156. data/lib/narwhal/tusk/remove.js +15 -0
  157. data/lib/narwhal/tusk/search.js +145 -0
  158. data/lib/narwhal/tusk/update.js +21 -0
  159. data/lib/narwhal/tusk/upgrade.js +0 -0
  160. data/lib/os.js +33 -0
  161. data/lib/packages.js +423 -0
  162. data/lib/printf.js +169 -0
  163. data/lib/promise.js +352 -0
  164. data/lib/querystring.js +176 -0
  165. data/lib/ref-send.js +257 -0
  166. data/lib/regexp.js +12 -0
  167. data/lib/sandbox.js +422 -0
  168. data/lib/sha.js +112 -0
  169. data/lib/sha256.js +102 -0
  170. data/lib/struct.js +228 -0
  171. data/lib/term.js +179 -0
  172. data/lib/test/assert.js +95 -0
  173. data/lib/test/equiv.js +188 -0
  174. data/lib/test/jsdump.js +165 -0
  175. data/lib/test/runner.js +129 -0
  176. data/lib/unload.js +13 -0
  177. data/lib/uri.js +378 -0
  178. data/lib/url.js +5 -0
  179. data/lib/utf8.js +64 -0
  180. data/lib/util.js +985 -0
  181. data/lib/uuid.js +89 -0
  182. data/lib/xregexp.js +521 -0
  183. data/local.json.template +1 -0
  184. data/narwhal.gemspec +105 -0
  185. data/narwhal.js +213 -0
  186. data/package.json +26 -0
  187. data/packages/readline/engines/default/lib/readline.js +4 -0
  188. data/packages/readline/engines/rhino/lib/readline.js +6 -0
  189. data/packages/readline/package.json +5 -0
  190. data/sources.json +207 -0
  191. data/tests/all-tests.js +17 -0
  192. data/tests/args.js +31 -0
  193. data/tests/args/domain.js +215 -0
  194. data/tests/args/options.js +36 -0
  195. data/tests/args/shifting.js +92 -0
  196. data/tests/args/validation.js +31 -0
  197. data/tests/base64.js +23 -0
  198. data/tests/commonjs.js +3 -0
  199. data/tests/commonjs/all-tests.js +12 -0
  200. data/tests/commonjs/bytearray-encodings-tests.js +69 -0
  201. data/tests/commonjs/bytearray-tests.js +465 -0
  202. data/tests/commonjs/bytestring-encodings-tests.js +89 -0
  203. data/tests/commonjs/bytestring-tests.js +263 -0
  204. data/tests/commonjs/es5/all-tests.js +3 -0
  205. data/tests/commonjs/es5/bind.js +29 -0
  206. data/tests/commonjs/file-tests.js +315 -0
  207. data/tests/commonjs/file/dirname.js +31 -0
  208. data/tests/commonjs/file/extension.js +45 -0
  209. data/tests/commonjs/file/is-absolute.js +11 -0
  210. data/tests/commonjs/file/iterator.js +101 -0
  211. data/tests/commonjs/file/normal.js +27 -0
  212. data/tests/commonjs/file/path.js +17 -0
  213. data/tests/commonjs/file/relative.js +42 -0
  214. data/tests/commonjs/file/resolve.js +44 -0
  215. data/tests/commonjs/module-tests.js +9 -0
  216. data/tests/commonjs/modules/absolute/b.js +1 -0
  217. data/tests/commonjs/modules/absolute/program.js +5 -0
  218. data/tests/commonjs/modules/absolute/submodule/a.js +3 -0
  219. data/tests/commonjs/modules/absolute/test.js +9 -0
  220. data/tests/commonjs/modules/all-tests.js +47 -0
  221. data/tests/commonjs/modules/config.js +11 -0
  222. data/tests/commonjs/modules/cyclic/a.js +4 -0
  223. data/tests/commonjs/modules/cyclic/b.js +4 -0
  224. data/tests/commonjs/modules/cyclic/program.js +10 -0
  225. data/tests/commonjs/modules/cyclic/test.js +9 -0
  226. data/tests/commonjs/modules/determinism/program.js +3 -0
  227. data/tests/commonjs/modules/determinism/submodule/a.js +8 -0
  228. data/tests/commonjs/modules/determinism/submodule/b.js +2 -0
  229. data/tests/commonjs/modules/determinism/test.js +9 -0
  230. data/tests/commonjs/modules/exactExports/a.js +3 -0
  231. data/tests/commonjs/modules/exactExports/program.js +4 -0
  232. data/tests/commonjs/modules/exactExports/test.js +9 -0
  233. data/tests/commonjs/modules/hasOwnProperty/hasOwnProperty.js +0 -0
  234. data/tests/commonjs/modules/hasOwnProperty/program.js +3 -0
  235. data/tests/commonjs/modules/hasOwnProperty/test.js +9 -0
  236. data/tests/commonjs/modules/hasOwnProperty/toString.js +0 -0
  237. data/tests/commonjs/modules/method/a.js +12 -0
  238. data/tests/commonjs/modules/method/program.js +8 -0
  239. data/tests/commonjs/modules/method/test.js +9 -0
  240. data/tests/commonjs/modules/missing/program.js +8 -0
  241. data/tests/commonjs/modules/missing/test.js +9 -0
  242. data/tests/commonjs/modules/monkeys/a.js +1 -0
  243. data/tests/commonjs/modules/monkeys/program.js +4 -0
  244. data/tests/commonjs/modules/monkeys/test.js +9 -0
  245. data/tests/commonjs/modules/nested/a/b/c/d.js +3 -0
  246. data/tests/commonjs/modules/nested/program.js +3 -0
  247. data/tests/commonjs/modules/nested/test.js +9 -0
  248. data/tests/commonjs/modules/relative/program.js +5 -0
  249. data/tests/commonjs/modules/relative/submodule/a.js +1 -0
  250. data/tests/commonjs/modules/relative/submodule/b.js +2 -0
  251. data/tests/commonjs/modules/relative/test.js +9 -0
  252. data/tests/commonjs/modules/transitive/a.js +1 -0
  253. data/tests/commonjs/modules/transitive/b.js +1 -0
  254. data/tests/commonjs/modules/transitive/c.js +3 -0
  255. data/tests/commonjs/modules/transitive/program.js +3 -0
  256. data/tests/commonjs/modules/transitive/test.js +9 -0
  257. data/tests/file/all-tests.js +61 -0
  258. data/tests/file/fnmatch.js +102 -0
  259. data/tests/file/glob.js +466 -0
  260. data/tests/file/match.js +102 -0
  261. data/tests/global.js +6 -0
  262. data/tests/global/array.js +19 -0
  263. data/tests/hashes.js +94 -0
  264. data/tests/html.js +13 -0
  265. data/tests/io/stringio.js +21 -0
  266. data/tests/os/all-tests.js +4 -0
  267. data/tests/os/popen.js +41 -0
  268. data/tests/os/system.js +22 -0
  269. data/tests/printf.js +123 -0
  270. data/tests/query-string.js +87 -0
  271. data/tests/sandbox/byte-io.js +20 -0
  272. data/tests/sandbox/fileName.js +3 -0
  273. data/tests/sandbox/foo.js +0 -0
  274. data/tests/sandbox/reload.js +79 -0
  275. data/tests/string.js +35 -0
  276. data/tests/uri.js +41 -0
  277. data/tests/util/all-tests.js +79 -0
  278. data/tests/util/array.js +207 -0
  279. data/tests/util/array/is-arguments.js +29 -0
  280. data/tests/util/array/is-array-like.js +29 -0
  281. data/tests/util/case.js +9 -0
  282. data/tests/util/collection.js +104 -0
  283. data/tests/util/eq.js +57 -0
  284. data/tests/util/expand.js +45 -0
  285. data/tests/util/object.js +125 -0
  286. data/tests/util/operator.js +25 -0
  287. data/tests/util/range.js +19 -0
  288. data/tests/util/repr.js +26 -0
  289. data/tests/util/string.js +34 -0
  290. data/tests/util/unique.js +12 -0
  291. metadata +434 -0
data/lib/printf.js ADDED
@@ -0,0 +1,169 @@
1
+
2
+ /** provides printf and sprintf.
3
+
4
+ This code is unrestricted: you are free to use it however you like.
5
+
6
+ The functions should work as expected, performing left or right alignment,
7
+ truncating strings, outputting numbers with a required precision etc.
8
+
9
+ For complex cases, these functions follow the Perl implementations of
10
+ (s)printf, allowing arguments to be passed out-of-order, and to set the
11
+ precision or length of the output based on arguments instead of fixed
12
+ numbers.
13
+
14
+ See http://perldoc.perl.org/functions/sprintf.html for more information.
15
+
16
+ Implemented:
17
+ - zero and space-padding
18
+ - right and left-alignment,
19
+ - base X prefix (binary, octal and hex)
20
+ - positive number prefix
21
+ - (minimum) width
22
+ - precision / truncation / maximum width
23
+ - out of order arguments
24
+
25
+ Not implemented (yet):
26
+ - vector flag
27
+ - size (bytes, words, long-words etc.)
28
+
29
+ Will not implement:
30
+ - %n or %p (no pointers in JavaScript)
31
+
32
+ by Ash Searle, 2007-04-27
33
+ Integrated into Chiron by Kris Kowal, 2007-10-19
34
+ Integrated into Narwhal by Kris Kowal, 2009-07-27
35
+
36
+ */
37
+
38
+ exports.printf = function () {
39
+ print(exports.sprintf.apply(null, arguments));
40
+ };
41
+
42
+ exports.fprintf = function () {
43
+ var args = Array.prototype.slice.call(arguments);
44
+ var stream = args.shift();
45
+ stream.print(exports.sprintf.apply(null, args));
46
+ };
47
+
48
+ exports.sprintf = function () {
49
+
50
+ var a = arguments, i = 1, format = a[0];
51
+ return format.replace(regex, function ($0, valueIndex, flags, minWidth, precision, type) {
52
+ if ($0 == '%%') return '%';
53
+
54
+ // parse flags
55
+ var leftJustify = flags.indexOf('-') > -1,
56
+ positivePrefix = (flags.match(/[ +]/g) || ['']).pop(),
57
+ zeroPad = flags.indexOf('0') > -1,
58
+ prefixBaseX = flags.indexOf('#') > -1;
59
+
60
+ // parameters may be null, undefined, empty-string or real valued
61
+ // we want to ignore null, undefined and empty-string values
62
+
63
+ if (!minWidth)
64
+ minWidth = 0;
65
+ else if (minWidth == '*')
66
+ minWidth = +a[i++];
67
+ else if (minWidth.charAt(0) == '*')
68
+ minWidth = +a[minWidth.slice(1, -1)];
69
+ else
70
+ minWidth = +minWidth;
71
+
72
+ // Note: undocumented perl feature:
73
+ if (minWidth < 0) {
74
+ minWidth = -minWidth;
75
+ leftJustify = true;
76
+ }
77
+
78
+ if (!isFinite(minWidth))
79
+ throw new Error('sprintf: (minimum-)width must be finite');
80
+
81
+ if (!precision)
82
+ precision = 'fFeE'.indexOf(type) > -1 ? 6 : (type == 'd' ? 0 : void(0));
83
+ else if (precision == '*')
84
+ precision = +a[i++];
85
+ else if (precision.charAt(0) == '*')
86
+ precision = +a[precision.slice(1, -1)];
87
+ else
88
+ precision = +precision;
89
+
90
+ // grab value using valueIndex if required?
91
+ var value = valueIndex ? a[valueIndex] : a[i++];
92
+
93
+ switch (type) {
94
+ case 's': return formatString(String(value), leftJustify, minWidth, precision, zeroPad);
95
+ case 'c': return formatString(String.fromCharCode(+value), leftJustify, minWidth, precision, zeroPad);
96
+ case 'b': return formatBaseX(value, 2, prefixBaseX, leftJustify, minWidth, precision, zeroPad);
97
+ case 'o': return formatBaseX(value, 8, prefixBaseX, leftJustify, minWidth, precision, zeroPad);
98
+ case 'x': return formatBaseX(value, 16, prefixBaseX, leftJustify, minWidth, precision, zeroPad);
99
+ case 'X': return formatBaseX(value, 16, prefixBaseX, leftJustify, minWidth, precision, zeroPad).toUpperCase();
100
+ case 'u': return formatBaseX(value, 10, prefixBaseX, leftJustify, minWidth, precision, zeroPad);
101
+ case 'i':
102
+ case 'd': {
103
+ var number = parseInt(+value);
104
+ var prefix = number < 0 ? '-' : positivePrefix;
105
+ value = prefix + pad(String(Math.abs(number)), precision, '0', false);
106
+ return justify(value, prefix, leftJustify, minWidth, zeroPad);
107
+ }
108
+ case 'e':
109
+ case 'E':
110
+ case 'f':
111
+ case 'F':
112
+ case 'g':
113
+ case 'G': {
114
+ var number = +value;
115
+ var prefix = number < 0 ? '-' : positivePrefix;
116
+ var method = ['toExponential', 'toFixed', 'toPrecision']['efg'.indexOf(type.toLowerCase())];
117
+ var textTransform = ['toString', 'toUpperCase']['eEfFgG'.indexOf(type) % 2];
118
+ value = prefix + Math.abs(number)[method](precision);
119
+ return justify(value, prefix, leftJustify, minWidth, zeroPad)[textTransform]();
120
+ }
121
+ default: return $0;
122
+ }
123
+ });
124
+ }
125
+
126
+ var pad = function (str, len, chr, leftJustify) {
127
+ var padding = (str.length >= len) ? '' : Array(1 + len - str.length >>> 0).join(chr);
128
+ return leftJustify ? str + padding : padding + str;
129
+ };
130
+
131
+ var justify = function (value, prefix, leftJustify, minWidth, zeroPad) {
132
+ var diff = minWidth - value.length;
133
+ if (diff > 0) {
134
+ if (leftJustify || !zeroPad) {
135
+ value = pad(value, minWidth, ' ', leftJustify);
136
+ } else {
137
+ value = value.slice(0, prefix.length) + pad('', diff, '0', true) + value.slice(prefix.length);
138
+ }
139
+ }
140
+ return value;
141
+ };
142
+
143
+ var formatBaseX = function (value, base, prefix, leftJustify, minWidth, precision, zeroPad) {
144
+ // Note: casts negative numbers to positive ones
145
+ var number = value >>> 0;
146
+ prefix = prefix && number && {'2': '0b', '8': '0', '16': '0x'}[base] || '';
147
+ value = prefix + pad(number.toString(base), precision || 0, '0', false);
148
+ return justify(value, prefix, leftJustify, minWidth, zeroPad);
149
+ };
150
+
151
+ var formatString = function (value, leftJustify, minWidth, precision, zeroPad) {
152
+ if (precision != null) {
153
+ value = value.slice(0, precision);
154
+ }
155
+ return justify(value, '', leftJustify, minWidth, zeroPad);
156
+ };
157
+
158
+ var regex = /%(?:%|(?:(\d+)\$)?([-+#0 ]*)([1-9][0-9]*|\*(?:[0-9]+\$)?)?(?:\.([0-9]+|\*(?:[0-9]+\$)?))?([csduoxefgXEGbi]))/g;
159
+
160
+ /*
161
+ References
162
+ ==========
163
+
164
+ http://www.webtoolkit.info/javascript-sprintf.html
165
+ An alternate implementation, if this one does not perform well
166
+ in practice.
167
+
168
+ */
169
+
data/lib/promise.js ADDED
@@ -0,0 +1,352 @@
1
+ // this is based on the CommonJS spec for promises:
2
+ // http://wiki.commonjs.org/wiki/Promises
3
+
4
+ // A typical usage:
5
+ // A default Promise constructor can be used to create a self-resolving promise:
6
+ // var Promise = require("promise").Promise;
7
+ // var promise = new Promise();
8
+ // asyncOperation(function(){
9
+ // Promise.resolve("succesful result");
10
+ // });
11
+ // promise -> given to the consumer
12
+ //
13
+ // A consumer can use the promise
14
+ // promise.then(function(result){
15
+ // ... when the action is complete this is executed ...
16
+ // },
17
+ // function(error){
18
+ // ... executed when the promise fails
19
+ // });
20
+ //
21
+ // Alternately, a provider can create a deferred and resolve it when it completes an action.
22
+ // The deferred object provides a separation of consumer and producer to protect
23
+ // promises from being fulfilled by untrusted code.
24
+ // var defer = require("promise").defer;
25
+ // var deferred = defer();
26
+ // asyncOperation(function(){
27
+ // deferred.resolve("succesful result");
28
+ // });
29
+ // deferred.promise -> given to the consumer
30
+ //
31
+ // Another way that a consumer can use the promise (using promise.then is also allowed)
32
+ // var when = require("promise").when;
33
+ // when(promise,function(result){
34
+ // ... when the action is complete this is executed ...
35
+ // },
36
+ // function(error){
37
+ // ... executed when the promise fails
38
+ // });
39
+ try{
40
+ var queue = require("event-queue");
41
+ }
42
+ catch(e){
43
+ // squelch the error, and only complain if the queue is needed
44
+ }
45
+
46
+ /**
47
+ * Default constructor that creates a self-resolving Promise. Not all promise implementations
48
+ * need to use this constructor.
49
+ */
50
+ var Promise = exports.Promise = function(canceller){
51
+ // make a deferred and copy everything to its promise so it can be self-resolving
52
+ var deferred = new Deferred();
53
+ var promise = deferred.promise;
54
+ for(var i in deferred){
55
+ promise[i] = deferred[i];
56
+ }
57
+ if(canceller){
58
+ promise.cancel = function(){
59
+ var error = canceller();
60
+ if(!(error instanceof Error)){
61
+ error = new Error(error);
62
+ }
63
+ promise.reject(error);
64
+ }
65
+ }
66
+ return promise;
67
+ };
68
+
69
+ /**
70
+ * Promise implementations must provide a "then" function.
71
+ */
72
+ Promise.prototype.then = function(resolvedCallback, errorCallback, progressCallback){
73
+ throw new TypeError("The Promise base class is abstract, this function must be implemented by the Promise implementation");
74
+ };
75
+
76
+ /**
77
+ * If an implementation of a promise supports a concurrency model that allows
78
+ * execution to block until the promise is resolved, the wait function may be
79
+ * added.
80
+ */
81
+ /**
82
+ * If an implementation of a promise can be cancelled, it may add this function
83
+ */
84
+ // Promise.prototype.cancel = function(){
85
+ // };
86
+
87
+ Promise.prototype.get = function(propertyName){
88
+ return this.then(function(value){
89
+ return value[propertyName];
90
+ });
91
+ };
92
+
93
+ Promise.prototype.put = function(propertyName, value){
94
+ return this.then(function(object){
95
+ return object[propertyName] = value;
96
+ });
97
+ };
98
+
99
+ Promise.prototype.call = function(functionName /*, args */){
100
+ return this.then(function(value){
101
+ return value[propertyName].apply(value, Array.prototype.slice.call(arguments, 1));
102
+ });
103
+ };
104
+
105
+ // if an error is not handled within 5 seconds then it should be reported
106
+ var errorHandledTimeout = 5000;
107
+ function DeferredPromise(){
108
+ }
109
+ DeferredPromise.prototype = Promise.prototype;
110
+ // A deferred provides an API for creating and resolveing a promise.
111
+ exports.defer = function(canceller){
112
+ return new Deferred(canceller);
113
+ }
114
+ function Deferred(){
115
+ var result, finished, isError, waiting = [], handled, errorTimeout;
116
+ var promise = this.promise = new DeferredPromise();
117
+
118
+ function notifyAll(value){
119
+ if(queue){
120
+ // if we have an event queue, we will enqueue it
121
+ queue.enqueue(doNotify);
122
+ }
123
+ else{
124
+ doNotify();
125
+ }
126
+
127
+ function doNotify(){
128
+ if(finished){
129
+ throw new Error("This deferred has already been resolved");
130
+ }
131
+ result = value;
132
+ finished = true;
133
+ for(var i = 0; i < waiting.length; i++){
134
+ notify(waiting[i]);
135
+ }
136
+ }
137
+ }
138
+ function notify(listener){
139
+ var func = (isError ? listener.error : listener.resolved);
140
+ try{
141
+ if(func){
142
+ handled = true;
143
+ if(typeof errorTimeout == "number"){
144
+ clearTimeout(errorTimeout);
145
+ }
146
+ var newResult = func(result);
147
+ }
148
+ if(newResult && typeof newResult.then === "function"){
149
+ newResult.then(function(result){
150
+ listener.deferred.resolve(result);
151
+ },
152
+ function(error){
153
+ listener.deferred.reject(error);
154
+ });
155
+ }
156
+ else{
157
+ listener.deferred.resolve(newResult === undefined ? result : newResult);
158
+ }
159
+ }
160
+ catch(e){
161
+ listener.deferred.reject(e);
162
+ }
163
+ }
164
+ // calling resolve will resolve the promise
165
+ this.resolve = function(value){
166
+ notifyAll(value);
167
+ };
168
+ // calling error will indicate that the promise failed
169
+ var errback = this.reject = function(error){
170
+ isError = true;
171
+ notifyAll(error);
172
+ if(typeof setTimeout != "undefined" && !handled){
173
+ errorTimeout = setTimeout(function(){
174
+ throw error;
175
+ }, errorHandledTimeout);
176
+ }
177
+ }
178
+ // call progress to provide updates on the progress on the completion of the promise
179
+ this.progress = function(update){
180
+ for(var i = 0; i < waiting.length; i++){
181
+ var progress = waiting[i].progress;
182
+ progress && progress(update);
183
+ }
184
+ }
185
+ // provide the implementation of the promise
186
+ promise.then = function(resolvedCallback, errorCallback, progressCallback){
187
+ var returnDeferred = new Deferred();
188
+ var listener = {resolved: resolvedCallback, error: errorCallback, progress: progressCallback, deferred: returnDeferred};
189
+ if(finished){
190
+ notify(listener);
191
+ }
192
+ else{
193
+ waiting.push(listener);
194
+ }
195
+ return returnDeferred.promise;
196
+ };
197
+ };
198
+
199
+ function perform(value, async, sync, promiseNotNeeded){
200
+ try{
201
+ if(value && typeof value.then === "function"){
202
+ value = async(value);
203
+ }
204
+ else{
205
+ value = sync(value);
206
+ }
207
+ if(value && typeof value.then === "function"){
208
+ return value;
209
+ }
210
+ if(promiseNotNeeded){
211
+ return value;
212
+ }
213
+ var deferred = new Deferred();
214
+ deferred.resolve(value);
215
+ return deferred.promise;
216
+ }catch(e){
217
+ if(promiseNotNeeded){
218
+ throw e;
219
+ }
220
+ var deferred = new Deferred();
221
+ deferred.reject(e);
222
+ return deferred.promise;
223
+ }
224
+
225
+ }
226
+ /**
227
+ * Promise manager to make it easier to consume promises
228
+ */
229
+
230
+ /**
231
+ * Registers an observer on a promise.
232
+ * @param value promise or value to observe
233
+ * @param resolvedCallback function to be called with the resolved value
234
+ * @param rejectCallback function to be called with the rejection reason
235
+ * @param progressCallback function to be called when progress is made
236
+ * @return promise for the return value from the invoked callback
237
+ */
238
+ exports.when = function(value, resolvedCallback, rejectCallback, progressCallback){
239
+ return perform(value, function(value){
240
+ return value.then(resolvedCallback, rejectCallback, progressCallback);
241
+ },
242
+ function(value){
243
+ return resolvedCallback(value);
244
+ });
245
+ };
246
+ /**
247
+ * Registers an observer on a promise.
248
+ * @param value promise or value to observe
249
+ * @param resolvedCallback function to be called with the resolved value
250
+ * @param rejectCallback function to be called with the rejection reason
251
+ * @param progressCallback function to be called when progress is made
252
+ * @return promise for the return value from the invoked callback or the value if it
253
+ * is a non-promise value
254
+ */
255
+ exports.whenPreservingType = function(value, resolvedCallback, rejectCallback, progressCallback){
256
+ return perform(value, function(value){
257
+ return value.then(resolvedCallback, rejectCallback, progressCallback);
258
+ },
259
+ function(value){
260
+ return resolvedCallback(value);
261
+ }, true);
262
+ };
263
+
264
+ /**
265
+ * Gets the value of a property in a future turn.
266
+ * @param target promise or value for target object
267
+ * @param property name of property to get
268
+ * @return promise for the property value
269
+ */
270
+ exports.get = function(target, property){
271
+ return perform(target, function(target){
272
+ return target.get(property);
273
+ },
274
+ function(target){
275
+ return target[property]
276
+ });
277
+ };
278
+
279
+ /**
280
+ * Invokes a method in a future turn.
281
+ * @param target promise or value for target object
282
+ * @param methodName name of method to invoke
283
+ * @param args array of invocation arguments
284
+ * @return promise for the return value
285
+ */
286
+ exports.post = function(target, methodName, args){
287
+ return perform(target, function(target){
288
+ return target.call(property, args);
289
+ },
290
+ function(target){
291
+ return target[methodName].apply(target, args);
292
+ });
293
+ };
294
+
295
+ /**
296
+ * Sets the value of a property in a future turn.
297
+ * @param target promise or value for target object
298
+ * @param property name of property to set
299
+ * @param value new value of property
300
+ * @return promise for the return value
301
+ */
302
+ exports.put = function(target, property, value){
303
+ return perform(target, function(target){
304
+ return target.put(property, value);
305
+ },
306
+ function(target){
307
+ return target[property] = value;
308
+ });
309
+ };
310
+
311
+
312
+ /**
313
+ * Waits for the given promise to finish, blocking (and executing other events)
314
+ * if necessary to wait for the promise to finish. If target is not a promise
315
+ * it will return the target immediately. If the promise results in an reject,
316
+ * that reject will be thrown.
317
+ * @param target promise or value to wait for.
318
+ * @return the value of the promise;
319
+ */
320
+ exports.wait = function(target){
321
+ if(!queue){
322
+ throw new Error("Can not wait, the event-queue module is not available");
323
+ }
324
+ if(target && typeof target.then === "function"){
325
+ var isFinished, isError, result;
326
+ target.then(function(value){
327
+ isFinished = true;
328
+ result = value;
329
+ },
330
+ function(error){
331
+ isFinished = true;
332
+ isError = true;
333
+ result = error;
334
+ });
335
+ while(!isFinished){
336
+ try{
337
+ queue.nextEvent()();
338
+ }catch(e){
339
+ print(e);
340
+ }
341
+ }
342
+ if(isError){
343
+ throw result;
344
+ }
345
+ return result;
346
+ }
347
+ else{
348
+ return target;
349
+ }
350
+ };
351
+
352
+