spade-packager 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (316) hide show
  1. data/.gitignore +2 -0
  2. data/.gitmodules +6 -0
  3. data/bin/spadepkg +8 -0
  4. data/lib/libgems_ext.rb +8 -0
  5. data/lib/libgems_ext/config_file.rb +33 -0
  6. data/lib/libgems_ext/dependency_installer.rb +150 -0
  7. data/lib/libgems_ext/installer.rb +39 -0
  8. data/lib/libgems_ext/libgems.rb +39 -0
  9. data/lib/libgems_ext/spec_fetcher.rb +11 -0
  10. data/lib/spade-packager.rb +1 -0
  11. data/lib/spade/packager.rb +18 -0
  12. data/lib/spade/packager/cli.rb +9 -0
  13. data/lib/spade/packager/cli/base.rb +196 -0
  14. data/lib/spade/packager/cli/owner.rb +46 -0
  15. data/lib/spade/packager/cli/project_generator.rb +117 -0
  16. data/lib/spade/packager/credentials.rb +38 -0
  17. data/lib/spade/packager/local.rb +50 -0
  18. data/lib/spade/packager/package.rb +160 -0
  19. data/lib/spade/packager/remote.rb +98 -0
  20. data/lib/spade/packager/repository.rb +18 -0
  21. data/lib/spade/packager/version.rb +5 -0
  22. data/packages/coffee-script/.gitignore +8 -0
  23. data/packages/coffee-script/.npmignore +11 -0
  24. data/packages/coffee-script/Cakefile +229 -0
  25. data/packages/coffee-script/LICENSE +22 -0
  26. data/packages/coffee-script/README +47 -0
  27. data/packages/coffee-script/Rakefile +78 -0
  28. data/packages/coffee-script/bin/cake +7 -0
  29. data/packages/coffee-script/bin/coffee +7 -0
  30. data/packages/coffee-script/documentation/coffee/aliases.coffee +11 -0
  31. data/packages/coffee-script/documentation/coffee/array_comprehensions.coffee +2 -0
  32. data/packages/coffee-script/documentation/coffee/block_comment.coffee +6 -0
  33. data/packages/coffee-script/documentation/coffee/cake_tasks.coffee +9 -0
  34. data/packages/coffee-script/documentation/coffee/classes.coffee +25 -0
  35. data/packages/coffee-script/documentation/coffee/comparisons.coffee +5 -0
  36. data/packages/coffee-script/documentation/coffee/conditionals.coffee +13 -0
  37. data/packages/coffee-script/documentation/coffee/default_args.coffee +8 -0
  38. data/packages/coffee-script/documentation/coffee/do.coffee +4 -0
  39. data/packages/coffee-script/documentation/coffee/embedded.coffee +5 -0
  40. data/packages/coffee-script/documentation/coffee/existence.coffee +10 -0
  41. data/packages/coffee-script/documentation/coffee/expressions.coffee +9 -0
  42. data/packages/coffee-script/documentation/coffee/expressions_assignment.coffee +3 -0
  43. data/packages/coffee-script/documentation/coffee/expressions_comprehension.coffee +3 -0
  44. data/packages/coffee-script/documentation/coffee/expressions_try.coffee +7 -0
  45. data/packages/coffee-script/documentation/coffee/fat_arrow.coffee +6 -0
  46. data/packages/coffee-script/documentation/coffee/functions.coffee +2 -0
  47. data/packages/coffee-script/documentation/coffee/heredocs.coffee +7 -0
  48. data/packages/coffee-script/documentation/coffee/heregexes.coffee +11 -0
  49. data/packages/coffee-script/documentation/coffee/interpolation.coffee +6 -0
  50. data/packages/coffee-script/documentation/coffee/multiple_return_values.coffee +7 -0
  51. data/packages/coffee-script/documentation/coffee/object_comprehensions.coffee +4 -0
  52. data/packages/coffee-script/documentation/coffee/object_extraction.coffee +13 -0
  53. data/packages/coffee-script/documentation/coffee/objects_and_arrays.coffee +19 -0
  54. data/packages/coffee-script/documentation/coffee/objects_reserved.coffee +5 -0
  55. data/packages/coffee-script/documentation/coffee/overview.coffee +28 -0
  56. data/packages/coffee-script/documentation/coffee/parallel_assignment.coffee +6 -0
  57. data/packages/coffee-script/documentation/coffee/patterns_and_splats.coffee +7 -0
  58. data/packages/coffee-script/documentation/coffee/prototypes.coffee +3 -0
  59. data/packages/coffee-script/documentation/coffee/range_comprehensions.coffee +2 -0
  60. data/packages/coffee-script/documentation/coffee/scope.coffee +5 -0
  61. data/packages/coffee-script/documentation/coffee/slices.coffee +7 -0
  62. data/packages/coffee-script/documentation/coffee/soaks.coffee +1 -0
  63. data/packages/coffee-script/documentation/coffee/splats.coffee +27 -0
  64. data/packages/coffee-script/documentation/coffee/splices.coffee +5 -0
  65. data/packages/coffee-script/documentation/coffee/strings.coffee +8 -0
  66. data/packages/coffee-script/documentation/coffee/switch.coffee +10 -0
  67. data/packages/coffee-script/documentation/coffee/try.coffee +8 -0
  68. data/packages/coffee-script/documentation/coffee/while.coffee +10 -0
  69. data/packages/coffee-script/documentation/css/docs.css +374 -0
  70. data/packages/coffee-script/documentation/css/idle.css +64 -0
  71. data/packages/coffee-script/documentation/docs/browser.html +25 -0
  72. data/packages/coffee-script/documentation/docs/cake.html +43 -0
  73. data/packages/coffee-script/documentation/docs/coffee-script.html +51 -0
  74. data/packages/coffee-script/documentation/docs/command.html +161 -0
  75. data/packages/coffee-script/documentation/docs/docco.css +186 -0
  76. data/packages/coffee-script/documentation/docs/grammar.html +399 -0
  77. data/packages/coffee-script/documentation/docs/helpers.html +31 -0
  78. data/packages/coffee-script/documentation/docs/index.html +3 -0
  79. data/packages/coffee-script/documentation/docs/lexer.html +490 -0
  80. data/packages/coffee-script/documentation/docs/nodes.html +1338 -0
  81. data/packages/coffee-script/documentation/docs/optparse.html +78 -0
  82. data/packages/coffee-script/documentation/docs/repl.html +24 -0
  83. data/packages/coffee-script/documentation/docs/rewriter.html +251 -0
  84. data/packages/coffee-script/documentation/docs/scope.html +54 -0
  85. data/packages/coffee-script/documentation/docs/underscore.html +295 -0
  86. data/packages/coffee-script/documentation/images/background.png +0 -0
  87. data/packages/coffee-script/documentation/images/banding.png +0 -0
  88. data/packages/coffee-script/documentation/images/button_bg.png +0 -0
  89. data/packages/coffee-script/documentation/images/button_bg_dark.gif +0 -0
  90. data/packages/coffee-script/documentation/images/button_bg_green.gif +0 -0
  91. data/packages/coffee-script/documentation/images/favicon.ico +0 -0
  92. data/packages/coffee-script/documentation/images/logo.png +0 -0
  93. data/packages/coffee-script/documentation/images/screenshadow.png +0 -0
  94. data/packages/coffee-script/documentation/index.html.erb +1607 -0
  95. data/packages/coffee-script/documentation/js/aliases.js +17 -0
  96. data/packages/coffee-script/documentation/js/array_comprehensions.js +6 -0
  97. data/packages/coffee-script/documentation/js/block_comment.js +4 -0
  98. data/packages/coffee-script/documentation/js/cake_tasks.js +10 -0
  99. data/packages/coffee-script/documentation/js/classes.js +44 -0
  100. data/packages/coffee-script/documentation/js/comparisons.js +3 -0
  101. data/packages/coffee-script/documentation/js/conditionals.js +12 -0
  102. data/packages/coffee-script/documentation/js/default_args.js +7 -0
  103. data/packages/coffee-script/documentation/js/do.js +10 -0
  104. data/packages/coffee-script/documentation/js/embedded.js +4 -0
  105. data/packages/coffee-script/documentation/js/existence.js +6 -0
  106. data/packages/coffee-script/documentation/js/expressions.js +15 -0
  107. data/packages/coffee-script/documentation/js/expressions_assignment.js +2 -0
  108. data/packages/coffee-script/documentation/js/expressions_comprehension.js +9 -0
  109. data/packages/coffee-script/documentation/js/expressions_try.js +7 -0
  110. data/packages/coffee-script/documentation/js/fat_arrow.js +9 -0
  111. data/packages/coffee-script/documentation/js/functions.js +7 -0
  112. data/packages/coffee-script/documentation/js/heredocs.js +2 -0
  113. data/packages/coffee-script/documentation/js/heregexes.js +2 -0
  114. data/packages/coffee-script/documentation/js/interpolation.js +4 -0
  115. data/packages/coffee-script/documentation/js/multiple_return_values.js +5 -0
  116. data/packages/coffee-script/documentation/js/object_comprehensions.js +15 -0
  117. data/packages/coffee-script/documentation/js/object_extraction.js +10 -0
  118. data/packages/coffee-script/documentation/js/objects_and_arrays.js +17 -0
  119. data/packages/coffee-script/documentation/js/objects_reserved.js +4 -0
  120. data/packages/coffee-script/documentation/js/overview.js +35 -0
  121. data/packages/coffee-script/documentation/js/parallel_assignment.js +4 -0
  122. data/packages/coffee-script/documentation/js/patterns_and_splats.js +4 -0
  123. data/packages/coffee-script/documentation/js/prototypes.js +3 -0
  124. data/packages/coffee-script/documentation/js/range_comprehensions.js +9 -0
  125. data/packages/coffee-script/documentation/js/scope.js +8 -0
  126. data/packages/coffee-script/documentation/js/slices.js +4 -0
  127. data/packages/coffee-script/documentation/js/soaks.js +2 -0
  128. data/packages/coffee-script/documentation/js/splats.js +15 -0
  129. data/packages/coffee-script/documentation/js/splices.js +3 -0
  130. data/packages/coffee-script/documentation/js/strings.js +2 -0
  131. data/packages/coffee-script/documentation/js/switch.js +23 -0
  132. data/packages/coffee-script/documentation/js/try.js +8 -0
  133. data/packages/coffee-script/documentation/js/while.js +18 -0
  134. data/packages/coffee-script/documentation/vendor/jquery-1.4.2.js +6240 -0
  135. data/packages/coffee-script/examples/beautiful_code/binary_search.coffee +16 -0
  136. data/packages/coffee-script/examples/beautiful_code/quicksort_runtime.coffee +13 -0
  137. data/packages/coffee-script/examples/beautiful_code/regular_expression_matcher.coffee +34 -0
  138. data/packages/coffee-script/examples/blocks.coffee +54 -0
  139. data/packages/coffee-script/examples/code.coffee +167 -0
  140. data/packages/coffee-script/examples/computer_science/README +4 -0
  141. data/packages/coffee-script/examples/computer_science/binary_search.coffee +25 -0
  142. data/packages/coffee-script/examples/computer_science/bubble_sort.coffee +11 -0
  143. data/packages/coffee-script/examples/computer_science/linked_list.coffee +108 -0
  144. data/packages/coffee-script/examples/computer_science/luhn_algorithm.coffee +36 -0
  145. data/packages/coffee-script/examples/computer_science/merge_sort.coffee +19 -0
  146. data/packages/coffee-script/examples/computer_science/selection_sort.coffee +23 -0
  147. data/packages/coffee-script/examples/poignant.coffee +181 -0
  148. data/packages/coffee-script/examples/potion.coffee +206 -0
  149. data/packages/coffee-script/examples/underscore.coffee +682 -0
  150. data/packages/coffee-script/examples/web_server.coffee +12 -0
  151. data/packages/coffee-script/extras/EXTRAS +7 -0
  152. data/packages/coffee-script/extras/coffee-script.js +8 -0
  153. data/packages/coffee-script/extras/jsl.conf +44 -0
  154. data/packages/coffee-script/index.html +2515 -0
  155. data/packages/coffee-script/lib/browser.js +52 -0
  156. data/packages/coffee-script/lib/cake.js +76 -0
  157. data/packages/coffee-script/lib/coffee-script.js +82 -0
  158. data/packages/coffee-script/lib/command.js +263 -0
  159. data/packages/coffee-script/lib/grammar.js +581 -0
  160. data/packages/coffee-script/lib/helpers.js +66 -0
  161. data/packages/coffee-script/lib/index.js +8 -0
  162. data/packages/coffee-script/lib/lexer.js +633 -0
  163. data/packages/coffee-script/lib/nodes.js +2165 -0
  164. data/packages/coffee-script/lib/optparse.js +111 -0
  165. data/packages/coffee-script/lib/parser.js +649 -0
  166. data/packages/coffee-script/lib/repl.js +42 -0
  167. data/packages/coffee-script/lib/rewriter.js +353 -0
  168. data/packages/coffee-script/lib/scope.js +120 -0
  169. data/packages/coffee-script/lib/spade-format.js +45 -0
  170. data/packages/coffee-script/package.json +26 -0
  171. data/packages/coffee-script/src/browser.coffee +43 -0
  172. data/packages/coffee-script/src/cake.coffee +69 -0
  173. data/packages/coffee-script/src/coffee-script.coffee +92 -0
  174. data/packages/coffee-script/src/command.coffee +214 -0
  175. data/packages/coffee-script/src/grammar.coffee +590 -0
  176. data/packages/coffee-script/src/helpers.coffee +56 -0
  177. data/packages/coffee-script/src/index.coffee +2 -0
  178. data/packages/coffee-script/src/lexer.coffee +653 -0
  179. data/packages/coffee-script/src/nodes.coffee +1754 -0
  180. data/packages/coffee-script/src/optparse.coffee +99 -0
  181. data/packages/coffee-script/src/repl.coffee +42 -0
  182. data/packages/coffee-script/src/rewriter.coffee +326 -0
  183. data/packages/coffee-script/src/scope.coffee +94 -0
  184. data/packages/coffee-script/test/arguments.coffee +127 -0
  185. data/packages/coffee-script/test/assignment.coffee +98 -0
  186. data/packages/coffee-script/test/break.coffee +18 -0
  187. data/packages/coffee-script/test/comments.coffee +201 -0
  188. data/packages/coffee-script/test/conditionals.coffee +181 -0
  189. data/packages/coffee-script/test/exception_handling.coffee +90 -0
  190. data/packages/coffee-script/test/helpers.coffee +96 -0
  191. data/packages/coffee-script/test/importing.coffee +18 -0
  192. data/packages/coffee-script/test/operators.coffee +225 -0
  193. data/packages/coffee-script/test/ranges_slices_and_splices.coffee +186 -0
  194. data/packages/coffee-script/test/regular_expressions.coffee +56 -0
  195. data/packages/coffee-script/test/test.html +123 -0
  196. data/packages/coffee-script/test/test_chaining.coffee +77 -0
  197. data/packages/coffee-script/test/test_classes.coffee +372 -0
  198. data/packages/coffee-script/test/test_compilation.coffee +26 -0
  199. data/packages/coffee-script/test/test_comprehensions.coffee +318 -0
  200. data/packages/coffee-script/test/test_existence.coffee +165 -0
  201. data/packages/coffee-script/test/test_functions.coffee +379 -0
  202. data/packages/coffee-script/test/test_heredocs.coffee +111 -0
  203. data/packages/coffee-script/test/test_literals.coffee +270 -0
  204. data/packages/coffee-script/test/test_option_parser.coffee +27 -0
  205. data/packages/coffee-script/test/test_pattern_matching.coffee +162 -0
  206. data/packages/coffee-script/test/test_returns.coffee +63 -0
  207. data/packages/coffee-script/test/test_splats.coffee +102 -0
  208. data/packages/coffee-script/test/test_strings.coffee +118 -0
  209. data/packages/coffee-script/test/test_switch.coffee +103 -0
  210. data/packages/coffee-script/test/test_while.coffee +71 -0
  211. data/packages/ivory/LICENSE.txt +1 -0
  212. data/packages/ivory/README.md +19 -0
  213. data/packages/ivory/lib/buffer.js +111 -0
  214. data/packages/ivory/lib/events.js +137 -0
  215. data/packages/ivory/lib/fs.js +266 -0
  216. data/packages/ivory/lib/main.js +13 -0
  217. data/packages/ivory/lib/path.js +158 -0
  218. data/packages/ivory/lib/ruby/buffer.rb +145 -0
  219. data/packages/ivory/lib/ruby/constants.rb +585 -0
  220. data/packages/ivory/lib/ruby/events.rb +32 -0
  221. data/packages/ivory/lib/ruby/fs.rb +245 -0
  222. data/packages/ivory/lib/ruby/process.rb +28 -0
  223. data/packages/ivory/lib/stream.js +115 -0
  224. data/packages/ivory/lib/util.js +414 -0
  225. data/packages/ivory/package.json +11 -0
  226. data/packages/ivory/spade-boot.js +78 -0
  227. data/packages/jquery/main.js +7179 -0
  228. data/packages/jquery/package.json +10 -0
  229. data/packages/json/lib/main.js +14 -0
  230. data/packages/json/package.json +8 -0
  231. data/packages/lproj/README.md +77 -0
  232. data/packages/lproj/examples/demo-app/en.lproj/localized.strings +2 -0
  233. data/packages/lproj/examples/demo-app/fr.lproj/localized.strings +3 -0
  234. data/packages/lproj/examples/demo-app/index.html +8 -0
  235. data/packages/lproj/examples/demo-app/lib/main.js +7 -0
  236. data/packages/lproj/examples/demo-app/package.json +9 -0
  237. data/packages/lproj/lib/main.js +78 -0
  238. data/packages/lproj/lib/strings-format.js +6 -0
  239. data/packages/lproj/package.json +9 -0
  240. data/packages/optparse/README.md +161 -0
  241. data/packages/optparse/TODO +1 -0
  242. data/packages/optparse/examples/browser-test.html +75 -0
  243. data/packages/optparse/examples/nodejs-test.js +90 -0
  244. data/packages/optparse/lib/optparse.js +309 -0
  245. data/packages/optparse/package.json +13 -0
  246. data/packages/optparse/seed.yml +5 -0
  247. data/packages/text/lib/main.js +8 -0
  248. data/packages/text/package.json +9 -0
  249. data/packages/web-file/README.md +7 -0
  250. data/packages/web-file/lib/errors.js +32 -0
  251. data/packages/web-file/lib/file-reader.js +10 -0
  252. data/packages/web-file/lib/file-system.js +234 -0
  253. data/packages/web-file/lib/file-writer.js +10 -0
  254. data/packages/web-file/lib/file.js +9 -0
  255. data/packages/web-file/lib/main.js +34 -0
  256. data/packages/web-file/lib/platform.js +25 -0
  257. data/packages/web-file/lib/ruby/file.rb +252 -0
  258. data/packages/web-file/lib/ruby/file_reader.rb +69 -0
  259. data/packages/web-file/lib/ruby/file_system.rb +134 -0
  260. data/packages/web-file/lib/ruby/file_writer.rb +78 -0
  261. data/packages/web-file/package.json +12 -0
  262. data/packages/web-typed-array/README.md +7 -0
  263. data/packages/web-typed-array/lib/array-buffer-view.js +9 -0
  264. data/packages/web-typed-array/lib/array-buffer.js +7 -0
  265. data/packages/web-typed-array/lib/main.js +33 -0
  266. data/packages/web-typed-array/lib/platform.js +20 -0
  267. data/packages/web-typed-array/lib/ruby/array_buffer.rb +31 -0
  268. data/packages/web-typed-array/lib/ruby/array_buffer_view.rb +130 -0
  269. data/packages/web-typed-array/lib/ruby/typed_array.rb +133 -0
  270. data/packages/web-typed-array/lib/typed-array.js +26 -0
  271. data/packages/web-typed-array/package.json +9 -0
  272. data/spade-packager.gemspec +39 -0
  273. data/spec/cli/build_spec.rb +57 -0
  274. data/spec/cli/install_spec.rb +119 -0
  275. data/spec/cli/installed_spec.rb +55 -0
  276. data/spec/cli/list_spec.rb +74 -0
  277. data/spec/cli/login_spec.rb +75 -0
  278. data/spec/cli/new_spec.rb +5 -0
  279. data/spec/cli/owner_spec.rb +114 -0
  280. data/spec/cli/push_spec.rb +73 -0
  281. data/spec/cli/uninstall_spec.rb +58 -0
  282. data/spec/cli/unpack_spec.rb +72 -0
  283. data/spec/cli/unyank_spec.rb +73 -0
  284. data/spec/cli/yank_spec.rb +73 -0
  285. data/spec/credentials_spec.rb +23 -0
  286. data/spec/fixtures/badrake-0.8.7.spd +0 -0
  287. data/spec/fixtures/builder-3.0.0.spd +0 -0
  288. data/spec/fixtures/bundler-1.1.pre.spd +0 -0
  289. data/spec/fixtures/coffee-1.0.1.pre.spd +0 -0
  290. data/spec/fixtures/core-test-0.4.3.spd +0 -0
  291. data/spec/fixtures/core-test/bin/cot +3 -0
  292. data/spec/fixtures/core-test/lib/main.js +1 -0
  293. data/spec/fixtures/core-test/resources/runner.css +0 -0
  294. data/spec/fixtures/core-test/tests/test.js +1 -0
  295. data/spec/fixtures/highline-1.6.1.spd +0 -0
  296. data/spec/fixtures/ivory-0.0.1.spd +0 -0
  297. data/spec/fixtures/jquery-1.4.3.spd +0 -0
  298. data/spec/fixtures/optparse-1.0.1.spd +0 -0
  299. data/spec/fixtures/package.json +30 -0
  300. data/spec/fixtures/rake-0.8.6.spd +0 -0
  301. data/spec/fixtures/rake-0.8.7.spd +0 -0
  302. data/spec/gauntlet_spec.rb +27 -0
  303. data/spec/package_spec.rb +267 -0
  304. data/spec/spec_helper.rb +32 -0
  305. data/spec/support/cli.rb +103 -0
  306. data/spec/support/fake.rb +48 -0
  307. data/spec/support/fake_gem_server.rb +67 -0
  308. data/spec/support/fake_gemcutter.rb +50 -0
  309. data/spec/support/matchers.rb +32 -0
  310. data/spec/support/path.rb +61 -0
  311. data/templates/project/LICENSE +19 -0
  312. data/templates/project/README.md +21 -0
  313. data/templates/project/lib/main.js +9 -0
  314. data/templates/project/project.json +31 -0
  315. data/templates/project/tests/main-test.js +8 -0
  316. 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']