spade-packager 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
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']