marty 0.5.12

Sign up to get free protection for your applications and to get access to all the features.
Files changed (345) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.md +84 -0
  4. data/Rakefile +29 -0
  5. data/app/assets/javascripts/marty/application.js +15 -0
  6. data/app/assets/stylesheets/marty/application.css +13 -0
  7. data/app/components/marty/api_auth_view.rb +32 -0
  8. data/app/components/marty/auth_app.rb +55 -0
  9. data/app/components/marty/auth_app.rb~ +51 -0
  10. data/app/components/marty/auth_app/javascripts/auth_app.js +91 -0
  11. data/app/components/marty/auth_app/javascripts/auth_app.js~ +91 -0
  12. data/app/components/marty/cm_form_panel.rb~ +5 -0
  13. data/app/components/marty/cm_grid_panel.rb~ +35 -0
  14. data/app/components/marty/data_import_view.rb~ +142 -0
  15. data/app/components/marty/extras/layout.rb +46 -0
  16. data/app/components/marty/extras/layout.rb~ +46 -0
  17. data/app/components/marty/extras/misc.rb +18 -0
  18. data/app/components/marty/form.rb +6 -0
  19. data/app/components/marty/grid.rb +45 -0
  20. data/app/components/marty/grid_append_only.rb +12 -0
  21. data/app/components/marty/import_type_view.rb +53 -0
  22. data/app/components/marty/live_search_grid_panel.rb +46 -0
  23. data/app/components/marty/live_search_grid_panel.rb~ +49 -0
  24. data/app/components/marty/main_auth_app.rb +269 -0
  25. data/app/components/marty/main_auth_app.rb~ +238 -0
  26. data/app/components/marty/mcfly_grid_panel.rb +62 -0
  27. data/app/components/marty/mcfly_grid_panel.rb~ +80 -0
  28. data/app/components/marty/new_posting_form.rb +46 -0
  29. data/app/components/marty/new_posting_form.rb~ +46 -0
  30. data/app/components/marty/new_posting_window.rb +21 -0
  31. data/app/components/marty/new_posting_window.rb~ +21 -0
  32. data/app/components/marty/panel.rb +12 -0
  33. data/app/components/marty/pivot_grid.rb +52 -0
  34. data/app/components/marty/pivot_grid/endpoints.rb +45 -0
  35. data/app/components/marty/pivot_grid/javascripts/extensions.js +150 -0
  36. data/app/components/marty/pivot_grid/javascripts/pivot_grid.js +86 -0
  37. data/app/components/marty/pivot_grid/services.rb +44 -0
  38. data/app/components/marty/posting_grid.rb +139 -0
  39. data/app/components/marty/posting_grid.rb~ +140 -0
  40. data/app/components/marty/posting_window.rb +27 -0
  41. data/app/components/marty/promise_view.rb +177 -0
  42. data/app/components/marty/promise_view.rb~ +157 -0
  43. data/app/components/marty/promise_view/stylesheets/promise_view.css +26 -0
  44. data/app/components/marty/promise_view/stylesheets/promise_view.css~ +15 -0
  45. data/app/components/marty/report_form.rb +225 -0
  46. data/app/components/marty/report_form.rb~ +217 -0
  47. data/app/components/marty/report_select.rb +145 -0
  48. data/app/components/marty/report_select.rb~ +133 -0
  49. data/app/components/marty/reporting.rb +39 -0
  50. data/app/components/marty/reporting.rb~ +39 -0
  51. data/app/components/marty/script_detail.rb~ +430 -0
  52. data/app/components/marty/script_form.rb +233 -0
  53. data/app/components/marty/script_form.rb~ +233 -0
  54. data/app/components/marty/script_form/javascripts/Ext.ux.form.field.CodeMirror.js +698 -0
  55. data/app/components/marty/script_form/javascripts/Ext.ux.form.field.CodeMirror.js~ +909 -0
  56. data/app/components/marty/script_form/javascripts/codemirror.js +3130 -0
  57. data/app/components/marty/script_form/javascripts/mode/clike/clike.js +284 -0
  58. data/app/components/marty/script_form/javascripts/mode/clike/index.html +102 -0
  59. data/app/components/marty/script_form/javascripts/mode/clike/scala.html +766 -0
  60. data/app/components/marty/script_form/javascripts/mode/clojure/clojure.js +206 -0
  61. data/app/components/marty/script_form/javascripts/mode/clojure/index.html +67 -0
  62. data/app/components/marty/script_form/javascripts/mode/coffeescript/LICENSE +22 -0
  63. data/app/components/marty/script_form/javascripts/mode/coffeescript/coffeescript.js +346 -0
  64. data/app/components/marty/script_form/javascripts/mode/coffeescript/index.html +728 -0
  65. data/app/components/marty/script_form/javascripts/mode/commonlisp/commonlisp.js +101 -0
  66. data/app/components/marty/script_form/javascripts/mode/commonlisp/index.html +165 -0
  67. data/app/components/marty/script_form/javascripts/mode/css/css.js +448 -0
  68. data/app/components/marty/script_form/javascripts/mode/css/index.html +58 -0
  69. data/app/components/marty/script_form/javascripts/mode/css/test.js +501 -0
  70. data/app/components/marty/script_form/javascripts/mode/delorean/delorean.js +189 -0
  71. data/app/components/marty/script_form/javascripts/mode/diff/diff.js +32 -0
  72. data/app/components/marty/script_form/javascripts/mode/diff/index.html +105 -0
  73. data/app/components/marty/script_form/javascripts/mode/ecl/ecl.js +203 -0
  74. data/app/components/marty/script_form/javascripts/mode/ecl/index.html +42 -0
  75. data/app/components/marty/script_form/javascripts/mode/erlang/erlang.js +463 -0
  76. data/app/components/marty/script_form/javascripts/mode/erlang/index.html +63 -0
  77. data/app/components/marty/script_form/javascripts/mode/gfm/gfm.js +150 -0
  78. data/app/components/marty/script_form/javascripts/mode/gfm/index.html +48 -0
  79. data/app/components/marty/script_form/javascripts/mode/go/go.js +170 -0
  80. data/app/components/marty/script_form/javascripts/mode/go/index.html +73 -0
  81. data/app/components/marty/script_form/javascripts/mode/groovy/groovy.js +210 -0
  82. data/app/components/marty/script_form/javascripts/mode/groovy/index.html +72 -0
  83. data/app/components/marty/script_form/javascripts/mode/haskell/haskell.js +242 -0
  84. data/app/components/marty/script_form/javascripts/mode/haskell/index.html +61 -0
  85. data/app/components/marty/script_form/javascripts/mode/haxe/haxe.js +429 -0
  86. data/app/components/marty/script_form/javascripts/mode/haxe/index.html +91 -0
  87. data/app/components/marty/script_form/javascripts/mode/htmlembedded/htmlembedded.js +72 -0
  88. data/app/components/marty/script_form/javascripts/mode/htmlembedded/index.html +50 -0
  89. data/app/components/marty/script_form/javascripts/mode/htmlmixed/htmlmixed.js +84 -0
  90. data/app/components/marty/script_form/javascripts/mode/htmlmixed/index.html +52 -0
  91. data/app/components/marty/script_form/javascripts/mode/javascript/index.html +78 -0
  92. data/app/components/marty/script_form/javascripts/mode/javascript/javascript.js +361 -0
  93. data/app/components/marty/script_form/javascripts/mode/jinja2/index.html +38 -0
  94. data/app/components/marty/script_form/javascripts/mode/jinja2/jinja2.js +42 -0
  95. data/app/components/marty/script_form/javascripts/mode/less/index.html +740 -0
  96. data/app/components/marty/script_form/javascripts/mode/less/less.js +266 -0
  97. data/app/components/marty/script_form/javascripts/mode/lua/index.html +73 -0
  98. data/app/components/marty/script_form/javascripts/mode/lua/lua.js +140 -0
  99. data/app/components/marty/script_form/javascripts/mode/markdown/index.html +343 -0
  100. data/app/components/marty/script_form/javascripts/mode/markdown/markdown.js +382 -0
  101. data/app/components/marty/script_form/javascripts/mode/markdown/test.js +1084 -0
  102. data/app/components/marty/script_form/javascripts/mode/mysql/index.html +42 -0
  103. data/app/components/marty/script_form/javascripts/mode/mysql/mysql.js +186 -0
  104. data/app/components/marty/script_form/javascripts/mode/ntriples/index.html +33 -0
  105. data/app/components/marty/script_form/javascripts/mode/ntriples/ntriples.js +172 -0
  106. data/app/components/marty/script_form/javascripts/mode/ocaml/index.html +130 -0
  107. data/app/components/marty/script_form/javascripts/mode/ocaml/ocaml.js +114 -0
  108. data/app/components/marty/script_form/javascripts/mode/pascal/LICENSE +7 -0
  109. data/app/components/marty/script_form/javascripts/mode/pascal/index.html +49 -0
  110. data/app/components/marty/script_form/javascripts/mode/pascal/pascal.js +94 -0
  111. data/app/components/marty/script_form/javascripts/mode/perl/LICENSE +19 -0
  112. data/app/components/marty/script_form/javascripts/mode/perl/index.html +63 -0
  113. data/app/components/marty/script_form/javascripts/mode/perl/perl.js +816 -0
  114. data/app/components/marty/script_form/javascripts/mode/php/index.html +49 -0
  115. data/app/components/marty/script_form/javascripts/mode/php/php.js +148 -0
  116. data/app/components/marty/script_form/javascripts/mode/pig/index.html +43 -0
  117. data/app/components/marty/script_form/javascripts/mode/pig/pig.js +172 -0
  118. data/app/components/marty/script_form/javascripts/mode/plsql/index.html +63 -0
  119. data/app/components/marty/script_form/javascripts/mode/plsql/plsql.js +217 -0
  120. data/app/components/marty/script_form/javascripts/mode/properties/index.html +41 -0
  121. data/app/components/marty/script_form/javascripts/mode/properties/properties.js +63 -0
  122. data/app/components/marty/script_form/javascripts/mode/python/LICENSE.txt +21 -0
  123. data/app/components/marty/script_form/javascripts/mode/python/index.html +123 -0
  124. data/app/components/marty/script_form/javascripts/mode/python/python.js +338 -0
  125. data/app/components/marty/script_form/javascripts/mode/r/LICENSE +24 -0
  126. data/app/components/marty/script_form/javascripts/mode/r/index.html +74 -0
  127. data/app/components/marty/script_form/javascripts/mode/r/r.js +141 -0
  128. data/app/components/marty/script_form/javascripts/mode/rpm/changes/changes.js +19 -0
  129. data/app/components/marty/script_form/javascripts/mode/rpm/changes/index.html +54 -0
  130. data/app/components/marty/script_form/javascripts/mode/rpm/spec/index.html +100 -0
  131. data/app/components/marty/script_form/javascripts/mode/rpm/spec/spec.css +5 -0
  132. data/app/components/marty/script_form/javascripts/mode/rpm/spec/spec.js +66 -0
  133. data/app/components/marty/script_form/javascripts/mode/rst/index.html +526 -0
  134. data/app/components/marty/script_form/javascripts/mode/rst/rst.js +326 -0
  135. data/app/components/marty/script_form/javascripts/mode/ruby/LICENSE +24 -0
  136. data/app/components/marty/script_form/javascripts/mode/ruby/index.html +172 -0
  137. data/app/components/marty/script_form/javascripts/mode/ruby/ruby.js +195 -0
  138. data/app/components/marty/script_form/javascripts/mode/rust/index.html +49 -0
  139. data/app/components/marty/script_form/javascripts/mode/rust/rust.js +432 -0
  140. data/app/components/marty/script_form/javascripts/mode/scheme/index.html +65 -0
  141. data/app/components/marty/script_form/javascripts/mode/scheme/scheme.js +230 -0
  142. data/app/components/marty/script_form/javascripts/mode/shell/index.html +50 -0
  143. data/app/components/marty/script_form/javascripts/mode/shell/shell.js +118 -0
  144. data/app/components/marty/script_form/javascripts/mode/sieve/LICENSE +23 -0
  145. data/app/components/marty/script_form/javascripts/mode/sieve/index.html +81 -0
  146. data/app/components/marty/script_form/javascripts/mode/sieve/sieve.js +156 -0
  147. data/app/components/marty/script_form/javascripts/mode/smalltalk/index.html +56 -0
  148. data/app/components/marty/script_form/javascripts/mode/smalltalk/smalltalk.js +139 -0
  149. data/app/components/marty/script_form/javascripts/mode/smarty/index.html +83 -0
  150. data/app/components/marty/script_form/javascripts/mode/smarty/smarty.js +148 -0
  151. data/app/components/marty/script_form/javascripts/mode/sparql/index.html +41 -0
  152. data/app/components/marty/script_form/javascripts/mode/sparql/sparql.js +143 -0
  153. data/app/components/marty/script_form/javascripts/mode/stex/index.html +98 -0
  154. data/app/components/marty/script_form/javascripts/mode/stex/stex.js +182 -0
  155. data/app/components/marty/script_form/javascripts/mode/stex/test.js +343 -0
  156. data/app/components/marty/script_form/javascripts/mode/tiddlywiki/index.html +141 -0
  157. data/app/components/marty/script_form/javascripts/mode/tiddlywiki/tiddlywiki.css +14 -0
  158. data/app/components/marty/script_form/javascripts/mode/tiddlywiki/tiddlywiki.js +384 -0
  159. data/app/components/marty/script_form/javascripts/mode/tiki/index.html +83 -0
  160. data/app/components/marty/script_form/javascripts/mode/tiki/tiki.css +26 -0
  161. data/app/components/marty/script_form/javascripts/mode/tiki/tiki.js +309 -0
  162. data/app/components/marty/script_form/javascripts/mode/vb/LICENSE.txt +21 -0
  163. data/app/components/marty/script_form/javascripts/mode/vb/index.html +89 -0
  164. data/app/components/marty/script_form/javascripts/mode/vb/vb.js +260 -0
  165. data/app/components/marty/script_form/javascripts/mode/vbscript/index.html +43 -0
  166. data/app/components/marty/script_form/javascripts/mode/vbscript/vbscript.js +26 -0
  167. data/app/components/marty/script_form/javascripts/mode/velocity/index.html +104 -0
  168. data/app/components/marty/script_form/javascripts/mode/velocity/velocity.js +146 -0
  169. data/app/components/marty/script_form/javascripts/mode/verilog/index.html +211 -0
  170. data/app/components/marty/script_form/javascripts/mode/verilog/verilog.js +194 -0
  171. data/app/components/marty/script_form/javascripts/mode/xml/index.html +45 -0
  172. data/app/components/marty/script_form/javascripts/mode/xml/xml.js +318 -0
  173. data/app/components/marty/script_form/javascripts/mode/xquery/LICENSE +20 -0
  174. data/app/components/marty/script_form/javascripts/mode/xquery/index.html +223 -0
  175. data/app/components/marty/script_form/javascripts/mode/xquery/test/index.html +27 -0
  176. data/app/components/marty/script_form/javascripts/mode/xquery/test/testBase.js +42 -0
  177. data/app/components/marty/script_form/javascripts/mode/xquery/test/testEmptySequenceKeyword.js +16 -0
  178. data/app/components/marty/script_form/javascripts/mode/xquery/test/testMultiAttr.js +16 -0
  179. data/app/components/marty/script_form/javascripts/mode/xquery/test/testNamespaces.js +91 -0
  180. data/app/components/marty/script_form/javascripts/mode/xquery/test/testProcessingInstructions.js +16 -0
  181. data/app/components/marty/script_form/javascripts/mode/xquery/test/testQuotes.js +19 -0
  182. data/app/components/marty/script_form/javascripts/mode/xquery/xquery.js +451 -0
  183. data/app/components/marty/script_form/javascripts/mode/yaml/index.html +68 -0
  184. data/app/components/marty/script_form/javascripts/mode/yaml/yaml.js +95 -0
  185. data/app/components/marty/script_form/javascripts/util/closetag.js +164 -0
  186. data/app/components/marty/script_form/javascripts/util/dialog.css +27 -0
  187. data/app/components/marty/script_form/javascripts/util/dialog.js +70 -0
  188. data/app/components/marty/script_form/javascripts/util/foldcode.js +196 -0
  189. data/app/components/marty/script_form/javascripts/util/formatting.js +193 -0
  190. data/app/components/marty/script_form/javascripts/util/javascript-hint.js +134 -0
  191. data/app/components/marty/script_form/javascripts/util/loadmode.js +51 -0
  192. data/app/components/marty/script_form/javascripts/util/match-highlighter.js +44 -0
  193. data/app/components/marty/script_form/javascripts/util/multiplex.js +77 -0
  194. data/app/components/marty/script_form/javascripts/util/overlay.js +54 -0
  195. data/app/components/marty/script_form/javascripts/util/pig-hint.js +123 -0
  196. data/app/components/marty/script_form/javascripts/util/runmode-standalone.js +90 -0
  197. data/app/components/marty/script_form/javascripts/util/runmode.js +53 -0
  198. data/app/components/marty/script_form/javascripts/util/search.js +118 -0
  199. data/app/components/marty/script_form/javascripts/util/searchcursor.js +119 -0
  200. data/app/components/marty/script_form/javascripts/util/simple-hint.css +16 -0
  201. data/app/components/marty/script_form/javascripts/util/simple-hint.js +97 -0
  202. data/app/components/marty/script_form/javascripts/util/xml-hint.js +137 -0
  203. data/app/components/marty/script_form/stylesheets/codemirror.css +172 -0
  204. data/app/components/marty/script_form/stylesheets/delorean.css +10 -0
  205. data/app/components/marty/script_form/stylesheets/theme/ambiance.css +81 -0
  206. data/app/components/marty/script_form/stylesheets/theme/blackboard.css +25 -0
  207. data/app/components/marty/script_form/stylesheets/theme/cobalt.css +18 -0
  208. data/app/components/marty/script_form/stylesheets/theme/eclipse.css +25 -0
  209. data/app/components/marty/script_form/stylesheets/theme/elegant.css +10 -0
  210. data/app/components/marty/script_form/stylesheets/theme/erlang-dark.css +21 -0
  211. data/app/components/marty/script_form/stylesheets/theme/lesser-dark.css +44 -0
  212. data/app/components/marty/script_form/stylesheets/theme/monokai.css +28 -0
  213. data/app/components/marty/script_form/stylesheets/theme/neat.css +9 -0
  214. data/app/components/marty/script_form/stylesheets/theme/night.css +21 -0
  215. data/app/components/marty/script_form/stylesheets/theme/rubyblue.css +21 -0
  216. data/app/components/marty/script_form/stylesheets/theme/vibrant-ink.css +27 -0
  217. data/app/components/marty/script_form/stylesheets/theme/xq-dark.css +46 -0
  218. data/app/components/marty/script_grid.rb +104 -0
  219. data/app/components/marty/script_grid.rb~ +99 -0
  220. data/app/components/marty/script_tester.rb +114 -0
  221. data/app/components/marty/script_tester.rb~ +213 -0
  222. data/app/components/marty/scripting.rb +132 -0
  223. data/app/components/marty/scripting.rb~ +124 -0
  224. data/app/components/marty/select_report.rb~ +143 -0
  225. data/app/components/marty/simple_app.rb +97 -0
  226. data/app/components/marty/simple_app.rb~ +101 -0
  227. data/app/components/marty/simple_app/javascripts/simple_app.js +50 -0
  228. data/app/components/marty/simple_app/javascripts/statusbar_ext.js +8 -0
  229. data/app/components/marty/tag_grid.rb +83 -0
  230. data/app/components/marty/tag_grid.rb~ +89 -0
  231. data/app/components/marty/tree_panel.rb~ +256 -0
  232. data/app/components/marty/tree_panel/javascripts/tree_panel.js~ +317 -0
  233. data/app/components/marty/user_pivot.rb +128 -0
  234. data/app/components/marty/user_view.rb +181 -0
  235. data/app/components/marty/user_view.rb~ +188 -0
  236. data/app/controllers/marty/application_controller.rb +124 -0
  237. data/app/controllers/marty/application_controller.rb~ +133 -0
  238. data/app/controllers/marty/components_controller.rb +41 -0
  239. data/app/controllers/marty/components_controller.rb~ +37 -0
  240. data/app/controllers/marty/job_controller.rb +28 -0
  241. data/app/controllers/marty/job_controller.rb~ +28 -0
  242. data/app/controllers/marty/rpc_controller.rb +64 -0
  243. data/app/controllers/marty/rpc_controller.rb~ +61 -0
  244. data/app/helpers/marty/application_helper.rb +4 -0
  245. data/app/helpers/marty/script_set.rb +57 -0
  246. data/app/helpers/marty/script_set.rb~ +59 -0
  247. data/app/models/marty/api_auth.rb +44 -0
  248. data/app/models/marty/api_auth.rb~ +48 -0
  249. data/app/models/marty/base.rb +4 -0
  250. data/app/models/marty/data_change.rb +179 -0
  251. data/app/models/marty/data_change.rb~ +141 -0
  252. data/app/models/marty/enum.rb +22 -0
  253. data/app/models/marty/enum.rb~ +16 -0
  254. data/app/models/marty/import_type.rb +44 -0
  255. data/app/models/marty/import_type.rb~ +48 -0
  256. data/app/models/marty/poop.rb~ +169 -0
  257. data/app/models/marty/posting.rb +101 -0
  258. data/app/models/marty/posting.rb~ +86 -0
  259. data/app/models/marty/posting_type.rb +12 -0
  260. data/app/models/marty/posting_type.rb~ +21 -0
  261. data/app/models/marty/promise.rb +252 -0
  262. data/app/models/marty/promise.rb~ +196 -0
  263. data/app/models/marty/role.rb +6 -0
  264. data/app/models/marty/role.rb~ +10 -0
  265. data/app/models/marty/script.rb +144 -0
  266. data/app/models/marty/script.rb~ +62 -0
  267. data/app/models/marty/tag.rb +96 -0
  268. data/app/models/marty/tag.rb~ +91 -0
  269. data/app/models/marty/token.rb +30 -0
  270. data/app/models/marty/user.rb +146 -0
  271. data/app/models/marty/user.rb~ +148 -0
  272. data/app/models/marty/user_role.rb +7 -0
  273. data/app/models/marty/user_role.rb~ +13 -0
  274. data/app/views/layouts/marty/application.html.erb +12 -0
  275. data/app/views/layouts/marty/application.html.erb~ +11 -0
  276. data/config/locales/en.yml +134 -0
  277. data/config/routes.rb +6 -0
  278. data/config/routes.rb~ +10 -0
  279. data/db/migrate/001_create_marty_scripts.rb +14 -0
  280. data/db/migrate/003_create_marty_users.rb +12 -0
  281. data/db/migrate/004_create_marty_roles.rb +7 -0
  282. data/db/migrate/005_create_marty_user_roles.rb +14 -0
  283. data/db/migrate/006_create_marty_tokens.rb +14 -0
  284. data/db/migrate/008_create_marty_posting_types.rb +7 -0
  285. data/db/migrate/019_create_marty_postings.rb +18 -0
  286. data/db/migrate/019_create_marty_postings.rb~ +19 -0
  287. data/db/migrate/068_create_marty_import_types.rb +12 -0
  288. data/db/migrate/069_create_marty_import_synonyms.rb +15 -0
  289. data/db/migrate/070_create_versions.rb +18 -0
  290. data/db/migrate/071_add_object_changes_column_to_versions.rb +9 -0
  291. data/db/migrate/072_add_validation_function_to_import_types.rb +6 -0
  292. data/db/migrate/073_add_preprocess_function_to_import_types.rb +5 -0
  293. data/db/migrate/090_create_delayed_jobs.rb +22 -0
  294. data/db/migrate/091_create_marty_promises.rb +36 -0
  295. data/db/migrate/095_create_marty_tags.rb +14 -0
  296. data/db/migrate/095_create_marty_tags.rb~ +19 -0
  297. data/db/migrate/096_add_user_roles_to_import_types.rb +11 -0
  298. data/db/migrate/097_drop_versions.rb +9 -0
  299. data/db/migrate/098_create_marty_api_auths.rb +20 -0
  300. data/db/seeds.rb +48 -0
  301. data/lib/marty.rb +18 -0
  302. data/lib/marty.rb~ +13 -0
  303. data/lib/marty/content_handler.rb +97 -0
  304. data/lib/marty/content_handler.rb~ +93 -0
  305. data/lib/marty/data_conversion.rb +298 -0
  306. data/lib/marty/data_exporter.rb +150 -0
  307. data/lib/marty/data_exporter.rb~ +137 -0
  308. data/lib/marty/data_importer.rb +122 -0
  309. data/lib/marty/data_importer.rb~ +114 -0
  310. data/lib/marty/data_row_processor.rb~ +206 -0
  311. data/lib/marty/drop_folder_hook.rb~ +17 -0
  312. data/lib/marty/engine.rb +10 -0
  313. data/lib/marty/folder_hook.rb~ +9 -0
  314. data/lib/marty/lazy_column_loader.rb +57 -0
  315. data/lib/marty/lazy_column_loader.rb~ +47 -0
  316. data/lib/marty/mcfly_query.rb +189 -0
  317. data/lib/marty/mcfly_query.rb~ +188 -0
  318. data/lib/marty/migrations.rb +108 -0
  319. data/lib/marty/migrations.rb~ +65 -0
  320. data/lib/marty/monkey.rb +163 -0
  321. data/lib/marty/monkey.rb~ +160 -0
  322. data/lib/marty/permissions.rb +64 -0
  323. data/lib/marty/permissions.rb~ +69 -0
  324. data/lib/marty/promise.rb~ +41 -0
  325. data/lib/marty/promise_job.rb +123 -0
  326. data/lib/marty/promise_job.rb~ +121 -0
  327. data/lib/marty/promise_proxy.rb +94 -0
  328. data/lib/marty/promise_proxy.rb~ +69 -0
  329. data/lib/marty/railtie.rb +5 -0
  330. data/lib/marty/relation.rb +39 -0
  331. data/lib/marty/util.rb +110 -0
  332. data/lib/marty/util.rb~ +80 -0
  333. data/lib/marty/version.rb +3 -0
  334. data/lib/marty/version.rb~ +3 -0
  335. data/lib/marty/xl.rb +527 -0
  336. data/lib/marty/xl.rb~ +526 -0
  337. data/lib/pyxll/README.txt +19 -0
  338. data/lib/pyxll/README.txt~ +16 -0
  339. data/lib/pyxll/gemini.py +155 -0
  340. data/lib/pyxll/gemini.py~ +110 -0
  341. data/lib/pyxll/pyxll.cfg +12 -0
  342. data/lib/pyxll/pyxll.cfg~ +12 -0
  343. data/lib/pyxll/sample.xlsx +0 -0
  344. data/lib/tasks/marty_tasks.rake +37 -0
  345. metadata +517 -0
