sproutit-narwhal 0.1.106
Sign up to get free protection for your applications and to get access to all the features.
- data/DISTRIBUTION.yml +15 -0
- data/README.md +86 -0
- data/Rakefile +349 -0
- data/VERSION.yml +7 -0
- data/bin/activate +50 -0
- data/bin/activate.bash +50 -0
- data/bin/activate.cmd +3 -0
- data/bin/js +67 -0
- data/bin/json +2 -0
- data/bin/narwhal +67 -0
- data/bin/narwhal.cmd +29 -0
- data/bin/sea +45 -0
- data/bin/sea.cmd +25 -0
- data/bin/tusk +2 -0
- data/bin/tusk.cmd +5 -0
- data/catalog.json +902 -0
- data/docs/available-packages.md +32 -0
- data/docs/browser-api-plan.md +290 -0
- data/docs/browser-api.md +153 -0
- data/docs/download.md +25 -0
- data/docs/engines.md +32 -0
- data/docs/json-tool.md +121 -0
- data/docs/lib/binary.wiki +242 -0
- data/docs/lib/file.wiki +325 -0
- data/docs/lib/os/popen.md +70 -0
- data/docs/modules.md +38 -0
- data/docs/narwhal.md +487 -0
- data/docs/packages-howto.md +32 -0
- data/docs/packages.md +30 -0
- data/docs/posts/2009-07-29-hello-0.1.md +19 -0
- data/docs/quick-start.md +69 -0
- data/docs/sea.md +49 -0
- data/engines/browser/lib/binary.js +2 -0
- data/engines/browser/lib/reactor.js +21 -0
- data/engines/browser/lib/system.js +3 -0
- data/engines/default/lib/array.js +164 -0
- data/engines/default/lib/binary-engine.js +53 -0
- data/engines/default/lib/binary.js +755 -0
- data/engines/default/lib/date.js +8 -0
- data/engines/default/lib/file-engine.js +119 -0
- data/engines/default/lib/function.js +119 -0
- data/engines/default/lib/global.js +11 -0
- data/engines/default/lib/io-engine.js +26 -0
- data/engines/default/lib/json.js +488 -0
- data/engines/default/lib/object.js +69 -0
- data/engines/default/lib/os-engine.js +3 -0
- data/engines/default/lib/reactor.js +12 -0
- data/engines/default/lib/string.js +84 -0
- data/engines/default/lib/system.js +20 -0
- data/engines/default/lib/worker.js +133 -0
- data/engines/jsc/README.md +18 -0
- data/engines/jsc/bootstrap.js +53 -0
- data/engines/jsc/deps/http-parser/LICENSE +77 -0
- data/engines/jsc/deps/http-parser/README.md +145 -0
- data/engines/jsc/deps/http-parser/http_parser.c +6087 -0
- data/engines/jsc/deps/http-parser/http_parser.h +141 -0
- data/engines/jsc/deps/http-parser/http_parser.rl +500 -0
- data/engines/jsc/deps/http-parser/test.c +858 -0
- data/engines/jsc/include/binary-engine.h +11 -0
- data/engines/jsc/include/io-engine.h +23 -0
- data/engines/jsc/include/narwhal.h +427 -0
- data/engines/jsc/lib/file-engine.js +31 -0
- data/engines/jsc/lib/http.js +1 -0
- data/engines/jsc/lib/io-engine.js +202 -0
- data/engines/jsc/lib/os-engine.js +25 -0
- data/engines/jsc/lib/system.js +18 -0
- data/engines/jsc/lib/zip.js +1 -0
- data/engines/jsc/narwhal-jsc.c +273 -0
- data/engines/jsc/narwhal.c +29 -0
- data/engines/jsc/package.json +8 -0
- data/engines/jsc/src/binary-engine.cc +290 -0
- data/engines/jsc/src/file-engine.cc +405 -0
- data/engines/jsc/src/io-engine.cc +423 -0
- data/engines/jsc/src/jack/handler/jill.cc +710 -0
- data/engines/jsc/src/os-engine.cc +210 -0
- data/engines/rhino/bin/narwhal-rhino +68 -0
- data/engines/rhino/bin/narwhal-rhino.cmd +34 -0
- data/engines/rhino/bootstrap.js +119 -0
- data/engines/rhino/jars/jline.jar +0 -0
- data/engines/rhino/jars/jna.jar +0 -0
- data/engines/rhino/jars/js.jar +0 -0
- data/engines/rhino/lib/binary-engine.js +83 -0
- data/engines/rhino/lib/concurrency.js +6 -0
- data/engines/rhino/lib/event-queue.js +18 -0
- data/engines/rhino/lib/file-engine.js +216 -0
- data/engines/rhino/lib/http-client-engine.js +90 -0
- data/engines/rhino/lib/http-engine.js +10 -0
- data/engines/rhino/lib/io-engine.js +347 -0
- data/engines/rhino/lib/md5-engine.js +40 -0
- data/engines/rhino/lib/os-engine.js +150 -0
- data/engines/rhino/lib/packages-engine.js +71 -0
- data/engines/rhino/lib/sandbox-engine.js +70 -0
- data/engines/rhino/lib/system.js +38 -0
- data/engines/rhino/lib/worker-engine.js +23 -0
- data/engines/rhino/lib/zip.js +78 -0
- data/engines/rhino/package.json +4 -0
- data/engines/secure/lib/file.js +6 -0
- data/engines/secure/lib/system.js +6 -0
- data/engines/template/bin/narwhal-engine-name +32 -0
- data/engines/template/bootstrap.js +40 -0
- data/engines/template/lib/file-engine.js +118 -0
- data/engines/template/lib/system.js +17 -0
- data/examples/browser-deployment-jackconfig.js +35 -0
- data/examples/fibonacci-worker.js +35 -0
- data/examples/fibonacci.js +19 -0
- data/examples/hello +2 -0
- data/examples/narwhal +3 -0
- data/examples/not-quite-a-quine.js +1 -0
- data/extconf.rb +44 -0
- data/gem_bin/narwhal +5 -0
- data/gem_bin/sea +4 -0
- data/gem_bin/tusk +4 -0
- data/lib/args.js +849 -0
- data/lib/base16.js +16 -0
- data/lib/base64.js +120 -0
- data/lib/codec/base64.js +8 -0
- data/lib/crc32.js +60 -0
- data/lib/file-bootstrap.js +187 -0
- data/lib/file.js +659 -0
- data/lib/hash.js +28 -0
- data/lib/hashp.js +65 -0
- data/lib/html.js +16 -0
- data/lib/http-client.js +134 -0
- data/lib/http.js +17 -0
- data/lib/io.js +98 -0
- data/lib/jsmin.js +315 -0
- data/lib/jsonpath.js +89 -0
- data/lib/logger.js +55 -0
- data/lib/md4.js +146 -0
- data/lib/md5.js +164 -0
- data/lib/mime.js +166 -0
- data/lib/narwhal.js +102 -0
- data/lib/narwhal/client.js +261 -0
- data/lib/narwhal/compile.js +99 -0
- data/lib/narwhal/env.js +140 -0
- data/lib/narwhal/inline.js +106 -0
- data/lib/narwhal/json.js +324 -0
- data/lib/narwhal/json.md +178 -0
- data/lib/narwhal/repl.js +96 -0
- data/lib/narwhal/server-test.js +6 -0
- data/lib/narwhal/server.js +270 -0
- data/lib/narwhal/tusk.js +170 -0
- data/lib/narwhal/tusk/bin.js +13 -0
- data/lib/narwhal/tusk/bundle.js +0 -0
- data/lib/narwhal/tusk/catalog.js +22 -0
- data/lib/narwhal/tusk/clone.js +66 -0
- data/lib/narwhal/tusk/consolidate.js +25 -0
- data/lib/narwhal/tusk/create-catalog.js +80 -0
- data/lib/narwhal/tusk/engine.js +42 -0
- data/lib/narwhal/tusk/freeze.js +0 -0
- data/lib/narwhal/tusk/init.js +56 -0
- data/lib/narwhal/tusk/install.js +288 -0
- data/lib/narwhal/tusk/list.js +20 -0
- data/lib/narwhal/tusk/orphans.js +0 -0
- data/lib/narwhal/tusk/reheat.js +15 -0
- data/lib/narwhal/tusk/remove.js +15 -0
- data/lib/narwhal/tusk/search.js +145 -0
- data/lib/narwhal/tusk/update.js +21 -0
- data/lib/narwhal/tusk/upgrade.js +0 -0
- data/lib/os.js +33 -0
- data/lib/packages.js +423 -0
- data/lib/printf.js +169 -0
- data/lib/promise.js +352 -0
- data/lib/querystring.js +176 -0
- data/lib/ref-send.js +257 -0
- data/lib/regexp.js +12 -0
- data/lib/sandbox.js +422 -0
- data/lib/sha.js +112 -0
- data/lib/sha256.js +102 -0
- data/lib/struct.js +228 -0
- data/lib/term.js +179 -0
- data/lib/test/assert.js +95 -0
- data/lib/test/equiv.js +188 -0
- data/lib/test/jsdump.js +165 -0
- data/lib/test/runner.js +129 -0
- data/lib/unload.js +13 -0
- data/lib/uri.js +378 -0
- data/lib/url.js +5 -0
- data/lib/utf8.js +64 -0
- data/lib/util.js +985 -0
- data/lib/uuid.js +89 -0
- data/lib/xregexp.js +521 -0
- data/local.json.template +1 -0
- data/narwhal.gemspec +105 -0
- data/narwhal.js +213 -0
- data/package.json +26 -0
- data/packages/readline/engines/default/lib/readline.js +4 -0
- data/packages/readline/engines/rhino/lib/readline.js +6 -0
- data/packages/readline/package.json +5 -0
- data/sources.json +207 -0
- data/tests/all-tests.js +17 -0
- data/tests/args.js +31 -0
- data/tests/args/domain.js +215 -0
- data/tests/args/options.js +36 -0
- data/tests/args/shifting.js +92 -0
- data/tests/args/validation.js +31 -0
- data/tests/base64.js +23 -0
- data/tests/commonjs.js +3 -0
- data/tests/commonjs/all-tests.js +12 -0
- data/tests/commonjs/bytearray-encodings-tests.js +69 -0
- data/tests/commonjs/bytearray-tests.js +465 -0
- data/tests/commonjs/bytestring-encodings-tests.js +89 -0
- data/tests/commonjs/bytestring-tests.js +263 -0
- data/tests/commonjs/es5/all-tests.js +3 -0
- data/tests/commonjs/es5/bind.js +29 -0
- data/tests/commonjs/file-tests.js +315 -0
- data/tests/commonjs/file/dirname.js +31 -0
- data/tests/commonjs/file/extension.js +45 -0
- data/tests/commonjs/file/is-absolute.js +11 -0
- data/tests/commonjs/file/iterator.js +101 -0
- data/tests/commonjs/file/normal.js +27 -0
- data/tests/commonjs/file/path.js +17 -0
- data/tests/commonjs/file/relative.js +42 -0
- data/tests/commonjs/file/resolve.js +44 -0
- data/tests/commonjs/module-tests.js +9 -0
- data/tests/commonjs/modules/absolute/b.js +1 -0
- data/tests/commonjs/modules/absolute/program.js +5 -0
- data/tests/commonjs/modules/absolute/submodule/a.js +3 -0
- data/tests/commonjs/modules/absolute/test.js +9 -0
- data/tests/commonjs/modules/all-tests.js +47 -0
- data/tests/commonjs/modules/config.js +11 -0
- data/tests/commonjs/modules/cyclic/a.js +4 -0
- data/tests/commonjs/modules/cyclic/b.js +4 -0
- data/tests/commonjs/modules/cyclic/program.js +10 -0
- data/tests/commonjs/modules/cyclic/test.js +9 -0
- data/tests/commonjs/modules/determinism/program.js +3 -0
- data/tests/commonjs/modules/determinism/submodule/a.js +8 -0
- data/tests/commonjs/modules/determinism/submodule/b.js +2 -0
- data/tests/commonjs/modules/determinism/test.js +9 -0
- data/tests/commonjs/modules/exactExports/a.js +3 -0
- data/tests/commonjs/modules/exactExports/program.js +4 -0
- data/tests/commonjs/modules/exactExports/test.js +9 -0
- data/tests/commonjs/modules/hasOwnProperty/hasOwnProperty.js +0 -0
- data/tests/commonjs/modules/hasOwnProperty/program.js +3 -0
- data/tests/commonjs/modules/hasOwnProperty/test.js +9 -0
- data/tests/commonjs/modules/hasOwnProperty/toString.js +0 -0
- data/tests/commonjs/modules/method/a.js +12 -0
- data/tests/commonjs/modules/method/program.js +8 -0
- data/tests/commonjs/modules/method/test.js +9 -0
- data/tests/commonjs/modules/missing/program.js +8 -0
- data/tests/commonjs/modules/missing/test.js +9 -0
- data/tests/commonjs/modules/monkeys/a.js +1 -0
- data/tests/commonjs/modules/monkeys/program.js +4 -0
- data/tests/commonjs/modules/monkeys/test.js +9 -0
- data/tests/commonjs/modules/nested/a/b/c/d.js +3 -0
- data/tests/commonjs/modules/nested/program.js +3 -0
- data/tests/commonjs/modules/nested/test.js +9 -0
- data/tests/commonjs/modules/relative/program.js +5 -0
- data/tests/commonjs/modules/relative/submodule/a.js +1 -0
- data/tests/commonjs/modules/relative/submodule/b.js +2 -0
- data/tests/commonjs/modules/relative/test.js +9 -0
- data/tests/commonjs/modules/transitive/a.js +1 -0
- data/tests/commonjs/modules/transitive/b.js +1 -0
- data/tests/commonjs/modules/transitive/c.js +3 -0
- data/tests/commonjs/modules/transitive/program.js +3 -0
- data/tests/commonjs/modules/transitive/test.js +9 -0
- data/tests/file/all-tests.js +61 -0
- data/tests/file/fnmatch.js +102 -0
- data/tests/file/glob.js +466 -0
- data/tests/file/match.js +102 -0
- data/tests/global.js +6 -0
- data/tests/global/array.js +19 -0
- data/tests/hashes.js +94 -0
- data/tests/html.js +13 -0
- data/tests/io/stringio.js +21 -0
- data/tests/os/all-tests.js +4 -0
- data/tests/os/popen.js +41 -0
- data/tests/os/system.js +22 -0
- data/tests/printf.js +123 -0
- data/tests/query-string.js +87 -0
- data/tests/sandbox/byte-io.js +20 -0
- data/tests/sandbox/fileName.js +3 -0
- data/tests/sandbox/foo.js +0 -0
- data/tests/sandbox/reload.js +79 -0
- data/tests/string.js +35 -0
- data/tests/uri.js +41 -0
- data/tests/util/all-tests.js +79 -0
- data/tests/util/array.js +207 -0
- data/tests/util/array/is-arguments.js +29 -0
- data/tests/util/array/is-array-like.js +29 -0
- data/tests/util/case.js +9 -0
- data/tests/util/collection.js +104 -0
- data/tests/util/eq.js +57 -0
- data/tests/util/expand.js +45 -0
- data/tests/util/object.js +125 -0
- data/tests/util/operator.js +25 -0
- data/tests/util/range.js +19 -0
- data/tests/util/repr.js +26 -0
- data/tests/util/string.js +34 -0
- data/tests/util/unique.js +12 -0
- 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
|
+
|