spade-packager 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- data/.gitignore +2 -0
- data/.gitmodules +6 -0
- data/bin/spadepkg +8 -0
- data/lib/libgems_ext.rb +8 -0
- data/lib/libgems_ext/config_file.rb +33 -0
- data/lib/libgems_ext/dependency_installer.rb +150 -0
- data/lib/libgems_ext/installer.rb +39 -0
- data/lib/libgems_ext/libgems.rb +39 -0
- data/lib/libgems_ext/spec_fetcher.rb +11 -0
- data/lib/spade-packager.rb +1 -0
- data/lib/spade/packager.rb +18 -0
- data/lib/spade/packager/cli.rb +9 -0
- data/lib/spade/packager/cli/base.rb +196 -0
- data/lib/spade/packager/cli/owner.rb +46 -0
- data/lib/spade/packager/cli/project_generator.rb +117 -0
- data/lib/spade/packager/credentials.rb +38 -0
- data/lib/spade/packager/local.rb +50 -0
- data/lib/spade/packager/package.rb +160 -0
- data/lib/spade/packager/remote.rb +98 -0
- data/lib/spade/packager/repository.rb +18 -0
- data/lib/spade/packager/version.rb +5 -0
- data/packages/coffee-script/.gitignore +8 -0
- data/packages/coffee-script/.npmignore +11 -0
- data/packages/coffee-script/Cakefile +229 -0
- data/packages/coffee-script/LICENSE +22 -0
- data/packages/coffee-script/README +47 -0
- data/packages/coffee-script/Rakefile +78 -0
- data/packages/coffee-script/bin/cake +7 -0
- data/packages/coffee-script/bin/coffee +7 -0
- data/packages/coffee-script/documentation/coffee/aliases.coffee +11 -0
- data/packages/coffee-script/documentation/coffee/array_comprehensions.coffee +2 -0
- data/packages/coffee-script/documentation/coffee/block_comment.coffee +6 -0
- data/packages/coffee-script/documentation/coffee/cake_tasks.coffee +9 -0
- data/packages/coffee-script/documentation/coffee/classes.coffee +25 -0
- data/packages/coffee-script/documentation/coffee/comparisons.coffee +5 -0
- data/packages/coffee-script/documentation/coffee/conditionals.coffee +13 -0
- data/packages/coffee-script/documentation/coffee/default_args.coffee +8 -0
- data/packages/coffee-script/documentation/coffee/do.coffee +4 -0
- data/packages/coffee-script/documentation/coffee/embedded.coffee +5 -0
- data/packages/coffee-script/documentation/coffee/existence.coffee +10 -0
- data/packages/coffee-script/documentation/coffee/expressions.coffee +9 -0
- data/packages/coffee-script/documentation/coffee/expressions_assignment.coffee +3 -0
- data/packages/coffee-script/documentation/coffee/expressions_comprehension.coffee +3 -0
- data/packages/coffee-script/documentation/coffee/expressions_try.coffee +7 -0
- data/packages/coffee-script/documentation/coffee/fat_arrow.coffee +6 -0
- data/packages/coffee-script/documentation/coffee/functions.coffee +2 -0
- data/packages/coffee-script/documentation/coffee/heredocs.coffee +7 -0
- data/packages/coffee-script/documentation/coffee/heregexes.coffee +11 -0
- data/packages/coffee-script/documentation/coffee/interpolation.coffee +6 -0
- data/packages/coffee-script/documentation/coffee/multiple_return_values.coffee +7 -0
- data/packages/coffee-script/documentation/coffee/object_comprehensions.coffee +4 -0
- data/packages/coffee-script/documentation/coffee/object_extraction.coffee +13 -0
- data/packages/coffee-script/documentation/coffee/objects_and_arrays.coffee +19 -0
- data/packages/coffee-script/documentation/coffee/objects_reserved.coffee +5 -0
- data/packages/coffee-script/documentation/coffee/overview.coffee +28 -0
- data/packages/coffee-script/documentation/coffee/parallel_assignment.coffee +6 -0
- data/packages/coffee-script/documentation/coffee/patterns_and_splats.coffee +7 -0
- data/packages/coffee-script/documentation/coffee/prototypes.coffee +3 -0
- data/packages/coffee-script/documentation/coffee/range_comprehensions.coffee +2 -0
- data/packages/coffee-script/documentation/coffee/scope.coffee +5 -0
- data/packages/coffee-script/documentation/coffee/slices.coffee +7 -0
- data/packages/coffee-script/documentation/coffee/soaks.coffee +1 -0
- data/packages/coffee-script/documentation/coffee/splats.coffee +27 -0
- data/packages/coffee-script/documentation/coffee/splices.coffee +5 -0
- data/packages/coffee-script/documentation/coffee/strings.coffee +8 -0
- data/packages/coffee-script/documentation/coffee/switch.coffee +10 -0
- data/packages/coffee-script/documentation/coffee/try.coffee +8 -0
- data/packages/coffee-script/documentation/coffee/while.coffee +10 -0
- data/packages/coffee-script/documentation/css/docs.css +374 -0
- data/packages/coffee-script/documentation/css/idle.css +64 -0
- data/packages/coffee-script/documentation/docs/browser.html +25 -0
- data/packages/coffee-script/documentation/docs/cake.html +43 -0
- data/packages/coffee-script/documentation/docs/coffee-script.html +51 -0
- data/packages/coffee-script/documentation/docs/command.html +161 -0
- data/packages/coffee-script/documentation/docs/docco.css +186 -0
- data/packages/coffee-script/documentation/docs/grammar.html +399 -0
- data/packages/coffee-script/documentation/docs/helpers.html +31 -0
- data/packages/coffee-script/documentation/docs/index.html +3 -0
- data/packages/coffee-script/documentation/docs/lexer.html +490 -0
- data/packages/coffee-script/documentation/docs/nodes.html +1338 -0
- data/packages/coffee-script/documentation/docs/optparse.html +78 -0
- data/packages/coffee-script/documentation/docs/repl.html +24 -0
- data/packages/coffee-script/documentation/docs/rewriter.html +251 -0
- data/packages/coffee-script/documentation/docs/scope.html +54 -0
- data/packages/coffee-script/documentation/docs/underscore.html +295 -0
- 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 +1607 -0
- data/packages/coffee-script/documentation/js/aliases.js +17 -0
- data/packages/coffee-script/documentation/js/array_comprehensions.js +6 -0
- data/packages/coffee-script/documentation/js/block_comment.js +4 -0
- data/packages/coffee-script/documentation/js/cake_tasks.js +10 -0
- data/packages/coffee-script/documentation/js/classes.js +44 -0
- data/packages/coffee-script/documentation/js/comparisons.js +3 -0
- data/packages/coffee-script/documentation/js/conditionals.js +12 -0
- data/packages/coffee-script/documentation/js/default_args.js +7 -0
- data/packages/coffee-script/documentation/js/do.js +10 -0
- data/packages/coffee-script/documentation/js/embedded.js +4 -0
- data/packages/coffee-script/documentation/js/existence.js +6 -0
- data/packages/coffee-script/documentation/js/expressions.js +15 -0
- data/packages/coffee-script/documentation/js/expressions_assignment.js +2 -0
- data/packages/coffee-script/documentation/js/expressions_comprehension.js +9 -0
- data/packages/coffee-script/documentation/js/expressions_try.js +7 -0
- data/packages/coffee-script/documentation/js/fat_arrow.js +9 -0
- data/packages/coffee-script/documentation/js/functions.js +7 -0
- data/packages/coffee-script/documentation/js/heredocs.js +2 -0
- data/packages/coffee-script/documentation/js/heregexes.js +2 -0
- data/packages/coffee-script/documentation/js/interpolation.js +4 -0
- data/packages/coffee-script/documentation/js/multiple_return_values.js +5 -0
- data/packages/coffee-script/documentation/js/object_comprehensions.js +15 -0
- data/packages/coffee-script/documentation/js/object_extraction.js +10 -0
- data/packages/coffee-script/documentation/js/objects_and_arrays.js +17 -0
- data/packages/coffee-script/documentation/js/objects_reserved.js +4 -0
- data/packages/coffee-script/documentation/js/overview.js +35 -0
- data/packages/coffee-script/documentation/js/parallel_assignment.js +4 -0
- data/packages/coffee-script/documentation/js/patterns_and_splats.js +4 -0
- data/packages/coffee-script/documentation/js/prototypes.js +3 -0
- data/packages/coffee-script/documentation/js/range_comprehensions.js +9 -0
- data/packages/coffee-script/documentation/js/scope.js +8 -0
- data/packages/coffee-script/documentation/js/slices.js +4 -0
- data/packages/coffee-script/documentation/js/soaks.js +2 -0
- data/packages/coffee-script/documentation/js/splats.js +15 -0
- data/packages/coffee-script/documentation/js/splices.js +3 -0
- data/packages/coffee-script/documentation/js/strings.js +2 -0
- data/packages/coffee-script/documentation/js/switch.js +23 -0
- data/packages/coffee-script/documentation/js/try.js +8 -0
- data/packages/coffee-script/documentation/js/while.js +18 -0
- data/packages/coffee-script/documentation/vendor/jquery-1.4.2.js +6240 -0
- data/packages/coffee-script/examples/beautiful_code/binary_search.coffee +16 -0
- data/packages/coffee-script/examples/beautiful_code/quicksort_runtime.coffee +13 -0
- data/packages/coffee-script/examples/beautiful_code/regular_expression_matcher.coffee +34 -0
- data/packages/coffee-script/examples/blocks.coffee +54 -0
- data/packages/coffee-script/examples/code.coffee +167 -0
- data/packages/coffee-script/examples/computer_science/README +4 -0
- data/packages/coffee-script/examples/computer_science/binary_search.coffee +25 -0
- data/packages/coffee-script/examples/computer_science/bubble_sort.coffee +11 -0
- data/packages/coffee-script/examples/computer_science/linked_list.coffee +108 -0
- data/packages/coffee-script/examples/computer_science/luhn_algorithm.coffee +36 -0
- data/packages/coffee-script/examples/computer_science/merge_sort.coffee +19 -0
- data/packages/coffee-script/examples/computer_science/selection_sort.coffee +23 -0
- data/packages/coffee-script/examples/poignant.coffee +181 -0
- data/packages/coffee-script/examples/potion.coffee +206 -0
- data/packages/coffee-script/examples/underscore.coffee +682 -0
- data/packages/coffee-script/examples/web_server.coffee +12 -0
- data/packages/coffee-script/extras/EXTRAS +7 -0
- data/packages/coffee-script/extras/coffee-script.js +8 -0
- data/packages/coffee-script/extras/jsl.conf +44 -0
- data/packages/coffee-script/index.html +2515 -0
- data/packages/coffee-script/lib/browser.js +52 -0
- data/packages/coffee-script/lib/cake.js +76 -0
- data/packages/coffee-script/lib/coffee-script.js +82 -0
- data/packages/coffee-script/lib/command.js +263 -0
- data/packages/coffee-script/lib/grammar.js +581 -0
- data/packages/coffee-script/lib/helpers.js +66 -0
- data/packages/coffee-script/lib/index.js +8 -0
- data/packages/coffee-script/lib/lexer.js +633 -0
- data/packages/coffee-script/lib/nodes.js +2165 -0
- data/packages/coffee-script/lib/optparse.js +111 -0
- data/packages/coffee-script/lib/parser.js +649 -0
- data/packages/coffee-script/lib/repl.js +42 -0
- data/packages/coffee-script/lib/rewriter.js +353 -0
- data/packages/coffee-script/lib/scope.js +120 -0
- data/packages/coffee-script/lib/spade-format.js +45 -0
- data/packages/coffee-script/package.json +26 -0
- data/packages/coffee-script/src/browser.coffee +43 -0
- data/packages/coffee-script/src/cake.coffee +69 -0
- data/packages/coffee-script/src/coffee-script.coffee +92 -0
- data/packages/coffee-script/src/command.coffee +214 -0
- data/packages/coffee-script/src/grammar.coffee +590 -0
- data/packages/coffee-script/src/helpers.coffee +56 -0
- data/packages/coffee-script/src/index.coffee +2 -0
- data/packages/coffee-script/src/lexer.coffee +653 -0
- data/packages/coffee-script/src/nodes.coffee +1754 -0
- data/packages/coffee-script/src/optparse.coffee +99 -0
- data/packages/coffee-script/src/repl.coffee +42 -0
- data/packages/coffee-script/src/rewriter.coffee +326 -0
- data/packages/coffee-script/src/scope.coffee +94 -0
- data/packages/coffee-script/test/arguments.coffee +127 -0
- data/packages/coffee-script/test/assignment.coffee +98 -0
- data/packages/coffee-script/test/break.coffee +18 -0
- data/packages/coffee-script/test/comments.coffee +201 -0
- data/packages/coffee-script/test/conditionals.coffee +181 -0
- data/packages/coffee-script/test/exception_handling.coffee +90 -0
- data/packages/coffee-script/test/helpers.coffee +96 -0
- data/packages/coffee-script/test/importing.coffee +18 -0
- data/packages/coffee-script/test/operators.coffee +225 -0
- data/packages/coffee-script/test/ranges_slices_and_splices.coffee +186 -0
- data/packages/coffee-script/test/regular_expressions.coffee +56 -0
- data/packages/coffee-script/test/test.html +123 -0
- data/packages/coffee-script/test/test_chaining.coffee +77 -0
- data/packages/coffee-script/test/test_classes.coffee +372 -0
- data/packages/coffee-script/test/test_compilation.coffee +26 -0
- data/packages/coffee-script/test/test_comprehensions.coffee +318 -0
- data/packages/coffee-script/test/test_existence.coffee +165 -0
- data/packages/coffee-script/test/test_functions.coffee +379 -0
- data/packages/coffee-script/test/test_heredocs.coffee +111 -0
- data/packages/coffee-script/test/test_literals.coffee +270 -0
- data/packages/coffee-script/test/test_option_parser.coffee +27 -0
- data/packages/coffee-script/test/test_pattern_matching.coffee +162 -0
- data/packages/coffee-script/test/test_returns.coffee +63 -0
- data/packages/coffee-script/test/test_splats.coffee +102 -0
- data/packages/coffee-script/test/test_strings.coffee +118 -0
- data/packages/coffee-script/test/test_switch.coffee +103 -0
- data/packages/coffee-script/test/test_while.coffee +71 -0
- data/packages/ivory/LICENSE.txt +1 -0
- data/packages/ivory/README.md +19 -0
- data/packages/ivory/lib/buffer.js +111 -0
- data/packages/ivory/lib/events.js +137 -0
- data/packages/ivory/lib/fs.js +266 -0
- data/packages/ivory/lib/main.js +13 -0
- data/packages/ivory/lib/path.js +158 -0
- data/packages/ivory/lib/ruby/buffer.rb +145 -0
- data/packages/ivory/lib/ruby/constants.rb +585 -0
- data/packages/ivory/lib/ruby/events.rb +32 -0
- data/packages/ivory/lib/ruby/fs.rb +245 -0
- data/packages/ivory/lib/ruby/process.rb +28 -0
- data/packages/ivory/lib/stream.js +115 -0
- data/packages/ivory/lib/util.js +414 -0
- data/packages/ivory/package.json +11 -0
- data/packages/ivory/spade-boot.js +78 -0
- data/packages/jquery/main.js +7179 -0
- data/packages/jquery/package.json +10 -0
- data/packages/json/lib/main.js +14 -0
- data/packages/json/package.json +8 -0
- data/packages/lproj/README.md +77 -0
- data/packages/lproj/examples/demo-app/en.lproj/localized.strings +2 -0
- data/packages/lproj/examples/demo-app/fr.lproj/localized.strings +3 -0
- data/packages/lproj/examples/demo-app/index.html +8 -0
- data/packages/lproj/examples/demo-app/lib/main.js +7 -0
- data/packages/lproj/examples/demo-app/package.json +9 -0
- data/packages/lproj/lib/main.js +78 -0
- data/packages/lproj/lib/strings-format.js +6 -0
- data/packages/lproj/package.json +9 -0
- data/packages/optparse/README.md +161 -0
- data/packages/optparse/TODO +1 -0
- data/packages/optparse/examples/browser-test.html +75 -0
- data/packages/optparse/examples/nodejs-test.js +90 -0
- data/packages/optparse/lib/optparse.js +309 -0
- data/packages/optparse/package.json +13 -0
- data/packages/optparse/seed.yml +5 -0
- data/packages/text/lib/main.js +8 -0
- data/packages/text/package.json +9 -0
- data/packages/web-file/README.md +7 -0
- data/packages/web-file/lib/errors.js +32 -0
- data/packages/web-file/lib/file-reader.js +10 -0
- data/packages/web-file/lib/file-system.js +234 -0
- data/packages/web-file/lib/file-writer.js +10 -0
- data/packages/web-file/lib/file.js +9 -0
- data/packages/web-file/lib/main.js +34 -0
- data/packages/web-file/lib/platform.js +25 -0
- data/packages/web-file/lib/ruby/file.rb +252 -0
- data/packages/web-file/lib/ruby/file_reader.rb +69 -0
- data/packages/web-file/lib/ruby/file_system.rb +134 -0
- data/packages/web-file/lib/ruby/file_writer.rb +78 -0
- data/packages/web-file/package.json +12 -0
- data/packages/web-typed-array/README.md +7 -0
- data/packages/web-typed-array/lib/array-buffer-view.js +9 -0
- data/packages/web-typed-array/lib/array-buffer.js +7 -0
- data/packages/web-typed-array/lib/main.js +33 -0
- data/packages/web-typed-array/lib/platform.js +20 -0
- data/packages/web-typed-array/lib/ruby/array_buffer.rb +31 -0
- data/packages/web-typed-array/lib/ruby/array_buffer_view.rb +130 -0
- data/packages/web-typed-array/lib/ruby/typed_array.rb +133 -0
- data/packages/web-typed-array/lib/typed-array.js +26 -0
- data/packages/web-typed-array/package.json +9 -0
- data/spade-packager.gemspec +39 -0
- data/spec/cli/build_spec.rb +57 -0
- data/spec/cli/install_spec.rb +119 -0
- data/spec/cli/installed_spec.rb +55 -0
- data/spec/cli/list_spec.rb +74 -0
- data/spec/cli/login_spec.rb +75 -0
- data/spec/cli/new_spec.rb +5 -0
- data/spec/cli/owner_spec.rb +114 -0
- data/spec/cli/push_spec.rb +73 -0
- data/spec/cli/uninstall_spec.rb +58 -0
- data/spec/cli/unpack_spec.rb +72 -0
- data/spec/cli/unyank_spec.rb +73 -0
- data/spec/cli/yank_spec.rb +73 -0
- data/spec/credentials_spec.rb +23 -0
- data/spec/fixtures/badrake-0.8.7.spd +0 -0
- data/spec/fixtures/builder-3.0.0.spd +0 -0
- data/spec/fixtures/bundler-1.1.pre.spd +0 -0
- 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 +3 -0
- data/spec/fixtures/core-test/lib/main.js +1 -0
- data/spec/fixtures/core-test/resources/runner.css +0 -0
- data/spec/fixtures/core-test/tests/test.js +1 -0
- data/spec/fixtures/highline-1.6.1.spd +0 -0
- 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 +30 -0
- data/spec/fixtures/rake-0.8.6.spd +0 -0
- data/spec/fixtures/rake-0.8.7.spd +0 -0
- data/spec/gauntlet_spec.rb +27 -0
- data/spec/package_spec.rb +267 -0
- data/spec/spec_helper.rb +32 -0
- data/spec/support/cli.rb +103 -0
- data/spec/support/fake.rb +48 -0
- data/spec/support/fake_gem_server.rb +67 -0
- data/spec/support/fake_gemcutter.rb +50 -0
- data/spec/support/matchers.rb +32 -0
- data/spec/support/path.rb +61 -0
- data/templates/project/LICENSE +19 -0
- data/templates/project/README.md +21 -0
- data/templates/project/lib/main.js +9 -0
- data/templates/project/project.json +31 -0
- data/templates/project/tests/main-test.js +8 -0
- metadata +484 -0
@@ -0,0 +1,99 @@
|
|
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
|
@@ -0,0 +1,42 @@
|
|
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()
|
@@ -0,0 +1,326 @@
|
|
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']
|