data/lib/marty/xl.rb~ ADDED
@@ -0,0 +1,526 @@
1
+ require 'axlsx'
2
+ require 'delorean_lang'
3
+
4
+ class Marty::Xl
5
+ include Delorean::Model
6
+
7
+ def self.spreadsheet(worksheets)
8
+ xl = Marty::Xl.new(worksheets)
9
+ xl.package
10
+ end
11
+
12
+ def deep_copy(value)
13
+ return value.each_with_object({}){|(k, v), h| h[k] = deep_copy(v)} if
14
+ value.is_a?(Hash)
15
+
16
+ value.is_a?(Array) ? value.map{|v| deep_copy(v)} : value
17
+ end
18
+
19
+ def merge_cell_edges(a, b)
20
+ return b unless a.kind_of?(Hash) && b.kind_of?(Hash)
21
+
22
+ a_border, b_border = a[:border], b[:border]
23
+
24
+ return b unless a_border.is_a?(Hash) && a_border[:edges].is_a?(Array)
25
+
26
+ non_match = a_border.detect {
27
+ |key, value|
28
+ key != :edges && b_border[key] != value
29
+ }
30
+
31
+ a_border[:edges].each do |edge|
32
+ unless b_border[:edges].include? edge
33
+ # add new edges:
34
+ b_border[:edges] << edge
35
+
36
+ # add new style/color for the new edge if there is no style
37
+ # match with the old edges:
38
+ b["border_#{edge}".to_sym] = a_border.each_with_object({}) {
39
+ |(key, value), h|
40
+ h[key] = value unless key == :edges
41
+ } if non_match
42
+ end
43
+ end
44
+ b
45
+ end
46
+
47
+ def merge_row_edges(a, b)
48
+ return b unless a.count > 0
49
+ a.each_index do |ind|
50
+ b[ind] = merge_cell_edges(a[ind], deep_copy(b[ind]))
51
+ end
52
+ b
53
+ end
54
+
55
+ def bordered_cells(a, b)
56
+ a.each_index do |c|
57
+ a[c] = merge_cell_edges(b[c], deep_copy(a[c])) if a[c] && b[c]
58
+ b[c] = a[c] unless a[c].nil?
59
+ end
60
+ b.each_index { |el| b[el] = {} unless b[el] }
61
+ end
62
+
63
+ def position_row(d, column_offset, r_number, rows, styles, row_styles)
64
+ rows[r_number] ||= []
65
+ styles[r_number] ||= []
66
+ row_styles[r_number] ||= {}
67
+
68
+ new_row, new_style = rows[r_number], styles[r_number]
69
+
70
+ (0...column_offset).each do |t|
71
+ new_row[t] ||= ""
72
+ new_style[t] ||= {}
73
+ end
74
+
75
+ d[1].each_index do |c_index|
76
+ new_row[c_index+column_offset] = d[1][c_index]
77
+ end if d[1].kind_of?(Array)
78
+
79
+ if (d.length > 2) && d[2].kind_of?(Hash) && d[2]["style"].kind_of?(Array)
80
+ d[2]["style"].each_index do |c_index|
81
+ new_style[c_index+column_offset] = d[2]["style"][c_index]
82
+ end
83
+ end
84
+
85
+ # apply style for the row as a whole:
86
+ if (d.length > 2) && d[2].kind_of?(Hash)
87
+ d[2].each do |key, value|
88
+ unless key == :style.to_s
89
+ row_styles[r_number][key] = value
90
+ else
91
+ # skip if the style is an array: /style as an array is
92
+ # handled by the 'apply a style to each cell' section/
93
+ next unless value.kind_of?(Hash)
94
+ d[1].length.times do |t|
95
+ new_style[t+column_offset] = value
96
+ end
97
+ end
98
+ end
99
+ end
100
+ # row data, style:
101
+ rows[r_number], styles[r_number] = new_row, new_style
102
+ end
103
+
104
+ def position_elem(d, offset, last_row, el)
105
+ x1, y1, x2, y2, w, h = d[1]
106
+ column_offset, row_offset = offset
107
+
108
+ y_coords = y2.is_a?(Fixnum) || d[0] != "image" ? [y1, y2] : [y1]
109
+ x_coords = x2.is_a?(Fixnum) || d[0] != "image" ? [x1, x2] : [x1]
110
+
111
+ # add the row offset:
112
+ y1, y2 = y_coords.map { |y|
113
+ if y.is_a?(Fixnum)
114
+ row_offset + y
115
+ elsif y.is_a?(Hash) && y["off"].is_a?(Fixnum)
116
+ last_row + y["off"]
117
+ else
118
+ raise "bad offset #{y}"
119
+ end
120
+ }
121
+
122
+ # add the column offset:
123
+ x1, x2 = x_coords.map { |x|
124
+ raise "bad range point #{x}" unless x.is_a? Fixnum
125
+ column_offset + x
126
+ }
127
+
128
+ el[last_row] = [] unless
129
+ el[last_row] || ["border", "image"].member?(d[0])
130
+
131
+ case d[0]
132
+ when "conditional_formatting"
133
+ el[last_row] << [d[0], [x1, y1, x2, y2], d[2]]
134
+ when "merge"
135
+ el[last_row] << [d[0], [x1, y1, x2, y2]]
136
+ when "border"
137
+ el << [d[0], [x1, y1, x2, y2], d[2]]
138
+ when "image"
139
+ el << [d[0], [x1, y1, x2, y2, w, h], d[2]]
140
+ end
141
+ end
142
+
143
+ def position_borders(borders)
144
+ b_styles = []
145
+ borders.each do |b|
146
+ top_row, middle_row, bottom_row, edge_h = [], [], [], {}
147
+ br, range, defaults = b
148
+ col0, row0, colw, rowh = range
149
+
150
+ raise "wrong border range #{range}" if
151
+ col0 > colw || row0 > rowh || (col0 == colw && row0 == rowh)
152
+
153
+ defaults = self.class.symbolize_keys(defaults, ':')
154
+
155
+ boxborders = Hash.new do |hash, key|
156
+ hash[key] = {
157
+ border: defaults.merge( {edges: key.to_s.split('_').map(&:to_sym)} )
158
+ }
159
+ end
160
+
161
+ boxborders[:nil] = {}
162
+
163
+ tro, mro, bro =
164
+ top_row.object_id, middle_row.object_id, bottom_row.object_id
165
+
166
+ if col0 == colw
167
+ # vertical line
168
+ edge_h[tro], edge_h[mro], edge_h[bro] =
169
+ ["left"], ["left"], ["left"]
170
+ elsif row0 == rowh
171
+ # horizontal line
172
+ edge_h[tro], edge_h[mro], edge_h[bro] =
173
+ ["top"]*3, ["top"]*3, ["top"]*3
174
+ else
175
+ # box
176
+ edge_h[tro] = ["top_left", "top_right", "top"]
177
+ edge_h[mro] = ["left", "right", "nil"]
178
+ edge_h[bro] = ["bottom_left", "bottom_right", "bottom"]
179
+ end
180
+
181
+ [top_row, middle_row, bottom_row].each do |r|
182
+ if col0 == colw
183
+ r[col0] = boxborders[edge_h[r.object_id][0].to_sym]
184
+ else
185
+ (col0...colw).each do |counter|
186
+ a = (counter == col0) ? boxborders[edge_h[r.object_id][0].to_sym] : {}
187
+
188
+ # counter == col0 == (colw - 1) => merge the edges:
189
+ a = boxborders[edge_h[r.object_id][1].to_sym] =
190
+ merge_cell_edges(a, deep_copy(boxborders[edge_h[r.object_id][1].to_sym])) if
191
+ counter == (colw - 1)
192
+
193
+ a = boxborders[edge_h[r.object_id][2].to_sym] unless
194
+ counter == col0 || counter == (colw - 1)
195
+
196
+ r[counter] = a
197
+ end
198
+ end
199
+ end
200
+
201
+ if row0 == rowh
202
+ b_styles[row0] ||= []
203
+ bordered_cells(top_row, b_styles[row0])
204
+ else
205
+ (row0...rowh).each_with_index do |r, i|
206
+ b_styles[r] ||= []
207
+
208
+ a = i == 0 ? top_row : []
209
+
210
+ a = merge_row_edges(a,bottom_row) if
211
+ i == (rowh - row0 - 1)
212
+
213
+ a = middle_row unless
214
+ i == 0 || i == (rowh - row0 - 1)
215
+
216
+ bordered_cells(a, b_styles[r])
217
+ end
218
+ end
219
+ end
220
+
221
+ b_styles
222
+ end
223
+
224
+ def worksheet_rows(ws, rows, styles, row_styles, format, borders, images)
225
+ wsrows = []
226
+
227
+ b_styles = position_borders(borders)
228
+ rlenmax = rows.map { |el| el.kind_of?(Array) ? el.length : 0 }.max
229
+
230
+ rows.each_with_index do |r, index|
231
+ rlen = r.kind_of?(Array) ? r.length : 0
232
+
233
+ if rlen < rlenmax
234
+ rows[index] ||= []
235
+ rows[index] += [""] * (rlenmax-rlen)
236
+ end
237
+
238
+ row_styles[index] ||= {}
239
+
240
+ rsi = row_styles[index]
241
+
242
+ rsi["style"] = styles[index].kind_of?(Array) ? styles[index] : []
243
+
244
+ if b_styles[index].kind_of?(Array) && b_styles[index].count > 0
245
+ len = [rsi["style"].count, b_styles[index].count].max
246
+
247
+ len.times do |ind|
248
+ b_styles[index][ind] ||= {}
249
+ rsi["style"][ind] ||= {}
250
+
251
+ rsi["style"][ind] =
252
+ rsi["style"][ind].merge(b_styles[index][ind])
253
+ end
254
+
255
+ rsi["style"] = rsi["style"].map{ |x| x || {} }
256
+ end
257
+
258
+ wsrows << ["row", rows[index], rsi]
259
+
260
+ if format[index] && format[index].kind_of?(Array)
261
+ format[index].each do |f|
262
+ raise "wrong number of arguments for #{f[0]}" unless
263
+ [
264
+ ["conditional_formatting", 3],
265
+ ["merge", 2]
266
+ ].member?([f[0], f.length])
267
+
268
+ wsrows << f
269
+ end
270
+ end
271
+
272
+ end
273
+
274
+ apply_relative_worksheet_ops(ws, wsrows + images)
275
+ end
276
+
277
+ attr_reader :styles, :package
278
+
279
+ def initialize(worksheets)
280
+ @styles = {}
281
+ @package = Axlsx::Package.new
282
+ wb = package.workbook
283
+
284
+ # We got some sort of error if the worksheets is an array
285
+ if worksheets.is_a? Hash
286
+ ws = wb.add_worksheet(name: "EXCEPTION")
287
+ ws.add_row ["error", worksheets["error"]]
288
+ ws.add_row ["backtrace", worksheets["backtrace"]]
289
+ return
290
+ end
291
+
292
+ raise "expected worksheets array, got: #{worksheets}" unless
293
+ worksheets.is_a?(Array)
294
+
295
+ worksheets << ["No data", []] if worksheets.count == 0
296
+
297
+ worksheets.each { |opl|
298
+ name, ops, opts = opl
299
+
300
+ raise "bad worksheet name: #{name}" unless name.is_a?(String)
301
+ raise "bad worksheet ops: #{ops.inspect}" unless ops.is_a?(Array)
302
+ raise "bad options #{opts}" unless opts.is_a?(Hash) || opts.nil?
303
+
304
+ # Remove special characters and truncate sheet name due to Excel
305
+ # limitations. The following chars are not allowed: []*?:\/
306
+ name = name.gsub(/[\[\]\*\?\/\\]/, '_').gsub(':', '').truncate(31)
307
+
308
+ opts = self.class.symbolize_keys(opts || {}, ':')
309
+ widths = opts[:widths] || []
310
+ gridlines = opts[:gridlines] != 0
311
+
312
+ ws = wb.add_worksheet(name: name)
313
+
314
+ ws.column_widths(*widths) if widths.is_a?(Array) && widths.count > 0
315
+ ws.sheet_view.show_grid_lines = gridlines
316
+
317
+ apply_relative_worksheet_ops(ws, ops)
318
+ }
319
+ @package.use_shared_strings = true
320
+ end
321
+
322
+ def add_style(style)
323
+ raise "bad style" unless style.is_a?(Hash) || style.is_a?(Array)
324
+
325
+ if style.is_a?(Array)
326
+ style.map { |s|
327
+ styles[s] ||= package.workbook.styles.add_style(s)
328
+ }
329
+ else
330
+ styles[style] ||= package.workbook.styles.add_style(style)
331
+ end
332
+ end
333
+
334
+ def intern_range(ws, range)
335
+ return range if range.is_a? String
336
+ raise "bad range #{range}" unless range.is_a?(Array) && range.length==4
337
+ x1, y1, x2, y2 = range
338
+
339
+ y1, y2 = [y1, y2].map { |y|
340
+ next y unless y.is_a?(Hash)
341
+ raise "bad offset #{y}" unless y["off"].is_a?(Fixnum)
342
+ ws.rows.last.index + y["off"]
343
+ }
344
+
345
+ [x1, y1, x2, y2].each { |x|
346
+ raise "bad range point #{x}" unless x.is_a? Fixnum
347
+ }
348
+ Axlsx.cell_r(x1, y1) + ":" + Axlsx.cell_r(x2, y2)
349
+ end
350
+
351
+ def recalc_offsets(ops_pos)
352
+ new_ops1, new_ops2, new_ops = [], [], []
353
+ # precalculate the offsets of pos options embedded in another pos opt:
354
+ ops_pos.each { |d|
355
+ new_ops1 += d[2].select { |inner_ops|
356
+ inner_ops if inner_ops[0] == "pos"
357
+ }.map { |inner|
358
+ [inner[0], d[1].zip(inner[1]).map { |x,y| x+y }, inner[2] ]
359
+ }
360
+ }
361
+ # keep the offsets of non-pos options embedded in pos opt:
362
+ new_ops2 = ops_pos.map { |d|
363
+ [ d[0], d[1], d[2].select{|inner| inner if inner[0] != "pos" } ]
364
+ }
365
+ new_ops = new_ops1 + new_ops2
366
+ count = new_ops.select { |d|
367
+ d[2].select { |inner_ops|
368
+ inner_ops if inner_ops[0] == "pos"
369
+ }.count > 0
370
+ }.count
371
+
372
+ count == 0 ? new_ops.sort : recalc_offsets(new_ops)
373
+ end
374
+
375
+ def apply_relative_worksheet_ops(ws, ops)
376
+
377
+ non_pos = ops.select {|opl| opl[0] != "pos" }
378
+ ops_pos = ops.select {|opl| opl[0] == "pos" }
379
+ ops_brd = ops.select {|opl| opl[0] == "border" }
380
+
381
+ if (ops_pos.count > 0)
382
+ # Wrap all non-pos options in a pos option with offset 0, 0:
383
+ pos_00_ops = non_pos.count > 0 ? [ ["pos", [0, 0], non_pos] ] : []
384
+ # Recalculate the offsets of embedded pos opts:
385
+ ops = pos_00_ops + recalc_offsets(ops_pos)
386
+ elsif (ops_brd.count > 0)
387
+ # Wrap the non-pos options in a pos opt with offset 0, 0:
388
+ pos_00_ops = [ ["pos", [0, 0], non_pos] ]
389
+ ops = pos_00_ops
390
+ end
391
+
392
+ rows, styles, row_styles, format, borders, images = [], [], [], [], [], []
393
+ ops.each { |opl|
394
+ raise "bad op #{opl}" unless opl.length > 1
395
+ case opl[0]
396
+ when "pos"
397
+ op, offset, data = opl
398
+ raise "bad offset #{offset}" unless
399
+ offset.is_a?(Array) && offset.length == 2 &&
400
+ offset.all? {|x| x.is_a? Fixnum}
401
+ # column offset, row offset:
402
+ column_offset, row_offset = offset
403
+ r_number, last_row = row_offset, row_offset
404
+
405
+ data.each do |d|
406
+ raise "non array data #{d[1]}" unless d[1].is_a?(Array)
407
+ raise "non hash data options #{d[2]}" unless
408
+ [NilClass, Hash, String, Array].member? d[2].class
409
+ case d[0]
410
+ when "row"
411
+ position_row(d, column_offset, r_number, rows, styles, row_styles)
412
+ last_row = r_number
413
+ r_number += 1
414
+ when "conditional_formatting", "merge"
415
+ position_elem(d, offset, last_row, format)
416
+ when "border"
417
+ position_elem(d, offset, last_row, borders)
418
+ when "image"
419
+ position_elem(d, offset, last_row, images)
420
+ else
421
+ raise "unknown op #{d[0]} embedded in 'position' option"
422
+ end
423
+ end
424
+
425
+ when "row"
426
+ op, data, options = opl
427
+
428
+ raise "bad row op #{opl}" unless data.is_a?(Array) || opl.length > 3
429
+ raise "non hash options #{options} for row" unless
430
+ options.nil? || options.is_a?(Hash)
431
+
432
+ options = self.class.symbolize_keys(options || {}, ':')
433
+
434
+ options[:style] = add_style(options[:style]) if options[:style]
435
+
436
+ ws.add_row data, options
437
+
438
+ when "row_style"
439
+ op, row_num, style = opl
440
+
441
+ # FIXME: need to handle Array?
442
+ raise "non hash arg for row_style" unless style.is_a?(Hash)
443
+ raise "bad row num #{opl}" unless row_num.is_a?(Fixnum)
444
+
445
+ style = self.class.symbolize_keys(style, ':')
446
+
447
+ style_id = add_style(style)
448
+ row = ws.rows[row_num]
449
+
450
+ row.style = style_id
451
+
452
+ when "merge"
453
+ op, range = opl
454
+
455
+ raise "bad merge op #{opl}" unless opl.length == 2
456
+ range = intern_range(ws, range)
457
+
458
+ ws.merge_cells range
459
+ when "conditional_formatting"
460
+ op, range, format = opl
461
+
462
+ raise "non hash arg for format" unless format.is_a?(Hash)
463
+
464
+ range = intern_range(ws, range)
465
+
466
+ format = self.class.symbolize_keys(format, ':')
467
+
468
+ color_scale_a = format[:color_scale]
469
+
470
+ if color_scale_a
471
+ raise "color_scale must be an array" unless
472
+ color_scale_a.is_a?(Array)
473
+
474
+ raise "non-hash color_scale element" unless
475
+ color_scale_a.all? {|x| x.is_a?(Hash)}
476
+
477
+ format[:color_scale] = Axlsx::ColorScale.new(*color_scale_a)
478
+ end
479
+
480
+ dxfid = format[:dxfId]
481
+ format[:dxfId] = add_style(dxfid) if dxfid.is_a?(Hash)
482
+
483
+ ws.add_conditional_formatting(range, format)
484
+ when "image"
485
+ op, range, img = opl
486
+ raise "bad image params #{range}" unless
487
+ range.is_a?(Array) && range.length == 6
488
+ x1, y1, x2, y2, w, h = range
489
+ raise "bad image range, width or height #{range}" unless
490
+ [ x1, y1, w, h ].all? {|x| x.is_a? Fixnum}
491
+ ws.add_image(image_src: "#{Rails.public_path}/images/#{img}",
492
+ noSelect: true,
493
+ noMove: true) do |image|
494
+ image.width = w
495
+ image.height = h
496
+ image.start_at x1, y1
497
+ image.end_at x2, y2 if x2.is_a?(Fixnum) && y2.is_a?(Fixnum)
498
+ end
499
+ else
500
+ raise "unknown op #{opl[0]}"
501
+ end
502
+ }
503
+ worksheet_rows(ws,rows,styles,row_styles,format,borders,images) unless
504
+ [ops_pos.count, ops_brd.count].all?{ |a| a == 0 }
505
+ end
506
+
507
+ # recursive symbolize_keys. FIXME: this belongs in a generic
508
+ # library somewhere.
509
+ def self.symbolize_keys(obj, sym_str=nil)
510
+ case obj
511
+ when Array
512
+ obj.map {|x| symbolize_keys(x, sym_str)}
513
+ when Hash
514
+ obj.inject({}) { |result, (key, value)|
515
+ key = key.to_sym if key.is_a?(String)
516
+ result[key] = symbolize_keys(value, sym_str)
517
+ result
518
+ }
519
+ when String
520
+ (sym_str && obj.starts_with?(sym_str)) ? obj[sym_str.length..-1].to_sym : obj
521
+ else
522
+ obj
523
+ end
524
+ end
525
+
526
+ end