marty 0.5.12
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/MIT-LICENSE +20 -0
- data/README.md +84 -0
- data/Rakefile +29 -0
- data/app/assets/javascripts/marty/application.js +15 -0
- data/app/assets/stylesheets/marty/application.css +13 -0
- data/app/components/marty/api_auth_view.rb +32 -0
- data/app/components/marty/auth_app.rb +55 -0
- data/app/components/marty/auth_app.rb~ +51 -0
- data/app/components/marty/auth_app/javascripts/auth_app.js +91 -0
- data/app/components/marty/auth_app/javascripts/auth_app.js~ +91 -0
- data/app/components/marty/cm_form_panel.rb~ +5 -0
- data/app/components/marty/cm_grid_panel.rb~ +35 -0
- data/app/components/marty/data_import_view.rb~ +142 -0
- data/app/components/marty/extras/layout.rb +46 -0
- data/app/components/marty/extras/layout.rb~ +46 -0
- data/app/components/marty/extras/misc.rb +18 -0
- data/app/components/marty/form.rb +6 -0
- data/app/components/marty/grid.rb +45 -0
- data/app/components/marty/grid_append_only.rb +12 -0
- data/app/components/marty/import_type_view.rb +53 -0
- data/app/components/marty/live_search_grid_panel.rb +46 -0
- data/app/components/marty/live_search_grid_panel.rb~ +49 -0
- data/app/components/marty/main_auth_app.rb +269 -0
- data/app/components/marty/main_auth_app.rb~ +238 -0
- data/app/components/marty/mcfly_grid_panel.rb +62 -0
- data/app/components/marty/mcfly_grid_panel.rb~ +80 -0
- data/app/components/marty/new_posting_form.rb +46 -0
- data/app/components/marty/new_posting_form.rb~ +46 -0
- data/app/components/marty/new_posting_window.rb +21 -0
- data/app/components/marty/new_posting_window.rb~ +21 -0
- data/app/components/marty/panel.rb +12 -0
- data/app/components/marty/pivot_grid.rb +52 -0
- data/app/components/marty/pivot_grid/endpoints.rb +45 -0
- data/app/components/marty/pivot_grid/javascripts/extensions.js +150 -0
- data/app/components/marty/pivot_grid/javascripts/pivot_grid.js +86 -0
- data/app/components/marty/pivot_grid/services.rb +44 -0
- data/app/components/marty/posting_grid.rb +139 -0
- data/app/components/marty/posting_grid.rb~ +140 -0
- data/app/components/marty/posting_window.rb +27 -0
- data/app/components/marty/promise_view.rb +177 -0
- data/app/components/marty/promise_view.rb~ +157 -0
- data/app/components/marty/promise_view/stylesheets/promise_view.css +26 -0
- data/app/components/marty/promise_view/stylesheets/promise_view.css~ +15 -0
- data/app/components/marty/report_form.rb +225 -0
- data/app/components/marty/report_form.rb~ +217 -0
- data/app/components/marty/report_select.rb +145 -0
- data/app/components/marty/report_select.rb~ +133 -0
- data/app/components/marty/reporting.rb +39 -0
- data/app/components/marty/reporting.rb~ +39 -0
- data/app/components/marty/script_detail.rb~ +430 -0
- data/app/components/marty/script_form.rb +233 -0
- data/app/components/marty/script_form.rb~ +233 -0
- data/app/components/marty/script_form/javascripts/Ext.ux.form.field.CodeMirror.js +698 -0
- data/app/components/marty/script_form/javascripts/Ext.ux.form.field.CodeMirror.js~ +909 -0
- data/app/components/marty/script_form/javascripts/codemirror.js +3130 -0
- data/app/components/marty/script_form/javascripts/mode/clike/clike.js +284 -0
- data/app/components/marty/script_form/javascripts/mode/clike/index.html +102 -0
- data/app/components/marty/script_form/javascripts/mode/clike/scala.html +766 -0
- data/app/components/marty/script_form/javascripts/mode/clojure/clojure.js +206 -0
- data/app/components/marty/script_form/javascripts/mode/clojure/index.html +67 -0
- data/app/components/marty/script_form/javascripts/mode/coffeescript/LICENSE +22 -0
- data/app/components/marty/script_form/javascripts/mode/coffeescript/coffeescript.js +346 -0
- data/app/components/marty/script_form/javascripts/mode/coffeescript/index.html +728 -0
- data/app/components/marty/script_form/javascripts/mode/commonlisp/commonlisp.js +101 -0
- data/app/components/marty/script_form/javascripts/mode/commonlisp/index.html +165 -0
- data/app/components/marty/script_form/javascripts/mode/css/css.js +448 -0
- data/app/components/marty/script_form/javascripts/mode/css/index.html +58 -0
- data/app/components/marty/script_form/javascripts/mode/css/test.js +501 -0
- data/app/components/marty/script_form/javascripts/mode/delorean/delorean.js +189 -0
- data/app/components/marty/script_form/javascripts/mode/diff/diff.js +32 -0
- data/app/components/marty/script_form/javascripts/mode/diff/index.html +105 -0
- data/app/components/marty/script_form/javascripts/mode/ecl/ecl.js +203 -0
- data/app/components/marty/script_form/javascripts/mode/ecl/index.html +42 -0
- data/app/components/marty/script_form/javascripts/mode/erlang/erlang.js +463 -0
- data/app/components/marty/script_form/javascripts/mode/erlang/index.html +63 -0
- data/app/components/marty/script_form/javascripts/mode/gfm/gfm.js +150 -0
- data/app/components/marty/script_form/javascripts/mode/gfm/index.html +48 -0
- data/app/components/marty/script_form/javascripts/mode/go/go.js +170 -0
- data/app/components/marty/script_form/javascripts/mode/go/index.html +73 -0
- data/app/components/marty/script_form/javascripts/mode/groovy/groovy.js +210 -0
- data/app/components/marty/script_form/javascripts/mode/groovy/index.html +72 -0
- data/app/components/marty/script_form/javascripts/mode/haskell/haskell.js +242 -0
- data/app/components/marty/script_form/javascripts/mode/haskell/index.html +61 -0
- data/app/components/marty/script_form/javascripts/mode/haxe/haxe.js +429 -0
- data/app/components/marty/script_form/javascripts/mode/haxe/index.html +91 -0
- data/app/components/marty/script_form/javascripts/mode/htmlembedded/htmlembedded.js +72 -0
- data/app/components/marty/script_form/javascripts/mode/htmlembedded/index.html +50 -0
- data/app/components/marty/script_form/javascripts/mode/htmlmixed/htmlmixed.js +84 -0
- data/app/components/marty/script_form/javascripts/mode/htmlmixed/index.html +52 -0
- data/app/components/marty/script_form/javascripts/mode/javascript/index.html +78 -0
- data/app/components/marty/script_form/javascripts/mode/javascript/javascript.js +361 -0
- data/app/components/marty/script_form/javascripts/mode/jinja2/index.html +38 -0
- data/app/components/marty/script_form/javascripts/mode/jinja2/jinja2.js +42 -0
- data/app/components/marty/script_form/javascripts/mode/less/index.html +740 -0
- data/app/components/marty/script_form/javascripts/mode/less/less.js +266 -0
- data/app/components/marty/script_form/javascripts/mode/lua/index.html +73 -0
- data/app/components/marty/script_form/javascripts/mode/lua/lua.js +140 -0
- data/app/components/marty/script_form/javascripts/mode/markdown/index.html +343 -0
- data/app/components/marty/script_form/javascripts/mode/markdown/markdown.js +382 -0
- data/app/components/marty/script_form/javascripts/mode/markdown/test.js +1084 -0
- data/app/components/marty/script_form/javascripts/mode/mysql/index.html +42 -0
- data/app/components/marty/script_form/javascripts/mode/mysql/mysql.js +186 -0
- data/app/components/marty/script_form/javascripts/mode/ntriples/index.html +33 -0
- data/app/components/marty/script_form/javascripts/mode/ntriples/ntriples.js +172 -0
- data/app/components/marty/script_form/javascripts/mode/ocaml/index.html +130 -0
- data/app/components/marty/script_form/javascripts/mode/ocaml/ocaml.js +114 -0
- data/app/components/marty/script_form/javascripts/mode/pascal/LICENSE +7 -0
- data/app/components/marty/script_form/javascripts/mode/pascal/index.html +49 -0
- data/app/components/marty/script_form/javascripts/mode/pascal/pascal.js +94 -0
- data/app/components/marty/script_form/javascripts/mode/perl/LICENSE +19 -0
- data/app/components/marty/script_form/javascripts/mode/perl/index.html +63 -0
- data/app/components/marty/script_form/javascripts/mode/perl/perl.js +816 -0
- data/app/components/marty/script_form/javascripts/mode/php/index.html +49 -0
- data/app/components/marty/script_form/javascripts/mode/php/php.js +148 -0
- data/app/components/marty/script_form/javascripts/mode/pig/index.html +43 -0
- data/app/components/marty/script_form/javascripts/mode/pig/pig.js +172 -0
- data/app/components/marty/script_form/javascripts/mode/plsql/index.html +63 -0
- data/app/components/marty/script_form/javascripts/mode/plsql/plsql.js +217 -0
- data/app/components/marty/script_form/javascripts/mode/properties/index.html +41 -0
- data/app/components/marty/script_form/javascripts/mode/properties/properties.js +63 -0
- data/app/components/marty/script_form/javascripts/mode/python/LICENSE.txt +21 -0
- data/app/components/marty/script_form/javascripts/mode/python/index.html +123 -0
- data/app/components/marty/script_form/javascripts/mode/python/python.js +338 -0
- data/app/components/marty/script_form/javascripts/mode/r/LICENSE +24 -0
- data/app/components/marty/script_form/javascripts/mode/r/index.html +74 -0
- data/app/components/marty/script_form/javascripts/mode/r/r.js +141 -0
- data/app/components/marty/script_form/javascripts/mode/rpm/changes/changes.js +19 -0
- data/app/components/marty/script_form/javascripts/mode/rpm/changes/index.html +54 -0
- data/app/components/marty/script_form/javascripts/mode/rpm/spec/index.html +100 -0
- data/app/components/marty/script_form/javascripts/mode/rpm/spec/spec.css +5 -0
- data/app/components/marty/script_form/javascripts/mode/rpm/spec/spec.js +66 -0
- data/app/components/marty/script_form/javascripts/mode/rst/index.html +526 -0
- data/app/components/marty/script_form/javascripts/mode/rst/rst.js +326 -0
- data/app/components/marty/script_form/javascripts/mode/ruby/LICENSE +24 -0
- data/app/components/marty/script_form/javascripts/mode/ruby/index.html +172 -0
- data/app/components/marty/script_form/javascripts/mode/ruby/ruby.js +195 -0
- data/app/components/marty/script_form/javascripts/mode/rust/index.html +49 -0
- data/app/components/marty/script_form/javascripts/mode/rust/rust.js +432 -0
- data/app/components/marty/script_form/javascripts/mode/scheme/index.html +65 -0
- data/app/components/marty/script_form/javascripts/mode/scheme/scheme.js +230 -0
- data/app/components/marty/script_form/javascripts/mode/shell/index.html +50 -0
- data/app/components/marty/script_form/javascripts/mode/shell/shell.js +118 -0
- data/app/components/marty/script_form/javascripts/mode/sieve/LICENSE +23 -0
- data/app/components/marty/script_form/javascripts/mode/sieve/index.html +81 -0
- data/app/components/marty/script_form/javascripts/mode/sieve/sieve.js +156 -0
- data/app/components/marty/script_form/javascripts/mode/smalltalk/index.html +56 -0
- data/app/components/marty/script_form/javascripts/mode/smalltalk/smalltalk.js +139 -0
- data/app/components/marty/script_form/javascripts/mode/smarty/index.html +83 -0
- data/app/components/marty/script_form/javascripts/mode/smarty/smarty.js +148 -0
- data/app/components/marty/script_form/javascripts/mode/sparql/index.html +41 -0
- data/app/components/marty/script_form/javascripts/mode/sparql/sparql.js +143 -0
- data/app/components/marty/script_form/javascripts/mode/stex/index.html +98 -0
- data/app/components/marty/script_form/javascripts/mode/stex/stex.js +182 -0
- data/app/components/marty/script_form/javascripts/mode/stex/test.js +343 -0
- data/app/components/marty/script_form/javascripts/mode/tiddlywiki/index.html +141 -0
- data/app/components/marty/script_form/javascripts/mode/tiddlywiki/tiddlywiki.css +14 -0
- data/app/components/marty/script_form/javascripts/mode/tiddlywiki/tiddlywiki.js +384 -0
- data/app/components/marty/script_form/javascripts/mode/tiki/index.html +83 -0
- data/app/components/marty/script_form/javascripts/mode/tiki/tiki.css +26 -0
- data/app/components/marty/script_form/javascripts/mode/tiki/tiki.js +309 -0
- data/app/components/marty/script_form/javascripts/mode/vb/LICENSE.txt +21 -0
- data/app/components/marty/script_form/javascripts/mode/vb/index.html +89 -0
- data/app/components/marty/script_form/javascripts/mode/vb/vb.js +260 -0
- data/app/components/marty/script_form/javascripts/mode/vbscript/index.html +43 -0
- data/app/components/marty/script_form/javascripts/mode/vbscript/vbscript.js +26 -0
- data/app/components/marty/script_form/javascripts/mode/velocity/index.html +104 -0
- data/app/components/marty/script_form/javascripts/mode/velocity/velocity.js +146 -0
- data/app/components/marty/script_form/javascripts/mode/verilog/index.html +211 -0
- data/app/components/marty/script_form/javascripts/mode/verilog/verilog.js +194 -0
- data/app/components/marty/script_form/javascripts/mode/xml/index.html +45 -0
- data/app/components/marty/script_form/javascripts/mode/xml/xml.js +318 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/LICENSE +20 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/index.html +223 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/test/index.html +27 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/test/testBase.js +42 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/test/testEmptySequenceKeyword.js +16 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/test/testMultiAttr.js +16 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/test/testNamespaces.js +91 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/test/testProcessingInstructions.js +16 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/test/testQuotes.js +19 -0
- data/app/components/marty/script_form/javascripts/mode/xquery/xquery.js +451 -0
- data/app/components/marty/script_form/javascripts/mode/yaml/index.html +68 -0
- data/app/components/marty/script_form/javascripts/mode/yaml/yaml.js +95 -0
- data/app/components/marty/script_form/javascripts/util/closetag.js +164 -0
- data/app/components/marty/script_form/javascripts/util/dialog.css +27 -0
- data/app/components/marty/script_form/javascripts/util/dialog.js +70 -0
- data/app/components/marty/script_form/javascripts/util/foldcode.js +196 -0
- data/app/components/marty/script_form/javascripts/util/formatting.js +193 -0
- data/app/components/marty/script_form/javascripts/util/javascript-hint.js +134 -0
- data/app/components/marty/script_form/javascripts/util/loadmode.js +51 -0
- data/app/components/marty/script_form/javascripts/util/match-highlighter.js +44 -0
- data/app/components/marty/script_form/javascripts/util/multiplex.js +77 -0
- data/app/components/marty/script_form/javascripts/util/overlay.js +54 -0
- data/app/components/marty/script_form/javascripts/util/pig-hint.js +123 -0
- data/app/components/marty/script_form/javascripts/util/runmode-standalone.js +90 -0
- data/app/components/marty/script_form/javascripts/util/runmode.js +53 -0
- data/app/components/marty/script_form/javascripts/util/search.js +118 -0
- data/app/components/marty/script_form/javascripts/util/searchcursor.js +119 -0
- data/app/components/marty/script_form/javascripts/util/simple-hint.css +16 -0
- data/app/components/marty/script_form/javascripts/util/simple-hint.js +97 -0
- data/app/components/marty/script_form/javascripts/util/xml-hint.js +137 -0
- data/app/components/marty/script_form/stylesheets/codemirror.css +172 -0
- data/app/components/marty/script_form/stylesheets/delorean.css +10 -0
- data/app/components/marty/script_form/stylesheets/theme/ambiance.css +81 -0
- data/app/components/marty/script_form/stylesheets/theme/blackboard.css +25 -0
- data/app/components/marty/script_form/stylesheets/theme/cobalt.css +18 -0
- data/app/components/marty/script_form/stylesheets/theme/eclipse.css +25 -0
- data/app/components/marty/script_form/stylesheets/theme/elegant.css +10 -0
- data/app/components/marty/script_form/stylesheets/theme/erlang-dark.css +21 -0
- data/app/components/marty/script_form/stylesheets/theme/lesser-dark.css +44 -0
- data/app/components/marty/script_form/stylesheets/theme/monokai.css +28 -0
- data/app/components/marty/script_form/stylesheets/theme/neat.css +9 -0
- data/app/components/marty/script_form/stylesheets/theme/night.css +21 -0
- data/app/components/marty/script_form/stylesheets/theme/rubyblue.css +21 -0
- data/app/components/marty/script_form/stylesheets/theme/vibrant-ink.css +27 -0
- data/app/components/marty/script_form/stylesheets/theme/xq-dark.css +46 -0
- data/app/components/marty/script_grid.rb +104 -0
- data/app/components/marty/script_grid.rb~ +99 -0
- data/app/components/marty/script_tester.rb +114 -0
- data/app/components/marty/script_tester.rb~ +213 -0
- data/app/components/marty/scripting.rb +132 -0
- data/app/components/marty/scripting.rb~ +124 -0
- data/app/components/marty/select_report.rb~ +143 -0
- data/app/components/marty/simple_app.rb +97 -0
- data/app/components/marty/simple_app.rb~ +101 -0
- data/app/components/marty/simple_app/javascripts/simple_app.js +50 -0
- data/app/components/marty/simple_app/javascripts/statusbar_ext.js +8 -0
- data/app/components/marty/tag_grid.rb +83 -0
- data/app/components/marty/tag_grid.rb~ +89 -0
- data/app/components/marty/tree_panel.rb~ +256 -0
- data/app/components/marty/tree_panel/javascripts/tree_panel.js~ +317 -0
- data/app/components/marty/user_pivot.rb +128 -0
- data/app/components/marty/user_view.rb +181 -0
- data/app/components/marty/user_view.rb~ +188 -0
- data/app/controllers/marty/application_controller.rb +124 -0
- data/app/controllers/marty/application_controller.rb~ +133 -0
- data/app/controllers/marty/components_controller.rb +41 -0
- data/app/controllers/marty/components_controller.rb~ +37 -0
- data/app/controllers/marty/job_controller.rb +28 -0
- data/app/controllers/marty/job_controller.rb~ +28 -0
- data/app/controllers/marty/rpc_controller.rb +64 -0
- data/app/controllers/marty/rpc_controller.rb~ +61 -0
- data/app/helpers/marty/application_helper.rb +4 -0
- data/app/helpers/marty/script_set.rb +57 -0
- data/app/helpers/marty/script_set.rb~ +59 -0
- data/app/models/marty/api_auth.rb +44 -0
- data/app/models/marty/api_auth.rb~ +48 -0
- data/app/models/marty/base.rb +4 -0
- data/app/models/marty/data_change.rb +179 -0
- data/app/models/marty/data_change.rb~ +141 -0
- data/app/models/marty/enum.rb +22 -0
- data/app/models/marty/enum.rb~ +16 -0
- data/app/models/marty/import_type.rb +44 -0
- data/app/models/marty/import_type.rb~ +48 -0
- data/app/models/marty/poop.rb~ +169 -0
- data/app/models/marty/posting.rb +101 -0
- data/app/models/marty/posting.rb~ +86 -0
- data/app/models/marty/posting_type.rb +12 -0
- data/app/models/marty/posting_type.rb~ +21 -0
- data/app/models/marty/promise.rb +252 -0
- data/app/models/marty/promise.rb~ +196 -0
- data/app/models/marty/role.rb +6 -0
- data/app/models/marty/role.rb~ +10 -0
- data/app/models/marty/script.rb +144 -0
- data/app/models/marty/script.rb~ +62 -0
- data/app/models/marty/tag.rb +96 -0
- data/app/models/marty/tag.rb~ +91 -0
- data/app/models/marty/token.rb +30 -0
- data/app/models/marty/user.rb +146 -0
- data/app/models/marty/user.rb~ +148 -0
- data/app/models/marty/user_role.rb +7 -0
- data/app/models/marty/user_role.rb~ +13 -0
- data/app/views/layouts/marty/application.html.erb +12 -0
- data/app/views/layouts/marty/application.html.erb~ +11 -0
- data/config/locales/en.yml +134 -0
- data/config/routes.rb +6 -0
- data/config/routes.rb~ +10 -0
- data/db/migrate/001_create_marty_scripts.rb +14 -0
- data/db/migrate/003_create_marty_users.rb +12 -0
- data/db/migrate/004_create_marty_roles.rb +7 -0
- data/db/migrate/005_create_marty_user_roles.rb +14 -0
- data/db/migrate/006_create_marty_tokens.rb +14 -0
- data/db/migrate/008_create_marty_posting_types.rb +7 -0
- data/db/migrate/019_create_marty_postings.rb +18 -0
- data/db/migrate/019_create_marty_postings.rb~ +19 -0
- data/db/migrate/068_create_marty_import_types.rb +12 -0
- data/db/migrate/069_create_marty_import_synonyms.rb +15 -0
- data/db/migrate/070_create_versions.rb +18 -0
- data/db/migrate/071_add_object_changes_column_to_versions.rb +9 -0
- data/db/migrate/072_add_validation_function_to_import_types.rb +6 -0
- data/db/migrate/073_add_preprocess_function_to_import_types.rb +5 -0
- data/db/migrate/090_create_delayed_jobs.rb +22 -0
- data/db/migrate/091_create_marty_promises.rb +36 -0
- data/db/migrate/095_create_marty_tags.rb +14 -0
- data/db/migrate/095_create_marty_tags.rb~ +19 -0
- data/db/migrate/096_add_user_roles_to_import_types.rb +11 -0
- data/db/migrate/097_drop_versions.rb +9 -0
- data/db/migrate/098_create_marty_api_auths.rb +20 -0
- data/db/seeds.rb +48 -0
- data/lib/marty.rb +18 -0
- data/lib/marty.rb~ +13 -0
- data/lib/marty/content_handler.rb +97 -0
- data/lib/marty/content_handler.rb~ +93 -0
- data/lib/marty/data_conversion.rb +298 -0
- data/lib/marty/data_exporter.rb +150 -0
- data/lib/marty/data_exporter.rb~ +137 -0
- data/lib/marty/data_importer.rb +122 -0
- data/lib/marty/data_importer.rb~ +114 -0
- data/lib/marty/data_row_processor.rb~ +206 -0
- data/lib/marty/drop_folder_hook.rb~ +17 -0
- data/lib/marty/engine.rb +10 -0
- data/lib/marty/folder_hook.rb~ +9 -0
- data/lib/marty/lazy_column_loader.rb +57 -0
- data/lib/marty/lazy_column_loader.rb~ +47 -0
- data/lib/marty/mcfly_query.rb +189 -0
- data/lib/marty/mcfly_query.rb~ +188 -0
- data/lib/marty/migrations.rb +108 -0
- data/lib/marty/migrations.rb~ +65 -0
- data/lib/marty/monkey.rb +163 -0
- data/lib/marty/monkey.rb~ +160 -0
- data/lib/marty/permissions.rb +64 -0
- data/lib/marty/permissions.rb~ +69 -0
- data/lib/marty/promise.rb~ +41 -0
- data/lib/marty/promise_job.rb +123 -0
- data/lib/marty/promise_job.rb~ +121 -0
- data/lib/marty/promise_proxy.rb +94 -0
- data/lib/marty/promise_proxy.rb~ +69 -0
- data/lib/marty/railtie.rb +5 -0
- data/lib/marty/relation.rb +39 -0
- data/lib/marty/util.rb +110 -0
- data/lib/marty/util.rb~ +80 -0
- data/lib/marty/version.rb +3 -0
- data/lib/marty/version.rb~ +3 -0
- data/lib/marty/xl.rb +527 -0
- data/lib/marty/xl.rb~ +526 -0
- data/lib/pyxll/README.txt +19 -0
- data/lib/pyxll/README.txt~ +16 -0
- data/lib/pyxll/gemini.py +155 -0
- data/lib/pyxll/gemini.py~ +110 -0
- data/lib/pyxll/pyxll.cfg +12 -0
- data/lib/pyxll/pyxll.cfg~ +12 -0
- data/lib/pyxll/sample.xlsx +0 -0
- data/lib/tasks/marty_tasks.rake +37 -0
- 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
|