spade 0.0.8.1 → 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- data/.gitignore +0 -1
- data/.gitmodules +4 -4
- data/README.md +4 -0
- data/bin/spade +1 -1
- data/lib/spade.rb +0 -51
- data/lib/spade/cli.rb +10 -4
- data/lib/spade/version.rb +1 -1
- data/spade.gemspec +4 -21
- metadata +13 -482
- data/Gemfile +0 -3
- data/Gemfile.lock +0 -40
- data/lib/index.js +0 -14
- data/lib/node/loader.js +0 -146
- data/lib/node/sandbox.js +0 -44
- data/lib/spade.js +0 -1130
- data/lib/spade/bundle.rb +0 -171
- data/lib/spade/cli/base.rb +0 -354
- data/lib/spade/cli/owner.rb +0 -45
- data/lib/spade/cli/project_generator.rb +0 -58
- data/lib/spade/compiler.rb +0 -34
- data/lib/spade/console.rb +0 -39
- data/lib/spade/context.rb +0 -116
- data/lib/spade/credentials.rb +0 -36
- data/lib/spade/dependency_installer.rb +0 -103
- data/lib/spade/environment.rb +0 -35
- data/lib/spade/exports.rb +0 -71
- data/lib/spade/installer.rb +0 -40
- data/lib/spade/loader.rb +0 -238
- data/lib/spade/local.rb +0 -46
- data/lib/spade/package.rb +0 -157
- data/lib/spade/reactor.rb +0 -159
- data/lib/spade/remote.rb +0 -99
- data/lib/spade/repository.rb +0 -18
- data/lib/spade/server.rb +0 -66
- data/lib/spade/shell.rb +0 -36
- data/lib/spade/templates/project/LICENSE +0 -19
- data/lib/spade/templates/project/README.md +0 -21
- data/lib/spade/templates/project/lib/main.js +0 -9
- data/lib/spade/templates/project/project.json +0 -31
- data/lib/spade/templates/project/tests/main-test.js +0 -8
- data/package.json +0 -27
- data/packages/coffee-script/.gitignore +0 -8
- data/packages/coffee-script/.npmignore +0 -11
- data/packages/coffee-script/Cakefile +0 -229
- data/packages/coffee-script/LICENSE +0 -22
- data/packages/coffee-script/README +0 -47
- data/packages/coffee-script/Rakefile +0 -78
- data/packages/coffee-script/bin/cake +0 -7
- data/packages/coffee-script/bin/coffee +0 -7
- data/packages/coffee-script/documentation/coffee/aliases.coffee +0 -11
- data/packages/coffee-script/documentation/coffee/array_comprehensions.coffee +0 -2
- data/packages/coffee-script/documentation/coffee/block_comment.coffee +0 -6
- data/packages/coffee-script/documentation/coffee/cake_tasks.coffee +0 -9
- data/packages/coffee-script/documentation/coffee/classes.coffee +0 -25
- data/packages/coffee-script/documentation/coffee/comparisons.coffee +0 -5
- data/packages/coffee-script/documentation/coffee/conditionals.coffee +0 -13
- data/packages/coffee-script/documentation/coffee/default_args.coffee +0 -8
- data/packages/coffee-script/documentation/coffee/do.coffee +0 -4
- data/packages/coffee-script/documentation/coffee/embedded.coffee +0 -5
- data/packages/coffee-script/documentation/coffee/existence.coffee +0 -10
- data/packages/coffee-script/documentation/coffee/expressions.coffee +0 -9
- data/packages/coffee-script/documentation/coffee/expressions_assignment.coffee +0 -3
- data/packages/coffee-script/documentation/coffee/expressions_comprehension.coffee +0 -3
- data/packages/coffee-script/documentation/coffee/expressions_try.coffee +0 -7
- data/packages/coffee-script/documentation/coffee/fat_arrow.coffee +0 -6
- data/packages/coffee-script/documentation/coffee/functions.coffee +0 -2
- data/packages/coffee-script/documentation/coffee/heredocs.coffee +0 -7
- data/packages/coffee-script/documentation/coffee/heregexes.coffee +0 -11
- data/packages/coffee-script/documentation/coffee/interpolation.coffee +0 -6
- data/packages/coffee-script/documentation/coffee/multiple_return_values.coffee +0 -7
- data/packages/coffee-script/documentation/coffee/object_comprehensions.coffee +0 -4
- data/packages/coffee-script/documentation/coffee/object_extraction.coffee +0 -13
- data/packages/coffee-script/documentation/coffee/objects_and_arrays.coffee +0 -19
- data/packages/coffee-script/documentation/coffee/objects_reserved.coffee +0 -5
- data/packages/coffee-script/documentation/coffee/overview.coffee +0 -28
- data/packages/coffee-script/documentation/coffee/parallel_assignment.coffee +0 -6
- data/packages/coffee-script/documentation/coffee/patterns_and_splats.coffee +0 -7
- data/packages/coffee-script/documentation/coffee/prototypes.coffee +0 -3
- data/packages/coffee-script/documentation/coffee/range_comprehensions.coffee +0 -2
- data/packages/coffee-script/documentation/coffee/scope.coffee +0 -5
- data/packages/coffee-script/documentation/coffee/slices.coffee +0 -7
- data/packages/coffee-script/documentation/coffee/soaks.coffee +0 -1
- data/packages/coffee-script/documentation/coffee/splats.coffee +0 -27
- data/packages/coffee-script/documentation/coffee/splices.coffee +0 -5
- data/packages/coffee-script/documentation/coffee/strings.coffee +0 -8
- data/packages/coffee-script/documentation/coffee/switch.coffee +0 -10
- data/packages/coffee-script/documentation/coffee/try.coffee +0 -8
- data/packages/coffee-script/documentation/coffee/while.coffee +0 -10
- data/packages/coffee-script/documentation/css/docs.css +0 -374
- data/packages/coffee-script/documentation/css/idle.css +0 -64
- data/packages/coffee-script/documentation/docs/browser.html +0 -25
- data/packages/coffee-script/documentation/docs/cake.html +0 -43
- data/packages/coffee-script/documentation/docs/coffee-script.html +0 -51
- data/packages/coffee-script/documentation/docs/command.html +0 -161
- data/packages/coffee-script/documentation/docs/docco.css +0 -186
- data/packages/coffee-script/documentation/docs/grammar.html +0 -399
- data/packages/coffee-script/documentation/docs/helpers.html +0 -31
- data/packages/coffee-script/documentation/docs/index.html +0 -3
- data/packages/coffee-script/documentation/docs/lexer.html +0 -490
- data/packages/coffee-script/documentation/docs/nodes.html +0 -1338
- data/packages/coffee-script/documentation/docs/optparse.html +0 -78
- data/packages/coffee-script/documentation/docs/repl.html +0 -24
- data/packages/coffee-script/documentation/docs/rewriter.html +0 -251
- data/packages/coffee-script/documentation/docs/scope.html +0 -54
- data/packages/coffee-script/documentation/docs/underscore.html +0 -295
- data/packages/coffee-script/documentation/images/background.png +0 -0
- data/packages/coffee-script/documentation/images/banding.png +0 -0
- data/packages/coffee-script/documentation/images/button_bg.png +0 -0
- data/packages/coffee-script/documentation/images/button_bg_dark.gif +0 -0
- data/packages/coffee-script/documentation/images/button_bg_green.gif +0 -0
- data/packages/coffee-script/documentation/images/favicon.ico +0 -0
- data/packages/coffee-script/documentation/images/logo.png +0 -0
- data/packages/coffee-script/documentation/images/screenshadow.png +0 -0
- data/packages/coffee-script/documentation/index.html.erb +0 -1607
- data/packages/coffee-script/documentation/js/aliases.js +0 -17
- data/packages/coffee-script/documentation/js/array_comprehensions.js +0 -6
- data/packages/coffee-script/documentation/js/block_comment.js +0 -4
- data/packages/coffee-script/documentation/js/cake_tasks.js +0 -10
- data/packages/coffee-script/documentation/js/classes.js +0 -44
- data/packages/coffee-script/documentation/js/comparisons.js +0 -3
- data/packages/coffee-script/documentation/js/conditionals.js +0 -12
- data/packages/coffee-script/documentation/js/default_args.js +0 -7
- data/packages/coffee-script/documentation/js/do.js +0 -10
- data/packages/coffee-script/documentation/js/embedded.js +0 -4
- data/packages/coffee-script/documentation/js/existence.js +0 -6
- data/packages/coffee-script/documentation/js/expressions.js +0 -15
- data/packages/coffee-script/documentation/js/expressions_assignment.js +0 -2
- data/packages/coffee-script/documentation/js/expressions_comprehension.js +0 -9
- data/packages/coffee-script/documentation/js/expressions_try.js +0 -7
- data/packages/coffee-script/documentation/js/fat_arrow.js +0 -9
- data/packages/coffee-script/documentation/js/functions.js +0 -7
- data/packages/coffee-script/documentation/js/heredocs.js +0 -2
- data/packages/coffee-script/documentation/js/heregexes.js +0 -2
- data/packages/coffee-script/documentation/js/interpolation.js +0 -4
- data/packages/coffee-script/documentation/js/multiple_return_values.js +0 -5
- data/packages/coffee-script/documentation/js/object_comprehensions.js +0 -15
- data/packages/coffee-script/documentation/js/object_extraction.js +0 -10
- data/packages/coffee-script/documentation/js/objects_and_arrays.js +0 -17
- data/packages/coffee-script/documentation/js/objects_reserved.js +0 -4
- data/packages/coffee-script/documentation/js/overview.js +0 -35
- data/packages/coffee-script/documentation/js/parallel_assignment.js +0 -4
- data/packages/coffee-script/documentation/js/patterns_and_splats.js +0 -4
- data/packages/coffee-script/documentation/js/prototypes.js +0 -3
- data/packages/coffee-script/documentation/js/range_comprehensions.js +0 -9
- data/packages/coffee-script/documentation/js/scope.js +0 -8
- data/packages/coffee-script/documentation/js/slices.js +0 -4
- data/packages/coffee-script/documentation/js/soaks.js +0 -2
- data/packages/coffee-script/documentation/js/splats.js +0 -15
- data/packages/coffee-script/documentation/js/splices.js +0 -3
- data/packages/coffee-script/documentation/js/strings.js +0 -2
- data/packages/coffee-script/documentation/js/switch.js +0 -23
- data/packages/coffee-script/documentation/js/try.js +0 -8
- data/packages/coffee-script/documentation/js/while.js +0 -18
- data/packages/coffee-script/documentation/vendor/jquery-1.4.2.js +0 -6240
- data/packages/coffee-script/examples/beautiful_code/binary_search.coffee +0 -16
- data/packages/coffee-script/examples/beautiful_code/quicksort_runtime.coffee +0 -13
- data/packages/coffee-script/examples/beautiful_code/regular_expression_matcher.coffee +0 -34
- data/packages/coffee-script/examples/blocks.coffee +0 -54
- data/packages/coffee-script/examples/code.coffee +0 -167
- data/packages/coffee-script/examples/computer_science/README +0 -4
- data/packages/coffee-script/examples/computer_science/binary_search.coffee +0 -25
- data/packages/coffee-script/examples/computer_science/bubble_sort.coffee +0 -11
- data/packages/coffee-script/examples/computer_science/linked_list.coffee +0 -108
- data/packages/coffee-script/examples/computer_science/luhn_algorithm.coffee +0 -36
- data/packages/coffee-script/examples/computer_science/merge_sort.coffee +0 -19
- data/packages/coffee-script/examples/computer_science/selection_sort.coffee +0 -23
- data/packages/coffee-script/examples/poignant.coffee +0 -181
- data/packages/coffee-script/examples/potion.coffee +0 -206
- data/packages/coffee-script/examples/underscore.coffee +0 -682
- data/packages/coffee-script/examples/web_server.coffee +0 -12
- data/packages/coffee-script/extras/EXTRAS +0 -7
- data/packages/coffee-script/extras/coffee-script.js +0 -8
- data/packages/coffee-script/extras/jsl.conf +0 -44
- data/packages/coffee-script/index.html +0 -2515
- data/packages/coffee-script/lib/browser.js +0 -52
- data/packages/coffee-script/lib/cake.js +0 -76
- data/packages/coffee-script/lib/coffee-script.js +0 -82
- data/packages/coffee-script/lib/command.js +0 -263
- data/packages/coffee-script/lib/grammar.js +0 -581
- data/packages/coffee-script/lib/helpers.js +0 -66
- data/packages/coffee-script/lib/index.js +0 -8
- data/packages/coffee-script/lib/lexer.js +0 -633
- data/packages/coffee-script/lib/nodes.js +0 -2165
- data/packages/coffee-script/lib/optparse.js +0 -111
- data/packages/coffee-script/lib/parser.js +0 -649
- data/packages/coffee-script/lib/repl.js +0 -42
- data/packages/coffee-script/lib/rewriter.js +0 -353
- data/packages/coffee-script/lib/scope.js +0 -120
- data/packages/coffee-script/lib/spade-format.js +0 -45
- data/packages/coffee-script/package.json +0 -26
- data/packages/coffee-script/src/browser.coffee +0 -43
- data/packages/coffee-script/src/cake.coffee +0 -69
- data/packages/coffee-script/src/coffee-script.coffee +0 -92
- data/packages/coffee-script/src/command.coffee +0 -214
- data/packages/coffee-script/src/grammar.coffee +0 -590
- data/packages/coffee-script/src/helpers.coffee +0 -56
- data/packages/coffee-script/src/index.coffee +0 -2
- data/packages/coffee-script/src/lexer.coffee +0 -653
- data/packages/coffee-script/src/nodes.coffee +0 -1754
- data/packages/coffee-script/src/optparse.coffee +0 -99
- data/packages/coffee-script/src/repl.coffee +0 -42
- data/packages/coffee-script/src/rewriter.coffee +0 -326
- data/packages/coffee-script/src/scope.coffee +0 -94
- data/packages/coffee-script/test/arguments.coffee +0 -127
- data/packages/coffee-script/test/assignment.coffee +0 -98
- data/packages/coffee-script/test/break.coffee +0 -18
- data/packages/coffee-script/test/comments.coffee +0 -201
- data/packages/coffee-script/test/conditionals.coffee +0 -181
- data/packages/coffee-script/test/exception_handling.coffee +0 -90
- data/packages/coffee-script/test/helpers.coffee +0 -96
- data/packages/coffee-script/test/importing.coffee +0 -18
- data/packages/coffee-script/test/operators.coffee +0 -225
- data/packages/coffee-script/test/ranges_slices_and_splices.coffee +0 -186
- data/packages/coffee-script/test/regular_expressions.coffee +0 -56
- data/packages/coffee-script/test/test.html +0 -123
- data/packages/coffee-script/test/test_chaining.coffee +0 -77
- data/packages/coffee-script/test/test_classes.coffee +0 -372
- data/packages/coffee-script/test/test_compilation.coffee +0 -26
- data/packages/coffee-script/test/test_comprehensions.coffee +0 -318
- data/packages/coffee-script/test/test_existence.coffee +0 -165
- data/packages/coffee-script/test/test_functions.coffee +0 -379
- data/packages/coffee-script/test/test_heredocs.coffee +0 -111
- data/packages/coffee-script/test/test_literals.coffee +0 -270
- data/packages/coffee-script/test/test_option_parser.coffee +0 -27
- data/packages/coffee-script/test/test_pattern_matching.coffee +0 -162
- data/packages/coffee-script/test/test_returns.coffee +0 -63
- data/packages/coffee-script/test/test_splats.coffee +0 -102
- data/packages/coffee-script/test/test_strings.coffee +0 -118
- data/packages/coffee-script/test/test_switch.coffee +0 -103
- data/packages/coffee-script/test/test_while.coffee +0 -71
- data/packages/ivory/LICENSE.txt +0 -1
- data/packages/ivory/README.md +0 -19
- data/packages/ivory/lib/buffer.js +0 -111
- data/packages/ivory/lib/events.js +0 -137
- data/packages/ivory/lib/fs.js +0 -266
- data/packages/ivory/lib/main.js +0 -13
- data/packages/ivory/lib/path.js +0 -158
- data/packages/ivory/lib/ruby/buffer.rb +0 -145
- data/packages/ivory/lib/ruby/constants.rb +0 -585
- data/packages/ivory/lib/ruby/events.rb +0 -32
- data/packages/ivory/lib/ruby/fs.rb +0 -245
- data/packages/ivory/lib/ruby/process.rb +0 -28
- data/packages/ivory/lib/stream.js +0 -115
- data/packages/ivory/lib/util.js +0 -414
- data/packages/ivory/package.json +0 -11
- data/packages/jquery/main.js +0 -7179
- data/packages/jquery/package.json +0 -10
- data/packages/json/lib/main.js +0 -14
- data/packages/json/package.json +0 -8
- data/packages/lproj/README.md +0 -77
- data/packages/lproj/examples/demo-app/en.lproj/localized.strings +0 -2
- data/packages/lproj/examples/demo-app/fr.lproj/localized.strings +0 -3
- data/packages/lproj/examples/demo-app/index.html +0 -8
- data/packages/lproj/examples/demo-app/lib/main.js +0 -7
- data/packages/lproj/examples/demo-app/package.json +0 -9
- data/packages/lproj/lib/main.js +0 -78
- data/packages/lproj/lib/strings-format.js +0 -6
- data/packages/lproj/package.json +0 -9
- data/packages/optparse/README.md +0 -161
- data/packages/optparse/TODO +0 -1
- data/packages/optparse/examples/browser-test.html +0 -75
- data/packages/optparse/examples/nodejs-test.js +0 -90
- data/packages/optparse/lib/optparse.js +0 -309
- data/packages/optparse/package.json +0 -13
- data/packages/optparse/seed.yml +0 -5
- data/packages/text/lib/main.js +0 -8
- data/packages/text/package.json +0 -9
- data/packages/web-file/README.md +0 -7
- data/packages/web-file/lib/errors.js +0 -32
- data/packages/web-file/lib/file-reader.js +0 -10
- data/packages/web-file/lib/file-system.js +0 -234
- data/packages/web-file/lib/file-writer.js +0 -10
- data/packages/web-file/lib/file.js +0 -9
- data/packages/web-file/lib/main.js +0 -34
- data/packages/web-file/lib/platform.js +0 -25
- data/packages/web-file/lib/ruby/file.rb +0 -252
- data/packages/web-file/lib/ruby/file_reader.rb +0 -69
- data/packages/web-file/lib/ruby/file_system.rb +0 -134
- data/packages/web-file/lib/ruby/file_writer.rb +0 -78
- data/packages/web-file/package.json +0 -12
- data/packages/web-typed-array/README.md +0 -7
- data/packages/web-typed-array/lib/array-buffer-view.js +0 -9
- data/packages/web-typed-array/lib/array-buffer.js +0 -7
- data/packages/web-typed-array/lib/main.js +0 -33
- data/packages/web-typed-array/lib/platform.js +0 -20
- data/packages/web-typed-array/lib/ruby/array_buffer.rb +0 -31
- data/packages/web-typed-array/lib/ruby/array_buffer_view.rb +0 -130
- data/packages/web-typed-array/lib/ruby/typed_array.rb +0 -133
- data/packages/web-typed-array/lib/typed-array.js +0 -26
- data/packages/web-typed-array/package.json +0 -9
- data/spec/cli/build_spec.rb +0 -59
- data/spec/cli/install_spec.rb +0 -120
- data/spec/cli/installed_spec.rb +0 -55
- data/spec/cli/list_spec.rb +0 -75
- data/spec/cli/login_spec.rb +0 -76
- data/spec/cli/new_spec.rb +0 -5
- data/spec/cli/owner_spec.rb +0 -115
- data/spec/cli/push_spec.rb +0 -74
- data/spec/cli/uninstall_spec.rb +0 -58
- data/spec/cli/unpack_spec.rb +0 -73
- data/spec/cli/unyank_spec.rb +0 -74
- data/spec/cli/update_spec.rb +0 -8
- data/spec/cli/yank_spec.rb +0 -74
- data/spec/credentials_spec.rb +0 -24
- data/spec/fixtures/coffee-1.0.1.pre.spd +0 -0
- data/spec/fixtures/core-test-0.4.3.spd +0 -0
- data/spec/fixtures/core-test/bin/cot +0 -3
- data/spec/fixtures/core-test/lib/main.js +0 -1
- data/spec/fixtures/core-test/resources/runner.css +0 -0
- data/spec/fixtures/core-test/tests/test.js +0 -1
- data/spec/fixtures/ivory-0.0.1.spd +0 -0
- data/spec/fixtures/jquery-1.4.3.spd +0 -0
- data/spec/fixtures/optparse-1.0.1.spd +0 -0
- data/spec/fixtures/package.json +0 -30
- data/spec/gauntlet_spec.rb +0 -27
- data/spec/javascript/async-test.js +0 -123
- data/spec/javascript/compiler/javascript.js +0 -13
- data/spec/javascript/compiler/ruby.js +0 -14
- data/spec/javascript/loader-test.js +0 -64
- data/spec/javascript/normalize-test.js +0 -73
- data/spec/javascript/packages-test.js +0 -44
- data/spec/javascript/relative-require-test.js +0 -72
- data/spec/javascript/require-test.js +0 -117
- data/spec/javascript/sandbox/compile.js +0 -37
- data/spec/javascript/sandbox/creation.js +0 -44
- data/spec/javascript/sandbox/format.js +0 -79
- data/spec/javascript/sandbox/misc.js +0 -57
- data/spec/javascript/sandbox/preprocessor.js +0 -81
- data/spec/javascript/sandbox/require.js +0 -48
- data/spec/javascript/sandbox/run-command.js +0 -21
- data/spec/javascript/spade/externs.js +0 -14
- data/spec/javascript/spade/load-factory.js +0 -15
- data/spec/javascript/spade/misc.js +0 -23
- data/spec/javascript/spade/ready.js +0 -12
- data/spec/javascript/spade/register.js +0 -13
- data/spec/javascript_spec.rb +0 -7
- data/spec/package_spec.rb +0 -267
- data/spec/spec_helper.rb +0 -30
- data/spec/support/cli.rb +0 -94
- data/spec/support/core_test.rb +0 -59
- data/spec/support/fake.rb +0 -44
- data/spec/support/fake_gem_server.rb +0 -66
- data/spec/support/fake_gemcutter.rb +0 -49
- data/spec/support/matchers.rb +0 -31
- data/spec/support/path.rb +0 -54
- data/test-spade.html +0 -8
@@ -1,99 +0,0 @@
|
|
1
|
-
# A simple **OptionParser** class to parse option flags from the command-line.
|
2
|
-
# Use it like so:
|
3
|
-
#
|
4
|
-
# parser = new OptionParser switches, helpBanner
|
5
|
-
# options = parser.parse process.argv
|
6
|
-
#
|
7
|
-
# The first non-option is considered to be the start of the file (and file
|
8
|
-
# option) list, and all subsequent arguments are left unparsed.
|
9
|
-
exports.OptionParser = class OptionParser
|
10
|
-
|
11
|
-
# Initialize with a list of valid options, in the form:
|
12
|
-
#
|
13
|
-
# [short-flag, long-flag, description]
|
14
|
-
#
|
15
|
-
# Along with an an optional banner for the usage help.
|
16
|
-
constructor: (rules, @banner) ->
|
17
|
-
@rules = buildRules rules
|
18
|
-
|
19
|
-
# Parse the list of arguments, populating an `options` object with all of the
|
20
|
-
# specified options, and returning it. `options.arguments` will be an array
|
21
|
-
# containing the remaining non-option arguments. `options.literals` will be
|
22
|
-
# an array of options that are meant to be passed through directly to the
|
23
|
-
# executing script. This is a simpler API than many option parsers that allow
|
24
|
-
# you to attach callback actions for every flag. Instead, you're responsible
|
25
|
-
# for interpreting the options object.
|
26
|
-
parse: (args) ->
|
27
|
-
options = arguments: [], literals: []
|
28
|
-
args = normalizeArguments args
|
29
|
-
for arg, i in args
|
30
|
-
if arg is '--'
|
31
|
-
options.literals = args[(i + 1)..]
|
32
|
-
break
|
33
|
-
isOption = !!(arg.match(LONG_FLAG) or arg.match(SHORT_FLAG))
|
34
|
-
matchedRule = no
|
35
|
-
for rule in @rules
|
36
|
-
if rule.shortFlag is arg or rule.longFlag is arg
|
37
|
-
value = if rule.hasArgument then args[i += 1] else true
|
38
|
-
options[rule.name] = if rule.isList then (options[rule.name] or []).concat value else value
|
39
|
-
matchedRule = yes
|
40
|
-
break
|
41
|
-
throw new Error "unrecognized option: #{arg}" if isOption and not matchedRule
|
42
|
-
if not isOption
|
43
|
-
options.arguments = args.slice i
|
44
|
-
break
|
45
|
-
options
|
46
|
-
|
47
|
-
# Return the help text for this **OptionParser**, listing and describing all
|
48
|
-
# of the valid options, for `--help` and such.
|
49
|
-
help: ->
|
50
|
-
lines = ['Available options:']
|
51
|
-
lines.unshift "#{@banner}\n" if @banner
|
52
|
-
for rule in @rules
|
53
|
-
spaces = 15 - rule.longFlag.length
|
54
|
-
spaces = if spaces > 0 then Array(spaces + 1).join(' ') else ''
|
55
|
-
letPart = if rule.shortFlag then rule.shortFlag + ', ' else ' '
|
56
|
-
lines.push ' ' + letPart + rule.longFlag + spaces + rule.description
|
57
|
-
"\n#{ lines.join('\n') }\n"
|
58
|
-
|
59
|
-
# Helpers
|
60
|
-
# -------
|
61
|
-
|
62
|
-
# Regex matchers for option flags.
|
63
|
-
LONG_FLAG = /^(--\w[\w\-]+)/
|
64
|
-
SHORT_FLAG = /^(-\w)/
|
65
|
-
MULTI_FLAG = /^-(\w{2,})/
|
66
|
-
OPTIONAL = /\[(\w+(\*?))\]/
|
67
|
-
|
68
|
-
# Build and return the list of option rules. If the optional *short-flag* is
|
69
|
-
# unspecified, leave it out by padding with `null`.
|
70
|
-
buildRules = (rules) ->
|
71
|
-
for tuple in rules
|
72
|
-
tuple.unshift null if tuple.length < 3
|
73
|
-
buildRule tuple...
|
74
|
-
|
75
|
-
# Build a rule from a `-o` short flag, a `--output [DIR]` long flag, and the
|
76
|
-
# description of what the option does.
|
77
|
-
buildRule = (shortFlag, longFlag, description, options = {}) ->
|
78
|
-
match = longFlag.match(OPTIONAL)
|
79
|
-
longFlag = longFlag.match(LONG_FLAG)[1]
|
80
|
-
{
|
81
|
-
name: longFlag.substr 2
|
82
|
-
shortFlag: shortFlag
|
83
|
-
longFlag: longFlag
|
84
|
-
description: description
|
85
|
-
hasArgument: !!(match and match[1])
|
86
|
-
isList: !!(match and match[2])
|
87
|
-
}
|
88
|
-
|
89
|
-
# Normalize arguments by expanding merged flags into multiple flags. This allows
|
90
|
-
# you to have `-wl` be the same as `--watch --lint`.
|
91
|
-
normalizeArguments = (args) ->
|
92
|
-
args = args.slice 0
|
93
|
-
result = []
|
94
|
-
for arg in args
|
95
|
-
if match = arg.match MULTI_FLAG
|
96
|
-
result.push '-' + l for l in match[1].split ''
|
97
|
-
else
|
98
|
-
result.push arg
|
99
|
-
result
|
@@ -1,42 +0,0 @@
|
|
1
|
-
# A very simple Read-Eval-Print-Loop. Compiles one line at a time to JavaScript
|
2
|
-
# and evaluates it. Good for simple tests, or poking around the **Node.js** API.
|
3
|
-
# Using it looks like this:
|
4
|
-
#
|
5
|
-
# coffee> console.log "#{num} bottles of beer" for num in [99..1]
|
6
|
-
|
7
|
-
# Require the **coffee-script** module to get access to the compiler.
|
8
|
-
CoffeeScript = require './coffee-script'
|
9
|
-
helpers = require './helpers'
|
10
|
-
readline = require 'readline'
|
11
|
-
|
12
|
-
# Start by opening up **stdio**.
|
13
|
-
stdio = process.openStdin()
|
14
|
-
|
15
|
-
# Log an error.
|
16
|
-
error = (err) ->
|
17
|
-
stdio.write (err.stack or err.toString()) + '\n\n'
|
18
|
-
|
19
|
-
# Quick alias for quitting the REPL.
|
20
|
-
helpers.extend global, quit: -> process.exit(0)
|
21
|
-
|
22
|
-
# The main REPL function. **run** is called every time a line of code is entered.
|
23
|
-
# Attempt to evaluate the command. If there's an exception, print it out instead
|
24
|
-
# of exiting.
|
25
|
-
run = (buffer) ->
|
26
|
-
try
|
27
|
-
val = CoffeeScript.eval buffer.toString(), bare: on, globals: on, fileName: 'repl'
|
28
|
-
console.log val if val isnt undefined
|
29
|
-
catch err
|
30
|
-
error err
|
31
|
-
repl.prompt()
|
32
|
-
|
33
|
-
# Make sure that uncaught exceptions don't kill the REPL.
|
34
|
-
process.on 'uncaughtException', error
|
35
|
-
|
36
|
-
# Create the REPL by listening to **stdin**.
|
37
|
-
repl = readline.createInterface stdio
|
38
|
-
repl.setPrompt 'coffee> '
|
39
|
-
stdio.on 'data', (buffer) -> repl.write buffer
|
40
|
-
repl.on 'close', -> stdio.destroy()
|
41
|
-
repl.on 'line', run
|
42
|
-
repl.prompt()
|
@@ -1,326 +0,0 @@
|
|
1
|
-
# The CoffeeScript language has a good deal of optional syntax, implicit syntax,
|
2
|
-
# and shorthand syntax. This can greatly complicate a grammar and bloat
|
3
|
-
# the resulting parse table. Instead of making the parser handle it all, we take
|
4
|
-
# a series of passes over the token stream, using this **Rewriter** to convert
|
5
|
-
# shorthand into the unambiguous long form, add implicit indentation and
|
6
|
-
# parentheses, balance incorrect nestings, and generally clean things up.
|
7
|
-
|
8
|
-
# The **Rewriter** class is used by the [Lexer](lexer.html), directly against
|
9
|
-
# its internal array of tokens.
|
10
|
-
class exports.Rewriter
|
11
|
-
|
12
|
-
# Helpful snippet for debugging:
|
13
|
-
# console.log (t[0] + '/' + t[1] for t in @tokens).join ' '
|
14
|
-
|
15
|
-
# Rewrite the token stream in multiple passes, one logical filter at
|
16
|
-
# a time. This could certainly be changed into a single pass through the
|
17
|
-
# stream, with a big ol' efficient switch, but it's much nicer to work with
|
18
|
-
# like this. The order of these passes matters -- indentation must be
|
19
|
-
# corrected before implicit parentheses can be wrapped around blocks of code.
|
20
|
-
rewrite: (@tokens) ->
|
21
|
-
@removeLeadingNewlines()
|
22
|
-
@removeMidExpressionNewlines()
|
23
|
-
@closeOpenCalls()
|
24
|
-
@closeOpenIndexes()
|
25
|
-
@addImplicitIndentation()
|
26
|
-
@tagPostfixConditionals()
|
27
|
-
@addImplicitBraces()
|
28
|
-
@addImplicitParentheses()
|
29
|
-
@ensureBalance BALANCED_PAIRS
|
30
|
-
@rewriteClosingParens()
|
31
|
-
@tokens
|
32
|
-
|
33
|
-
# Rewrite the token stream, looking one token ahead and behind.
|
34
|
-
# Allow the return value of the block to tell us how many tokens to move
|
35
|
-
# forwards (or backwards) in the stream, to make sure we don't miss anything
|
36
|
-
# as tokens are inserted and removed, and the stream changes length under
|
37
|
-
# our feet.
|
38
|
-
scanTokens: (block) ->
|
39
|
-
{tokens} = this
|
40
|
-
i = 0
|
41
|
-
i += block.call this, token, i, tokens while token = tokens[i]
|
42
|
-
true
|
43
|
-
|
44
|
-
detectEnd: (i, condition, action) ->
|
45
|
-
{tokens} = this
|
46
|
-
levels = 0
|
47
|
-
while token = tokens[i]
|
48
|
-
return action.call this, token, i if levels is 0 and condition.call this, token, i
|
49
|
-
return action.call this, token, i - 1 if not token or levels < 0
|
50
|
-
if token[0] in EXPRESSION_START
|
51
|
-
levels += 1
|
52
|
-
else if token[0] in EXPRESSION_END
|
53
|
-
levels -= 1
|
54
|
-
i += 1
|
55
|
-
i - 1
|
56
|
-
|
57
|
-
# Leading newlines would introduce an ambiguity in the grammar, so we
|
58
|
-
# dispatch them here.
|
59
|
-
removeLeadingNewlines: ->
|
60
|
-
break for [tag], i in @tokens when tag isnt 'TERMINATOR'
|
61
|
-
@tokens.splice 0, i if i
|
62
|
-
|
63
|
-
# Some blocks occur in the middle of expressions -- when we're expecting
|
64
|
-
# this, remove their trailing newlines.
|
65
|
-
removeMidExpressionNewlines: ->
|
66
|
-
@scanTokens (token, i, tokens) ->
|
67
|
-
return 1 unless token[0] is 'TERMINATOR' and @tag(i + 1) in EXPRESSION_CLOSE
|
68
|
-
tokens.splice i, 1
|
69
|
-
0
|
70
|
-
|
71
|
-
# The lexer has tagged the opening parenthesis of a method call. Match it with
|
72
|
-
# its paired close. We have the mis-nested outdent case included here for
|
73
|
-
# calls that close on the same line, just before their outdent.
|
74
|
-
closeOpenCalls: ->
|
75
|
-
condition = (token, i) ->
|
76
|
-
token[0] in [')', 'CALL_END'] or
|
77
|
-
token[0] is 'OUTDENT' and @tag(i - 1) is ')'
|
78
|
-
action = (token, i) ->
|
79
|
-
@tokens[if token[0] is 'OUTDENT' then i - 1 else i][0] = 'CALL_END'
|
80
|
-
@scanTokens (token, i) ->
|
81
|
-
@detectEnd i + 1, condition, action if token[0] is 'CALL_START'
|
82
|
-
1
|
83
|
-
|
84
|
-
# The lexer has tagged the opening parenthesis of an indexing operation call.
|
85
|
-
# Match it with its paired close.
|
86
|
-
closeOpenIndexes: ->
|
87
|
-
condition = (token, i) -> token[0] in [']', 'INDEX_END']
|
88
|
-
action = (token, i) -> token[0] = 'INDEX_END'
|
89
|
-
@scanTokens (token, i) ->
|
90
|
-
@detectEnd i + 1, condition, action if token[0] is 'INDEX_START'
|
91
|
-
1
|
92
|
-
|
93
|
-
# Object literals may be written with implicit braces, for simple cases.
|
94
|
-
# Insert the missing braces here, so that the parser doesn't have to.
|
95
|
-
addImplicitBraces: ->
|
96
|
-
stack = []
|
97
|
-
start = null
|
98
|
-
startIndent = 0
|
99
|
-
condition = (token, i) ->
|
100
|
-
[one, two, three] = @tokens[i + 1 .. i + 3]
|
101
|
-
return false if 'HERECOMMENT' is one?[0]
|
102
|
-
[tag] = token
|
103
|
-
(tag in ['TERMINATOR', 'OUTDENT'] and
|
104
|
-
not (two?[0] is ':' or one?[0] is '@' and three?[0] is ':')) or
|
105
|
-
(tag is ',' and one and
|
106
|
-
one[0] not in ['IDENTIFIER', 'NUMBER', 'STRING', '@', 'TERMINATOR', 'OUTDENT'])
|
107
|
-
action = (token, i) -> @tokens.splice i, 0, ['}', '}', token[2]]
|
108
|
-
@scanTokens (token, i, tokens) ->
|
109
|
-
if (tag = token[0]) in EXPRESSION_START
|
110
|
-
stack.push [(if tag is 'INDENT' and @tag(i - 1) is '{' then '{' else tag), i]
|
111
|
-
return 1
|
112
|
-
if tag in EXPRESSION_END
|
113
|
-
start = stack.pop()
|
114
|
-
return 1
|
115
|
-
return 1 unless tag is ':' and
|
116
|
-
((ago = @tag i - 2) is ':' or stack[stack.length - 1]?[0] isnt '{')
|
117
|
-
stack.push ['{']
|
118
|
-
idx = if ago is '@' then i - 2 else i - 1
|
119
|
-
idx -= 2 while @tag(idx - 2) is 'HERECOMMENT'
|
120
|
-
value = new String('{')
|
121
|
-
value.generated = yes
|
122
|
-
tok = ['{', value, token[2]]
|
123
|
-
tok.generated = yes
|
124
|
-
tokens.splice idx, 0, tok
|
125
|
-
@detectEnd i + 2, condition, action
|
126
|
-
2
|
127
|
-
|
128
|
-
# Methods may be optionally called without parentheses, for simple cases.
|
129
|
-
# Insert the implicit parentheses here, so that the parser doesn't have to
|
130
|
-
# deal with them.
|
131
|
-
addImplicitParentheses: ->
|
132
|
-
noCall = no
|
133
|
-
action = (token, i) ->
|
134
|
-
idx = if token[0] is 'OUTDENT' then i + 1 else i
|
135
|
-
@tokens.splice idx, 0, ['CALL_END', ')', token[2]]
|
136
|
-
@scanTokens (token, i, tokens) ->
|
137
|
-
tag = token[0]
|
138
|
-
noCall = yes if tag in ['CLASS', 'IF']
|
139
|
-
[prev, current, next] = tokens[i - 1 .. i + 1]
|
140
|
-
callObject = not noCall and tag is 'INDENT' and
|
141
|
-
next and next.generated and next[0] is '{' and
|
142
|
-
prev and prev[0] in IMPLICIT_FUNC
|
143
|
-
seenSingle = no
|
144
|
-
noCall = no if tag in LINEBREAKS
|
145
|
-
token.call = yes if prev and not prev.spaced and tag is '?'
|
146
|
-
return 1 unless callObject or
|
147
|
-
prev?.spaced and (prev.call or prev[0] in IMPLICIT_FUNC) and
|
148
|
-
(tag in IMPLICIT_CALL or not (token.spaced or token.newLine) and tag in IMPLICIT_UNSPACED_CALL)
|
149
|
-
tokens.splice i, 0, ['CALL_START', '(', token[2]]
|
150
|
-
@detectEnd i + 1, (token, i) ->
|
151
|
-
[tag] = token
|
152
|
-
return yes if not seenSingle and token.fromThen
|
153
|
-
seenSingle = yes if tag in ['IF', 'ELSE', '->', '=>']
|
154
|
-
return yes if tag in ['.', '?.', '::'] and @tag(i - 1) is 'OUTDENT'
|
155
|
-
not token.generated and @tag(i - 1) isnt ',' and tag in IMPLICIT_END and
|
156
|
-
(tag isnt 'INDENT' or
|
157
|
-
(@tag(i - 2) isnt 'CLASS' and @tag(i - 1) not in IMPLICIT_BLOCK and
|
158
|
-
not ((post = @tokens[i + 1]) and post.generated and post[0] is '{')))
|
159
|
-
, action
|
160
|
-
prev[0] = 'FUNC_EXIST' if prev[0] is '?'
|
161
|
-
2
|
162
|
-
|
163
|
-
# Because our grammar is LALR(1), it can't handle some single-line
|
164
|
-
# expressions that lack ending delimiters. The **Rewriter** adds the implicit
|
165
|
-
# blocks, so it doesn't need to. ')' can close a single-line block,
|
166
|
-
# but we need to make sure it's balanced.
|
167
|
-
addImplicitIndentation: ->
|
168
|
-
@scanTokens (token, i, tokens) ->
|
169
|
-
[tag] = token
|
170
|
-
if tag is 'TERMINATOR' and @tag(i + 1) is 'THEN'
|
171
|
-
tokens.splice i, 1
|
172
|
-
return 0
|
173
|
-
if tag is 'ELSE' and @tag(i - 1) isnt 'OUTDENT'
|
174
|
-
tokens.splice i, 0, @indentation(token)...
|
175
|
-
return 2
|
176
|
-
if tag is 'CATCH' and @tag(i + 2) in ['OUTDENT', 'TERMINATOR', 'FINALLY']
|
177
|
-
tokens.splice i + 2, 0, @indentation(token)...
|
178
|
-
return 4
|
179
|
-
if tag in SINGLE_LINERS and @tag(i + 1) isnt 'INDENT' and
|
180
|
-
not (tag is 'ELSE' and @tag(i + 1) is 'IF')
|
181
|
-
starter = tag
|
182
|
-
[indent, outdent] = @indentation token
|
183
|
-
indent.fromThen = true if starter is 'THEN'
|
184
|
-
indent.generated = outdent.generated = true
|
185
|
-
tokens.splice i + 1, 0, indent
|
186
|
-
condition = (token, i) ->
|
187
|
-
token[1] isnt ';' and token[0] in SINGLE_CLOSERS and
|
188
|
-
not (token[0] is 'ELSE' and starter not in ['IF', 'THEN'])
|
189
|
-
action = (token, i) ->
|
190
|
-
@tokens.splice (if @tag(i - 1) is ',' then i - 1 else i), 0, outdent
|
191
|
-
@detectEnd i + 2, condition, action
|
192
|
-
tokens.splice i, 1 if tag is 'THEN'
|
193
|
-
return 1
|
194
|
-
return 1
|
195
|
-
|
196
|
-
# Tag postfix conditionals as such, so that we can parse them with a
|
197
|
-
# different precedence.
|
198
|
-
tagPostfixConditionals: ->
|
199
|
-
condition = (token, i) -> token[0] in ['TERMINATOR', 'INDENT']
|
200
|
-
@scanTokens (token, i) ->
|
201
|
-
return 1 unless token[0] is 'IF'
|
202
|
-
original = token
|
203
|
-
@detectEnd i + 1, condition, (token, i) ->
|
204
|
-
original[0] = 'POST_' + original[0] if token[0] isnt 'INDENT'
|
205
|
-
1
|
206
|
-
|
207
|
-
# Ensure that all listed pairs of tokens are correctly balanced throughout
|
208
|
-
# the course of the token stream.
|
209
|
-
ensureBalance: (pairs) ->
|
210
|
-
levels = {}
|
211
|
-
openLine = {}
|
212
|
-
for token in @tokens
|
213
|
-
[tag] = token
|
214
|
-
for [open, close] in pairs
|
215
|
-
levels[open] |= 0
|
216
|
-
if tag is open
|
217
|
-
openLine[open] = token[2] if levels[open]++ is 0
|
218
|
-
else if tag is close and --levels[open] < 0
|
219
|
-
throw Error "too many #{token[1]} on line #{token[2] + 1}"
|
220
|
-
for open, level of levels when level > 0
|
221
|
-
throw Error "unclosed #{ open } on line #{openLine[open] + 1}"
|
222
|
-
this
|
223
|
-
|
224
|
-
# We'd like to support syntax like this:
|
225
|
-
#
|
226
|
-
# el.click((event) ->
|
227
|
-
# el.hide())
|
228
|
-
#
|
229
|
-
# In order to accomplish this, move outdents that follow closing parens
|
230
|
-
# inwards, safely. The steps to accomplish this are:
|
231
|
-
#
|
232
|
-
# 1. Check that all paired tokens are balanced and in order.
|
233
|
-
# 2. Rewrite the stream with a stack: if you see an `EXPRESSION_START`, add it
|
234
|
-
# to the stack. If you see an `EXPRESSION_END`, pop the stack and replace
|
235
|
-
# it with the inverse of what we've just popped.
|
236
|
-
# 3. Keep track of "debt" for tokens that we manufacture, to make sure we end
|
237
|
-
# up balanced in the end.
|
238
|
-
# 4. Be careful not to alter array or parentheses delimiters with overzealous
|
239
|
-
# rewriting.
|
240
|
-
rewriteClosingParens: ->
|
241
|
-
stack = []
|
242
|
-
debt = {}
|
243
|
-
debt[key] = 0 for key of INVERSES
|
244
|
-
@scanTokens (token, i, tokens) ->
|
245
|
-
if (tag = token[0]) in EXPRESSION_START
|
246
|
-
stack.push token
|
247
|
-
return 1
|
248
|
-
return 1 unless tag in EXPRESSION_END
|
249
|
-
if debt[inv = INVERSES[tag]] > 0
|
250
|
-
debt[inv] -= 1
|
251
|
-
tokens.splice i, 1
|
252
|
-
return 0
|
253
|
-
match = stack.pop()
|
254
|
-
mtag = match[0]
|
255
|
-
oppos = INVERSES[mtag]
|
256
|
-
return 1 if tag is oppos
|
257
|
-
debt[mtag] += 1
|
258
|
-
val = [oppos, if mtag is 'INDENT' then match[1] else oppos]
|
259
|
-
if @tag(i + 2) is mtag
|
260
|
-
tokens.splice i + 3, 0, val
|
261
|
-
stack.push match
|
262
|
-
else
|
263
|
-
tokens.splice i, 0, val
|
264
|
-
1
|
265
|
-
|
266
|
-
# Generate the indentation tokens, based on another token on the same line.
|
267
|
-
indentation: (token) ->
|
268
|
-
[['INDENT', 2, token[2]], ['OUTDENT', 2, token[2]]]
|
269
|
-
|
270
|
-
# Look up a tag by token index.
|
271
|
-
tag: (i) -> @tokens[i]?[0]
|
272
|
-
|
273
|
-
# Constants
|
274
|
-
# ---------
|
275
|
-
|
276
|
-
# List of the token pairs that must be balanced.
|
277
|
-
BALANCED_PAIRS = [
|
278
|
-
['(', ')']
|
279
|
-
['[', ']']
|
280
|
-
['{', '}']
|
281
|
-
['INDENT', 'OUTDENT'],
|
282
|
-
['CALL_START', 'CALL_END']
|
283
|
-
['PARAM_START', 'PARAM_END']
|
284
|
-
['INDEX_START', 'INDEX_END']
|
285
|
-
]
|
286
|
-
|
287
|
-
# The inverse mappings of `BALANCED_PAIRS` we're trying to fix up, so we can
|
288
|
-
# look things up from either end.
|
289
|
-
INVERSES = {}
|
290
|
-
|
291
|
-
# The tokens that signal the start/end of a balanced pair.
|
292
|
-
EXPRESSION_START = []
|
293
|
-
EXPRESSION_END = []
|
294
|
-
|
295
|
-
for [left, rite] in BALANCED_PAIRS
|
296
|
-
EXPRESSION_START.push INVERSES[rite] = left
|
297
|
-
EXPRESSION_END .push INVERSES[left] = rite
|
298
|
-
|
299
|
-
# Tokens that indicate the close of a clause of an expression.
|
300
|
-
EXPRESSION_CLOSE = ['CATCH', 'WHEN', 'ELSE', 'FINALLY'].concat EXPRESSION_END
|
301
|
-
|
302
|
-
# Tokens that, if followed by an `IMPLICIT_CALL`, indicate a function invocation.
|
303
|
-
IMPLICIT_FUNC = ['IDENTIFIER', 'SUPER', ')', 'CALL_END', ']', 'INDEX_END', '@', 'THIS']
|
304
|
-
|
305
|
-
# If preceded by an `IMPLICIT_FUNC`, indicates a function invocation.
|
306
|
-
IMPLICIT_CALL = [
|
307
|
-
'IDENTIFIER', 'NUMBER', 'STRING', 'JS', 'REGEX', 'NEW', 'PARAM_START', 'CLASS'
|
308
|
-
'IF', 'TRY', 'SWITCH', 'THIS', 'BOOL', 'UNARY', 'SUPER'
|
309
|
-
'@', '->', '=>', '[', '(', '{', '--', '++'
|
310
|
-
]
|
311
|
-
|
312
|
-
IMPLICIT_UNSPACED_CALL = ['+', '-']
|
313
|
-
|
314
|
-
# Tokens indicating that the implicit call must enclose a block of expressions.
|
315
|
-
IMPLICIT_BLOCK = ['->', '=>', '{', '[', ',']
|
316
|
-
|
317
|
-
# Tokens that always mark the end of an implicit call for single-liners.
|
318
|
-
IMPLICIT_END = ['POST_IF', 'FOR', 'WHILE', 'UNTIL', 'WHEN', 'BY', 'LOOP', 'TERMINATOR', 'INDENT']
|
319
|
-
|
320
|
-
# Single-line flavors of block expressions that have unclosed endings.
|
321
|
-
# The grammar can't disambiguate them, so we insert the implicit indentation.
|
322
|
-
SINGLE_LINERS = ['ELSE', '->', '=>', 'TRY', 'FINALLY', 'THEN']
|
323
|
-
SINGLE_CLOSERS = ['TERMINATOR', 'CATCH', 'FINALLY', 'ELSE', 'OUTDENT', 'LEADING_WHEN']
|
324
|
-
|
325
|
-
# Tokens that end a line.
|
326
|
-
LINEBREAKS = ['TERMINATOR', 'INDENT', 'OUTDENT']
|