refinerycms 0.9.6.19 → 0.9.6.21
Sign up to get free protection for your applications and to get access to all the features.
- data/.gems +0 -8
- data/.gitignore +3 -1
- data/Rakefile +1 -5
- data/VERSION +1 -1
- data/bin/refinery-update-core +11 -2
- data/config/application.rb +6 -1
- data/config/environments/development.rb +10 -6
- data/config/settings.rb +3 -0
- data/db/migrate/20100305023036_change_users_columns_for_authlogic.rb +46 -0
- data/db/migrate/20100305023037_remove_unused_users_columns.rb +16 -0
- data/db/migrate/20100312155331_make_user_perishable_token_nullable.rb +13 -0
- data/db/migrate/20100312160327_make_user_persistence_token_nullable.rb +13 -0
- data/db/migrate/20100315203301_remove_state_from_users.rb +9 -0
- data/db/schema.rb +8 -21
- data/db/seeds.rb +5 -19
- data/lib/refinery/tasks/refinery.rb +1 -1
- data/public/javascripts/admin.js +6 -2
- data/public/javascripts/jquery-ui-1.8.min.js +374 -0
- data/public/javascripts/jquery.js +606 -444
- data/public/javascripts/refinery/admin.js +30 -9
- data/public/javascripts/refinery/boot_wym.js +4 -0
- data/public/javascripts/wymeditor/jquery.refinery.wymeditor.js +3 -4
- data/public/stylesheets/refinery/refinery.css +11 -0
- data/public/stylesheets/wymeditor/skins/refinery/skin.css +3 -1
- data/public/stylesheets/wymeditor/skins/refinery/wymiframe.css +5 -5
- data/readme.md +13 -10
- data/test/fixtures/users.yml +5 -5
- data/test/functional/dashboard_controller_test.rb +0 -4
- data/test/functional/images_controller_test.rb +0 -3
- data/test/test_helper.rb +7 -3
- data/themes/hemingway/views/layouts/application.html.erb +1 -12
- data/vendor/plugins/authentication/app/controllers/admin/users_controller.rb +1 -3
- data/vendor/plugins/authentication/app/controllers/sessions_controller.rb +3 -14
- data/vendor/plugins/authentication/app/controllers/users_controller.rb +26 -37
- data/vendor/plugins/authentication/app/models/user.rb +29 -112
- data/vendor/plugins/authentication/app/models/user_mailer.rb +8 -5
- data/vendor/plugins/authentication/app/models/user_session.rb +9 -0
- data/vendor/plugins/authentication/app/views/admin/users/index.html.erb +1 -1
- data/vendor/plugins/authentication/app/views/sessions/new.html.erb +5 -4
- data/vendor/plugins/authentication/lib/authenticated_system.rb +40 -30
- data/vendor/plugins/authentication/rails/init.rb +9 -0
- data/vendor/plugins/authlogic/CHANGELOG.rdoc +345 -0
- data/vendor/plugins/authlogic/LICENSE +20 -0
- data/vendor/plugins/authlogic/README.rdoc +246 -0
- data/vendor/plugins/authlogic/Rakefile +42 -0
- data/vendor/plugins/authlogic/VERSION.yml +5 -0
- data/vendor/plugins/authlogic/authlogic.gemspec +217 -0
- data/vendor/plugins/authlogic/generators/session/session_generator.rb +9 -0
- data/vendor/plugins/authlogic/generators/session/templates/session.rb +2 -0
- data/vendor/plugins/authlogic/init.rb +1 -0
- data/vendor/plugins/authlogic/lib/authlogic.rb +64 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/base.rb +107 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/email.rb +110 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/logged_in_status.rb +60 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/login.rb +141 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/magic_columns.rb +24 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/password.rb +355 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/perishable_token.rb +105 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/persistence_token.rb +68 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/restful_authentication.rb +61 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/session_maintenance.rb +139 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/single_access_token.rb +65 -0
- data/vendor/plugins/authlogic/lib/authlogic/acts_as_authentic/validations_scope.rb +32 -0
- data/vendor/plugins/authlogic/lib/authlogic/authenticates_many/association.rb +42 -0
- data/vendor/plugins/authlogic/lib/authlogic/authenticates_many/base.rb +55 -0
- data/vendor/plugins/authlogic/lib/authlogic/controller_adapters/abstract_adapter.rb +67 -0
- data/vendor/plugins/authlogic/lib/authlogic/controller_adapters/merb_adapter.rb +30 -0
- data/vendor/plugins/authlogic/lib/authlogic/controller_adapters/rails_adapter.rb +48 -0
- data/vendor/plugins/authlogic/lib/authlogic/controller_adapters/sinatra_adapter.rb +61 -0
- data/vendor/plugins/authlogic/lib/authlogic/crypto_providers/aes256.rb +43 -0
- data/vendor/plugins/authlogic/lib/authlogic/crypto_providers/bcrypt.rb +90 -0
- data/vendor/plugins/authlogic/lib/authlogic/crypto_providers/md5.rb +34 -0
- data/vendor/plugins/authlogic/lib/authlogic/crypto_providers/sha1.rb +35 -0
- data/vendor/plugins/authlogic/lib/authlogic/crypto_providers/sha256.rb +50 -0
- data/vendor/plugins/authlogic/lib/authlogic/crypto_providers/sha512.rb +50 -0
- data/vendor/plugins/authlogic/lib/authlogic/crypto_providers/wordpress.rb +43 -0
- data/vendor/plugins/authlogic/lib/authlogic/i18n.rb +83 -0
- data/vendor/plugins/authlogic/lib/authlogic/i18n/translator.rb +15 -0
- data/vendor/plugins/authlogic/lib/authlogic/random.rb +33 -0
- data/vendor/plugins/authlogic/lib/authlogic/regex.rb +25 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/activation.rb +58 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/active_record_trickery.rb +64 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/base.rb +37 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/brute_force_protection.rb +96 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/callbacks.rb +99 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/cookies.rb +130 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/existence.rb +93 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/foundation.rb +63 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/http_auth.rb +58 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/id.rb +41 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/klass.rb +78 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/magic_columns.rb +95 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/magic_states.rb +59 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/params.rb +101 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/password.rb +240 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/perishable_token.rb +18 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/persistence.rb +70 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/priority_record.rb +34 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/scopes.rb +101 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/session.rb +62 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/timeout.rb +82 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/unauthorized_record.rb +50 -0
- data/vendor/plugins/authlogic/lib/authlogic/session/validation.rb +82 -0
- data/vendor/plugins/authlogic/lib/authlogic/test_case.rb +120 -0
- data/vendor/plugins/authlogic/lib/authlogic/test_case/mock_controller.rb +45 -0
- data/vendor/plugins/authlogic/lib/authlogic/test_case/mock_cookie_jar.rb +14 -0
- data/vendor/plugins/authlogic/lib/authlogic/test_case/mock_logger.rb +10 -0
- data/vendor/plugins/authlogic/lib/authlogic/test_case/mock_request.rb +19 -0
- data/vendor/plugins/authlogic/lib/authlogic/test_case/rails_request_adapter.rb +30 -0
- data/vendor/plugins/authlogic/rails/init.rb +1 -0
- data/vendor/plugins/authlogic/shoulda_macros/authlogic.rb +69 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/base_test.rb +18 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/email_test.rb +97 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/logged_in_status_test.rb +36 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/login_test.rb +109 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/magic_columns_test.rb +27 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/password_test.rb +236 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/perishable_token_test.rb +90 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/persistence_token_test.rb +55 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/restful_authentication_test.rb +40 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/session_maintenance_test.rb +84 -0
- data/vendor/plugins/authlogic/test/acts_as_authentic_test/single_access_test.rb +44 -0
- data/vendor/plugins/authlogic/test/authenticates_many_test.rb +16 -0
- data/vendor/plugins/authlogic/test/crypto_provider_test/aes256_test.rb +14 -0
- data/vendor/plugins/authlogic/test/crypto_provider_test/bcrypt_test.rb +14 -0
- data/vendor/plugins/authlogic/test/crypto_provider_test/sha1_test.rb +23 -0
- data/vendor/plugins/authlogic/test/crypto_provider_test/sha256_test.rb +14 -0
- data/vendor/plugins/authlogic/test/crypto_provider_test/sha512_test.rb +14 -0
- data/vendor/plugins/authlogic/test/fixtures/companies.yml +5 -0
- data/vendor/plugins/authlogic/test/fixtures/employees.yml +17 -0
- data/vendor/plugins/authlogic/test/fixtures/projects.yml +3 -0
- data/vendor/plugins/authlogic/test/fixtures/users.yml +24 -0
- data/vendor/plugins/authlogic/test/i18n_test.rb +33 -0
- data/vendor/plugins/authlogic/test/libs/affiliate.rb +7 -0
- data/vendor/plugins/authlogic/test/libs/company.rb +6 -0
- data/vendor/plugins/authlogic/test/libs/employee.rb +7 -0
- data/vendor/plugins/authlogic/test/libs/employee_session.rb +2 -0
- data/vendor/plugins/authlogic/test/libs/ldaper.rb +3 -0
- data/vendor/plugins/authlogic/test/libs/ordered_hash.rb +9 -0
- data/vendor/plugins/authlogic/test/libs/project.rb +3 -0
- data/vendor/plugins/authlogic/test/libs/user.rb +5 -0
- data/vendor/plugins/authlogic/test/libs/user_session.rb +6 -0
- data/vendor/plugins/authlogic/test/random_test.rb +49 -0
- data/vendor/plugins/authlogic/test/session_test/activation_test.rb +43 -0
- data/vendor/plugins/authlogic/test/session_test/active_record_trickery_test.rb +36 -0
- data/vendor/plugins/authlogic/test/session_test/brute_force_protection_test.rb +101 -0
- data/vendor/plugins/authlogic/test/session_test/callbacks_test.rb +6 -0
- data/vendor/plugins/authlogic/test/session_test/cookies_test.rb +112 -0
- data/vendor/plugins/authlogic/test/session_test/credentials_test.rb +0 -0
- data/vendor/plugins/authlogic/test/session_test/existence_test.rb +64 -0
- data/vendor/plugins/authlogic/test/session_test/http_auth_test.rb +28 -0
- data/vendor/plugins/authlogic/test/session_test/id_test.rb +17 -0
- data/vendor/plugins/authlogic/test/session_test/klass_test.rb +40 -0
- data/vendor/plugins/authlogic/test/session_test/magic_columns_test.rb +62 -0
- data/vendor/plugins/authlogic/test/session_test/magic_states_test.rb +60 -0
- data/vendor/plugins/authlogic/test/session_test/params_test.rb +53 -0
- data/vendor/plugins/authlogic/test/session_test/password_test.rb +106 -0
- data/vendor/plugins/authlogic/test/session_test/perishability_test.rb +15 -0
- data/vendor/plugins/authlogic/test/session_test/persistence_test.rb +21 -0
- data/vendor/plugins/authlogic/test/session_test/scopes_test.rb +60 -0
- data/vendor/plugins/authlogic/test/session_test/session_test.rb +59 -0
- data/vendor/plugins/authlogic/test/session_test/timeout_test.rb +52 -0
- data/vendor/plugins/authlogic/test/session_test/unauthorized_record_test.rb +13 -0
- data/vendor/plugins/authlogic/test/session_test/validation_test.rb +23 -0
- data/vendor/plugins/authlogic/test/test_helper.rb +182 -0
- data/vendor/plugins/dashboard/app/helpers/admin/dashboard_helper.rb +1 -1
- data/vendor/plugins/friendly_id/Changelog.md +243 -0
- data/vendor/plugins/friendly_id/Contributors.md +30 -0
- data/vendor/plugins/friendly_id/Gemfile +2 -0
- data/vendor/plugins/friendly_id/Guide.md +509 -0
- data/vendor/plugins/friendly_id/LICENSE +19 -0
- data/vendor/plugins/friendly_id/README.md +76 -0
- data/vendor/plugins/friendly_id/Rakefile +68 -0
- data/vendor/plugins/friendly_id/extras/README.txt +3 -0
- data/vendor/plugins/friendly_id/extras/bench.rb +59 -0
- data/vendor/plugins/friendly_id/extras/extras.rb +31 -0
- data/vendor/plugins/friendly_id/extras/prof.rb +14 -0
- data/vendor/plugins/friendly_id/extras/template-gem.rb +26 -0
- data/vendor/plugins/friendly_id/extras/template-plugin.rb +28 -0
- data/vendor/plugins/friendly_id/friendly_id.gemspec +40 -0
- data/vendor/plugins/friendly_id/generators/friendly_id/friendly_id_generator.rb +28 -0
- data/vendor/plugins/friendly_id/generators/friendly_id/templates/create_slugs.rb +18 -0
- data/vendor/plugins/friendly_id/lib/friendly_id.rb +67 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/active_record2.rb +47 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/active_record2/configuration.rb +66 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/active_record2/finders.rb +140 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/active_record2/simple_model.rb +162 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/active_record2/slug.rb +111 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/active_record2/slugged_model.rb +323 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/active_record2/tasks.rb +66 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/active_record2/tasks/friendly_id.rake +19 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/configuration.rb +132 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/finders.rb +106 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/slug_string.rb +292 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/slugged.rb +91 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/status.rb +35 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/test.rb +175 -0
- data/vendor/plugins/friendly_id/lib/friendly_id/version.rb +8 -0
- data/vendor/plugins/friendly_id/rails/init.rb +2 -0
- data/vendor/plugins/friendly_id/test/active_record2/basic_slugged_model_test.rb +14 -0
- data/vendor/plugins/friendly_id/test/active_record2/cached_slug_test.rb +61 -0
- data/vendor/plugins/friendly_id/test/active_record2/core.rb +93 -0
- data/vendor/plugins/friendly_id/test/active_record2/custom_normalizer_test.rb +20 -0
- data/vendor/plugins/friendly_id/test/active_record2/custom_table_name_test.rb +22 -0
- data/vendor/plugins/friendly_id/test/active_record2/deprecated_test.rb +23 -0
- data/vendor/plugins/friendly_id/test/active_record2/scoped_model_test.rb +111 -0
- data/vendor/plugins/friendly_id/test/active_record2/simple_test.rb +59 -0
- data/vendor/plugins/friendly_id/test/active_record2/slug_test.rb +34 -0
- data/vendor/plugins/friendly_id/test/active_record2/slugged.rb +30 -0
- data/vendor/plugins/friendly_id/test/active_record2/slugged_status_test.rb +61 -0
- data/vendor/plugins/friendly_id/test/active_record2/sti_test.rb +22 -0
- data/vendor/plugins/friendly_id/test/active_record2/support/database.mysql.yml +4 -0
- data/vendor/plugins/friendly_id/test/active_record2/support/database.postgres.yml +6 -0
- data/vendor/plugins/friendly_id/test/active_record2/support/database.sqlite3.yml +2 -0
- data/vendor/plugins/friendly_id/test/active_record2/support/models.rb +78 -0
- data/vendor/plugins/friendly_id/test/active_record2/tasks_test.rb +82 -0
- data/vendor/plugins/friendly_id/test/active_record2/test_helper.rb +114 -0
- data/vendor/plugins/friendly_id/test/friendly_id_test.rb +60 -0
- data/vendor/plugins/friendly_id/test/slug_string_test.rb +78 -0
- data/vendor/plugins/friendly_id/test/test_helper.rb +9 -0
- data/vendor/plugins/hpricot/CHANGELOG +88 -0
- data/vendor/plugins/hpricot/COPYING +18 -0
- data/vendor/plugins/hpricot/README +275 -0
- data/vendor/plugins/hpricot/Rakefile +272 -0
- data/vendor/plugins/hpricot/ext/fast_xs/FastXsService.java +1030 -0
- data/vendor/plugins/hpricot/ext/fast_xs/extconf.rb +4 -0
- data/vendor/plugins/hpricot/ext/fast_xs/fast_xs.c +201 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/HpricotCss.java +831 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/HpricotScanService.java +2086 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/MANIFEST +0 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/extconf.rb +6 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/hpricot_common.rl +76 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/hpricot_css.c +3503 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/hpricot_css.java.rl +155 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/hpricot_css.rl +115 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/hpricot_scan.c +6927 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/hpricot_scan.h +79 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/hpricot_scan.java.rl +1152 -0
- data/vendor/plugins/hpricot/ext/hpricot_scan/hpricot_scan.rl +788 -0
- data/vendor/plugins/hpricot/extras/mingw-rbconfig.rb +176 -0
- data/vendor/plugins/hpricot/hpricot.gemspec +17 -0
- data/vendor/plugins/hpricot/lib/hpricot.rb +26 -0
- data/vendor/plugins/hpricot/lib/hpricot/blankslate.rb +63 -0
- data/vendor/plugins/hpricot/lib/hpricot/builder.rb +216 -0
- data/vendor/plugins/hpricot/lib/hpricot/elements.rb +510 -0
- data/vendor/plugins/hpricot/lib/hpricot/htmlinfo.rb +691 -0
- data/vendor/plugins/hpricot/lib/hpricot/inspect.rb +103 -0
- data/vendor/plugins/hpricot/lib/hpricot/modules.rb +40 -0
- data/vendor/plugins/hpricot/lib/hpricot/parse.rb +38 -0
- data/vendor/plugins/hpricot/lib/hpricot/tag.rb +219 -0
- data/vendor/plugins/hpricot/lib/hpricot/tags.rb +164 -0
- data/vendor/plugins/hpricot/lib/hpricot/traverse.rb +839 -0
- data/vendor/plugins/hpricot/lib/hpricot/xchar.rb +94 -0
- data/vendor/plugins/hpricot/setup.rb +1585 -0
- data/vendor/plugins/hpricot/test/files/basic.xhtml +17 -0
- data/vendor/plugins/hpricot/test/files/boingboing.html +2266 -0
- data/vendor/plugins/hpricot/test/files/cy0.html +3653 -0
- data/vendor/plugins/hpricot/test/files/immob.html +400 -0
- data/vendor/plugins/hpricot/test/files/pace_application.html +1320 -0
- data/vendor/plugins/hpricot/test/files/tenderlove.html +16 -0
- data/vendor/plugins/hpricot/test/files/uswebgen.html +220 -0
- data/vendor/plugins/hpricot/test/files/utf8.html +1054 -0
- data/vendor/plugins/hpricot/test/files/week9.html +1723 -0
- data/vendor/plugins/hpricot/test/files/why.xml +19 -0
- data/vendor/plugins/hpricot/test/load_files.rb +7 -0
- data/vendor/plugins/hpricot/test/nokogiri-bench.rb +64 -0
- data/vendor/plugins/hpricot/test/test_alter.rb +96 -0
- data/vendor/plugins/hpricot/test/test_builder.rb +37 -0
- data/vendor/plugins/hpricot/test/test_parser.rb +428 -0
- data/vendor/plugins/hpricot/test/test_paths.rb +25 -0
- data/vendor/plugins/hpricot/test/test_preserved.rb +88 -0
- data/vendor/plugins/hpricot/test/test_xml.rb +28 -0
- data/vendor/plugins/inquiries/app/controllers/admin/inquiries_controller.rb +1 -6
- data/vendor/plugins/inquiries/app/helpers/inquiries_helper.rb +0 -11
- data/vendor/plugins/inquiries/app/views/admin/inquiries/_inquiry.html.erb +10 -6
- data/vendor/plugins/inquiries/app/views/inquiry_mailer/confirmation.html.erb +1 -1
- data/vendor/plugins/inquiries/config/routes.rb +1 -1
- data/vendor/plugins/pages/app/controllers/admin/pages_controller.rb +16 -3
- data/vendor/plugins/pages/app/models/page_part.rb +9 -0
- data/vendor/plugins/rails_indexes/lib/indexer.rb +0 -2
- data/vendor/plugins/refinery/app/views/admin/_head.html.erb +2 -1
- data/vendor/plugins/refinery/app/views/shared/_head.html.erb +0 -1
- data/vendor/plugins/refinery/app/views/shared/_ie6check.html.erb +23 -11
- data/vendor/plugins/refinery/app/views/welcome.html.erb +1 -1
- data/vendor/plugins/refinery/app/views/wymiframe.html.erb +2 -1
- data/vendor/plugins/refinery/lib/generators/refinery/USAGE +1 -1
- data/vendor/plugins/refinery/lib/generators/refinery/templates/migration.rb +2 -2
- data/vendor/plugins/refinery/lib/generators/refinery/templates/model.rb +4 -0
- data/vendor/plugins/refinery/lib/generators/refinery/templates/views/admin/_form.html.erb +6 -0
- data/vendor/plugins/refinery/lib/generators/refinery/templates/views/show.html.erb +5 -3
- data/vendor/plugins/refinery/lib/refinery/application_helper.rb +2 -2
- data/vendor/plugins/refinery/lib/refinery/initializer.rb +4 -41
- data/vendor/plugins/refinery_settings/app/models/refinery_setting.rb +7 -3
- data/vendor/plugins/slim_scrooge/README.textile +112 -0
- data/vendor/plugins/slim_scrooge/Rakefile +29 -0
- data/vendor/plugins/slim_scrooge/VERSION.yml +5 -0
- data/vendor/plugins/slim_scrooge/ext/Rakefile +42 -0
- data/vendor/plugins/slim_scrooge/ext/extconf.rb +5 -0
- data/vendor/plugins/slim_scrooge/lib/slim_scrooge.rb +16 -0
- data/vendor/plugins/slim_scrooge/lib/slim_scrooge/callsite.rb +96 -0
- data/vendor/plugins/slim_scrooge/lib/slim_scrooge/callsites.rb +70 -0
- data/vendor/plugins/slim_scrooge/lib/slim_scrooge/monitored_hash.rb +103 -0
- data/vendor/plugins/slim_scrooge/lib/slim_scrooge/result_set.rb +38 -0
- data/vendor/plugins/slim_scrooge/lib/slim_scrooge/simple_set.rb +34 -0
- data/vendor/plugins/slim_scrooge/lib/slim_scrooge/slim_scrooge.rb +46 -0
- data/vendor/plugins/slim_scrooge/rails/init.rb +8 -0
- data/vendor/plugins/slim_scrooge/slim_scrooge.gemspec +62 -0
- data/vendor/plugins/slim_scrooge/slim_scrooge_windows.gemspec +59 -0
- data/vendor/plugins/slim_scrooge/test/active_record_setup.rb +3 -0
- data/vendor/plugins/slim_scrooge/test/helper.rb +91 -0
- data/vendor/plugins/slim_scrooge/test/models/course.rb +2 -0
- data/vendor/plugins/slim_scrooge/test/schema/schema.rb +5 -0
- data/vendor/plugins/slim_scrooge/test/setup.rb +5 -0
- data/vendor/plugins/themes/app/models/theme.rb +7 -0
- data/vendor/plugins/themes/lib/theme_server.rb +3 -1
- data/vendor/plugins/themes/rails/init.rb +16 -13
- data/vendor/plugins/will_paginate/.manifest +43 -0
- data/vendor/plugins/will_paginate/CHANGELOG.rdoc +139 -0
- data/vendor/plugins/will_paginate/LICENSE +18 -0
- data/vendor/plugins/will_paginate/README.rdoc +107 -0
- data/vendor/plugins/will_paginate/Rakefile +53 -0
- data/vendor/plugins/will_paginate/examples/apple-circle.gif +0 -0
- data/vendor/plugins/will_paginate/examples/index.haml +69 -0
- data/vendor/plugins/will_paginate/examples/index.html +92 -0
- data/vendor/plugins/will_paginate/examples/pagination.css +90 -0
- data/vendor/plugins/will_paginate/examples/pagination.sass +91 -0
- data/vendor/plugins/will_paginate/init.rb +1 -0
- data/vendor/plugins/will_paginate/lib/will_paginate.rb +90 -0
- data/vendor/plugins/will_paginate/lib/will_paginate/array.rb +16 -0
- data/vendor/plugins/will_paginate/lib/will_paginate/collection.rb +146 -0
- data/vendor/plugins/will_paginate/lib/will_paginate/core_ext.rb +43 -0
- data/vendor/plugins/will_paginate/lib/will_paginate/finder.rb +264 -0
- data/vendor/plugins/will_paginate/lib/will_paginate/named_scope.rb +170 -0
- data/vendor/plugins/will_paginate/lib/will_paginate/named_scope_patch.rb +37 -0
- data/vendor/plugins/will_paginate/lib/will_paginate/version.rb +9 -0
- data/vendor/plugins/will_paginate/lib/will_paginate/view_helpers.rb +408 -0
- data/vendor/plugins/will_paginate/test/boot.rb +21 -0
- data/vendor/plugins/will_paginate/test/collection_test.rb +143 -0
- data/vendor/plugins/will_paginate/test/console +8 -0
- data/vendor/plugins/will_paginate/test/database.yml +22 -0
- data/vendor/plugins/will_paginate/test/finder_test.rb +473 -0
- data/vendor/plugins/will_paginate/test/fixtures/admin.rb +3 -0
- data/vendor/plugins/will_paginate/test/fixtures/developer.rb +14 -0
- data/vendor/plugins/will_paginate/test/fixtures/developers_projects.yml +13 -0
- data/vendor/plugins/will_paginate/test/fixtures/project.rb +15 -0
- data/vendor/plugins/will_paginate/test/fixtures/projects.yml +6 -0
- data/vendor/plugins/will_paginate/test/fixtures/replies.yml +29 -0
- data/vendor/plugins/will_paginate/test/fixtures/reply.rb +7 -0
- data/vendor/plugins/will_paginate/test/fixtures/schema.rb +38 -0
- data/vendor/plugins/will_paginate/test/fixtures/topic.rb +10 -0
- data/vendor/plugins/will_paginate/test/fixtures/topics.yml +30 -0
- data/vendor/plugins/will_paginate/test/fixtures/user.rb +2 -0
- data/vendor/plugins/will_paginate/test/fixtures/users.yml +35 -0
- data/vendor/plugins/will_paginate/test/helper.rb +37 -0
- data/vendor/plugins/will_paginate/test/lib/activerecord_test_case.rb +43 -0
- data/vendor/plugins/will_paginate/test/lib/activerecord_test_connector.rb +75 -0
- data/vendor/plugins/will_paginate/test/lib/load_fixtures.rb +11 -0
- data/vendor/plugins/will_paginate/test/lib/view_test_process.rb +179 -0
- data/vendor/plugins/will_paginate/test/tasks.rake +59 -0
- data/vendor/plugins/will_paginate/test/view_test.rb +373 -0
- data/vendor/plugins/will_paginate/will_paginate.gemspec +19 -0
- metadata +308 -27
- data/db/migrate/20091029034951_remove_blurb_from_news_items.rb +0 -9
- data/public/javascripts/jquery-ui-1.8rc1.min.js +0 -375
- data/test/fixtures/news_items.yml +0 -14
- data/test/unit/news_items_test.rb +0 -33
- data/vendor/plugins/authentication/test/fixtures/users.yml +0 -19
- data/vendor/plugins/authentication/test/functional/sessions_controller_test.rb +0 -85
- data/vendor/plugins/authentication/test/functional/users_controller_test.rb +0 -99
- data/vendor/plugins/authentication/test/unit/user_test.rb +0 -164
- data/vendor/plugins/news/app/controllers/admin/news_items_controller.rb +0 -5
- data/vendor/plugins/news/app/controllers/news_items_controller.rb +0 -20
- data/vendor/plugins/news/app/models/news_item.rb +0 -24
- data/vendor/plugins/news/app/views/admin/news_items/_form.html.erb +0 -19
- data/vendor/plugins/news/app/views/admin/news_items/_news_item.html.erb +0 -15
- data/vendor/plugins/news/app/views/admin/news_items/edit.html.erb +0 -1
- data/vendor/plugins/news/app/views/admin/news_items/index.html.erb +0 -31
- data/vendor/plugins/news/app/views/admin/news_items/new.html.erb +0 -1
- data/vendor/plugins/news/app/views/news_items/_recent_posts.html.erb +0 -8
- data/vendor/plugins/news/app/views/news_items/index.html.erb +0 -17
- data/vendor/plugins/news/app/views/news_items/index.rss.builder +0 -26
- data/vendor/plugins/news/app/views/news_items/show.html.erb +0 -13
- data/vendor/plugins/news/config/routes.rb +0 -7
- data/vendor/plugins/news/news.md +0 -20
- data/vendor/plugins/news/rails/init.rb +0 -11
@@ -8,12 +8,16 @@ class RefinerySetting < ActiveRecord::Base
|
|
8
8
|
cache_write(object.name, object.value)
|
9
9
|
end
|
10
10
|
|
11
|
+
def self.cache_key(name)
|
12
|
+
"refinery_setting_#{name}"
|
13
|
+
end
|
14
|
+
|
11
15
|
def self.cache_write(name, value)
|
12
|
-
Rails.cache.write(
|
16
|
+
Rails.cache.write(cache_key(name), value)
|
13
17
|
end
|
14
18
|
|
15
19
|
def self.cache_read(name)
|
16
|
-
Rails.cache.read(
|
20
|
+
Rails.cache.read(cache_key(name))
|
17
21
|
end
|
18
22
|
|
19
23
|
# Number of settings to show per page when using will_paginate
|
@@ -75,7 +79,7 @@ class RefinerySetting < ActiveRecord::Base
|
|
75
79
|
def self.[]=(name, value)
|
76
80
|
setting = find_or_create_by_name(name.to_s)
|
77
81
|
setting.value = value
|
78
|
-
setting.save
|
82
|
+
setting.save
|
79
83
|
end
|
80
84
|
|
81
85
|
# Below is not very nice, but seems to be required
|
@@ -0,0 +1,112 @@
|
|
1
|
+
h1. SlimScrooge - serious optimisation for activerecord
|
2
|
+
|
3
|
+
h2. What is it?
|
4
|
+
|
5
|
+
It's an optimization layer to ensure your application only fetches the database content needed to minimize wire traffic, excessive SQL queries and reduce conversion overheads to native Ruby types.
|
6
|
+
|
7
|
+
SlimScrooge implements inline query optimisation, automatically restricting the columns fetched based on what was used during previous passes through the same part of your code.
|
8
|
+
|
9
|
+
SlimScrooge is similar to (and is partly derived from) "Scrooge":http://github.com/methodmissing/scrooge but has many fewer lines of code and is faster.
|
10
|
+
|
11
|
+
h2. Benchmark
|
12
|
+
|
13
|
+
SlimScrooge performs best when the database is not on the same machine as your rails app. In this case the overhead of fetching unnecessary data from the database can become more important.
|
14
|
+
|
15
|
+
I ran a benchmark that consisted of fetching 400 real urls (culled from the log file) from our complex web app. In this test I found a consistent *12% improvement* in performance over plain active record. Not earth-shattering, but worthwhile.
|
16
|
+
|
17
|
+
Note that this result was for comparing the time taken for running complete requests through rails - of which database accesses are only one part. So the result is better than it at first sounds.
|
18
|
+
|
19
|
+
In future releases I expect further gains.
|
20
|
+
|
21
|
+
h2. Installation
|
22
|
+
|
23
|
+
Requirements: Rails 2.2 or above, Ruby 1.8.6 or above.
|
24
|
+
|
25
|
+
h3. Gem
|
26
|
+
|
27
|
+
<pre>
|
28
|
+
# if you haven't already, add gemcutter to your gem sources
|
29
|
+
sudo gem install gemcutter
|
30
|
+
gem tumble
|
31
|
+
# install slim_scrooge
|
32
|
+
sudo gem install slim_scrooge
|
33
|
+
</pre>
|
34
|
+
|
35
|
+
Note that the C extension will only build in MRI Ruby 1.8 on non-windows at present, but do not worry, for other platforms the backup callsite mechanism is used.
|
36
|
+
|
37
|
+
Next add this to your Rails::Initializer section in environment.rb:
|
38
|
+
|
39
|
+
<pre>
|
40
|
+
config.gem 'slim_scrooge'
|
41
|
+
</pre>
|
42
|
+
|
43
|
+
h3. Plugin
|
44
|
+
|
45
|
+
<pre>
|
46
|
+
script/plugin install git://github.com/sdsykes/slim_scrooge.git
|
47
|
+
</pre>
|
48
|
+
|
49
|
+
h2. What it does
|
50
|
+
|
51
|
+
<pre>
|
52
|
+
# 1st request, sql is unchanged but columns accesses are recorded
|
53
|
+
Brochure Load SlimScrooged 1st time (27.1ms) SELECT * FROM `brochures` WHERE (expires_at IS NULL)
|
54
|
+
|
55
|
+
# 2nd request, only fetch columns that were used the first time
|
56
|
+
Brochure Load SlimScrooged (4.5ms) SELECT `brochures`.expires_at,`brochures`.operator_id,`brochures`.id FROM
|
57
|
+
`brochures` WHERE (expires_at IS NULL)
|
58
|
+
|
59
|
+
# 2nd request, later in code we need another column which causes a reload of all remaining columns
|
60
|
+
Brochure Reload SlimScrooged (0.6ms) `brochures`.name,`brochures`.comment,`brochures`.image_height,`brochures`.id,
|
61
|
+
`brochures`.tel,`brochures`.long_comment,`brochures`.image_name,`brochures`.image_width FROM
|
62
|
+
`brochures` WHERE `brochures`.id IN ('5646','5476','4562','3456','4567','7355')
|
63
|
+
|
64
|
+
# 3rd request
|
65
|
+
Brochure Load SlimScrooged (4.5ms) SELECT `brochures`.expires_at,`brochures`.operator_id,`brochures`.name,
|
66
|
+
`brochures`.id FROM `brochures` WHERE (expires_at IS NULL)
|
67
|
+
</pre>
|
68
|
+
|
69
|
+
h2. Technical discussion
|
70
|
+
|
71
|
+
SlimScrooge hooks in at just one particular place in ActiveRecord - and that place is the find_all_by_sql method. All select queries pass through this method.
|
72
|
+
|
73
|
+
SlimScrooge is able to record each call (and where it came from in your code), and to modify queries that do SELECT * FROM en-route to the database so that they only select the rows that are actually used by that piece of code.
|
74
|
+
|
75
|
+
How does SlimScrooge know which columns are actually used?
|
76
|
+
|
77
|
+
It tracks them using a monitored hash - the hash is a subclass of Hash that has is instantiated with a proc for its default value. In this proc we handle any attributes that were not fetched from the DB - fetching the remaining columns as needed.
|
78
|
+
|
79
|
+
In fact for efficiency, column accesses are only recorded when they are for newly accessed columns. Columns we already know about are accessed completely normally, and there is no additional overhead.
|
80
|
+
|
81
|
+
h2. Caveats
|
82
|
+
|
83
|
+
* It is possible to delete an object and then to try to use its attributes to access another object that perhaps must be also deleted (like the option :dependent=>:destroy in Rails associations).
|
84
|
+
|
85
|
+
This situation is likely to be found rarely because SlimScrooge particularly checks for columns used by :dependent=>:destroy, but if you are doing something similar manually in your code then you may run into problems. Your attempt to access the key of the dependent object could cause a reload if the column is not already noted by SlimScrooge, and the reload will fail if you have already destroyed the parent object.
|
86
|
+
|
87
|
+
* Some users have complained that running their test suite is slower when SlimScrooge is enabled. This is expected - it's doing a little more work for each query, and most queries in tests are only executed once.
|
88
|
+
|
89
|
+
If it proves to be a problem, you can try this in environment.rb:
|
90
|
+
|
91
|
+
<pre>
|
92
|
+
config.gem 'slim_scrooge' unless Rails.env.test?
|
93
|
+
</pre>
|
94
|
+
|
95
|
+
I do recommend you test with SlimScrooge enabled at some point though, to make sure that everything is working as expected.
|
96
|
+
|
97
|
+
h2. Tests
|
98
|
+
|
99
|
+
SlimScrooge performs the full activerecord test suite without errors, except for a couple of tests that check the number of queries performed.
|
100
|
+
|
101
|
+
To run the tests you need to set your database up with appropriate access for the rails user, and make activerecord_unittest and activerecord_unittest2 databases. Then run:
|
102
|
+
|
103
|
+
<pre>
|
104
|
+
rake test_with_active_record
|
105
|
+
</pre>
|
106
|
+
|
107
|
+
h2. References
|
108
|
+
* "Scrooge":http://github.com/methodmissing/scrooge
|
109
|
+
|
110
|
+
h2. Authors
|
111
|
+
* Stephen Sykes (sdsykes)
|
112
|
+
* Special thanks to Lourens Naudé (methodmissing) for the original idea, and the C implementation of callsite_hash as well as some other bits of code that I borrowed from the original project.
|
@@ -0,0 +1,29 @@
|
|
1
|
+
require 'rake'
|
2
|
+
require 'rake/testtask'
|
3
|
+
require 'test/helper'
|
4
|
+
|
5
|
+
Rake::TestTask.new(:test_with_active_record) do |t|
|
6
|
+
t.libs << SlimScrooge::ActiveRecordTest::AR_TEST_SUITE
|
7
|
+
t.libs << SlimScrooge::ActiveRecordTest.connection
|
8
|
+
t.test_files = SlimScrooge::ActiveRecordTest.test_files
|
9
|
+
t.ruby_opts = ["-r #{File.join(File.dirname(__FILE__), 'test', 'active_record_setup')}"]
|
10
|
+
t.verbose = true
|
11
|
+
end
|
12
|
+
|
13
|
+
begin
|
14
|
+
require 'jeweler'
|
15
|
+
Jeweler::Tasks.new do |s|
|
16
|
+
s.name = "slim_scrooge"
|
17
|
+
s.summary = "Slim_scrooge - serious optimisation for ActiveRecord"
|
18
|
+
s.email = "sdsykes@gmail.com"
|
19
|
+
s.homepage = "http://github.com/sdsykes/slim_scrooge"
|
20
|
+
s.description = "Slim scrooge boosts speed in Rails ActiveRecord Models by only querying the database for what is needed."
|
21
|
+
s.authors = ["Stephen Sykes"]
|
22
|
+
s.files = FileList["[A-Z]*", "{ext,lib,test}/**/*"]
|
23
|
+
s.extensions = "ext/Rakefile"
|
24
|
+
end
|
25
|
+
Jeweler::GemcutterTasks.new
|
26
|
+
rescue LoadError
|
27
|
+
puts "Jeweler not available. Install it with: sudo gem install technicalpickles-jeweler -s http://
|
28
|
+
gems.github.com"
|
29
|
+
end
|
@@ -0,0 +1,42 @@
|
|
1
|
+
task :default=>[:build]
|
2
|
+
|
3
|
+
task :build do
|
4
|
+
# only try to install extension on non windows / MRI 1.8
|
5
|
+
unless /mswin/ =~ RUBY_PLATFORM || RUBY_VERSION >= "1.9" || defined?(RUBY_ENGINE) && RUBY_ENGINE != "ruby"
|
6
|
+
require 'rbconfig'
|
7
|
+
|
8
|
+
ruby = File.join(Config::CONFIG["bindir"],
|
9
|
+
Config::CONFIG["RUBY_INSTALL_NAME"])
|
10
|
+
ruby << Config::CONFIG["EXEEXT"]
|
11
|
+
|
12
|
+
# escape string in case path to ruby executable contain spaces.
|
13
|
+
ruby.sub!(/.*\s.*/m, '"\&"')
|
14
|
+
|
15
|
+
results = `#{ruby} extconf.rb`
|
16
|
+
|
17
|
+
unless File.exist? 'Makefile'
|
18
|
+
raise "Makefile not found:\n\n#{results.join "\n"}"
|
19
|
+
end
|
20
|
+
|
21
|
+
dest_path = ENV["RUBYARCHDIR"] || File.join(File.dirname(__FILE__), "..", "lib")
|
22
|
+
|
23
|
+
mf = File.read('Makefile')
|
24
|
+
mf = mf.gsub(/^RUBYARCHDIR\s*=\s*\$[^$]*/, "RUBYARCHDIR = #{dest_path}")
|
25
|
+
mf = mf.gsub(/^RUBYLIBDIR\s*=\s*\$[^$]*/, "RUBYLIBDIR = #{dest_path}")
|
26
|
+
|
27
|
+
File.open('Makefile', 'wb') {|f| f.print mf}
|
28
|
+
|
29
|
+
make_program = ENV['make']
|
30
|
+
unless make_program
|
31
|
+
make_program = (/mswin/ =~ RUBY_PLATFORM) ? 'nmake' : 'make'
|
32
|
+
end
|
33
|
+
|
34
|
+
['', ' install'].each do |target|
|
35
|
+
cmd = "#{make_program}#{target}"
|
36
|
+
results << cmd + "\n"
|
37
|
+
results << `#{cmd}`
|
38
|
+
|
39
|
+
raise "make#{target} failed:\n\n#{results}" unless $?.success?
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
@@ -0,0 +1,16 @@
|
|
1
|
+
# Author: Stephen Sykes
|
2
|
+
begin
|
3
|
+
unless File.exists?(File.join(File.dirname(__FILE__), "../", "ext", "Makefile"))
|
4
|
+
Dir.chdir(File.join(File.dirname(__FILE__), "../", "ext"))
|
5
|
+
`rake`
|
6
|
+
end
|
7
|
+
rescue Exception
|
8
|
+
end
|
9
|
+
|
10
|
+
begin; require 'callsite_hash'; rescue LoadError; end
|
11
|
+
require 'slim_scrooge/simple_set'
|
12
|
+
require 'slim_scrooge/callsites'
|
13
|
+
require 'slim_scrooge/callsite'
|
14
|
+
require 'slim_scrooge/result_set'
|
15
|
+
require 'slim_scrooge/monitored_hash'
|
16
|
+
require 'slim_scrooge/slim_scrooge'
|
@@ -0,0 +1,96 @@
|
|
1
|
+
# Author: Stephen Sykes
|
2
|
+
|
3
|
+
module SlimScrooge
|
4
|
+
# A Callsite contains the list of columns that are accessed when an SQL
|
5
|
+
# query is made from a particular place in the app
|
6
|
+
#
|
7
|
+
class Callsite
|
8
|
+
ScroogeComma = ",".freeze
|
9
|
+
ScroogeRegexJoin = /(?:LEFT|INNER|OUTER|CROSS)*\s*(?:STRAIGHT_JOIN|JOIN)/i
|
10
|
+
|
11
|
+
attr_accessor :seen_columns
|
12
|
+
attr_reader :columns_hash, :primary_key, :model_class_name
|
13
|
+
|
14
|
+
class << self
|
15
|
+
# Make a callsite if the query is of the right type for us to optimise
|
16
|
+
#
|
17
|
+
def make_callsite(model_class, original_sql)
|
18
|
+
if use_scrooge?(model_class, original_sql)
|
19
|
+
new(model_class)
|
20
|
+
else
|
21
|
+
nil
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
# Check if query can be optimised
|
26
|
+
#
|
27
|
+
def use_scrooge?(model_class, original_sql)
|
28
|
+
original_sql =~ select_regexp(model_class.table_name) &&
|
29
|
+
model_class.columns_hash.has_key?(model_class.primary_key) &&
|
30
|
+
original_sql !~ ScroogeRegexJoin
|
31
|
+
end
|
32
|
+
|
33
|
+
# The regexp that enables us to replace the * from SELECT * with
|
34
|
+
# the list of columns we actually need
|
35
|
+
#
|
36
|
+
def select_regexp(table_name)
|
37
|
+
%r{SELECT (`?(?:#{table_name})?`?.?\\*) FROM}
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def initialize(model_class)
|
42
|
+
@all_columns = SimpleSet.new(model_class.column_names)
|
43
|
+
@model_class_name = model_class.to_s
|
44
|
+
@quoted_table_name = model_class.quoted_table_name
|
45
|
+
@primary_key = model_class.primary_key
|
46
|
+
@quoted_primary_key = model_class.connection.quote_column_name(@primary_key)
|
47
|
+
@columns_hash = model_class.columns_hash
|
48
|
+
@select_regexp = self.class.select_regexp(model_class.table_name)
|
49
|
+
@seen_columns = SimpleSet.new(essential_columns(model_class))
|
50
|
+
end
|
51
|
+
|
52
|
+
# List of columns that should always be fetched no matter what
|
53
|
+
#
|
54
|
+
def essential_columns(model_class)
|
55
|
+
model_class.reflect_on_all_associations.inject([@primary_key]) do |arr, assoc|
|
56
|
+
if assoc.options[:dependent] && assoc.macro == :belongs_to
|
57
|
+
arr << assoc.association_foreign_key
|
58
|
+
end
|
59
|
+
arr
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
# Returns suitable sql given a list of columns and the original query
|
64
|
+
#
|
65
|
+
def scrooged_sql(seen_columns, sql)
|
66
|
+
sql.gsub(@select_regexp, "SELECT #{scrooge_select_sql(seen_columns)} FROM")
|
67
|
+
end
|
68
|
+
|
69
|
+
# List if columns what were not fetched
|
70
|
+
#
|
71
|
+
def missing_columns(fetched_columns)
|
72
|
+
(@all_columns - SimpleSet.new(fetched_columns)) << @primary_key
|
73
|
+
end
|
74
|
+
|
75
|
+
# Returns sql for fetching the unfetched columns for all the rows
|
76
|
+
# in the result set, specified by primary_keys
|
77
|
+
#
|
78
|
+
def reload_sql(primary_keys, fetched_columns)
|
79
|
+
sql_keys = primary_keys.collect{|pk| "'#{pk}'"}.join(ScroogeComma)
|
80
|
+
cols = scrooge_select_sql(missing_columns(fetched_columns))
|
81
|
+
"SELECT #{cols} FROM #{@quoted_table_name} WHERE #{@quoted_primary_key} IN (#{sql_keys})"
|
82
|
+
end
|
83
|
+
|
84
|
+
def connection
|
85
|
+
@model_class_name.constantize.connection
|
86
|
+
end
|
87
|
+
|
88
|
+
# Change a set of columns into a correctly quoted comma separated list
|
89
|
+
#
|
90
|
+
def scrooge_select_sql(set)
|
91
|
+
set.collect do |name|
|
92
|
+
"#{@quoted_table_name}.#{connection.quote_column_name(name)}"
|
93
|
+
end.join(ScroogeComma)
|
94
|
+
end
|
95
|
+
end
|
96
|
+
end
|
@@ -0,0 +1,70 @@
|
|
1
|
+
# Author: Stephen Sykes
|
2
|
+
|
3
|
+
module SlimScrooge
|
4
|
+
# Contains the complete list of callsites
|
5
|
+
#
|
6
|
+
class Callsites
|
7
|
+
CallsitesMutex = Mutex.new
|
8
|
+
@@callsites = {}
|
9
|
+
ScroogeCallsiteSample = 1..16
|
10
|
+
|
11
|
+
class << self
|
12
|
+
# Whether we have encountered a callsite before
|
13
|
+
#
|
14
|
+
def has_key?(callsite_key)
|
15
|
+
@@callsites.has_key?(callsite_key)
|
16
|
+
end
|
17
|
+
|
18
|
+
# Return the callsite for this key
|
19
|
+
#
|
20
|
+
def [](callsite_key)
|
21
|
+
@@callsites[callsite_key]
|
22
|
+
end
|
23
|
+
|
24
|
+
# Fallback if you can't compile extension for some reason
|
25
|
+
#
|
26
|
+
begin
|
27
|
+
callsite_hash
|
28
|
+
rescue NameError
|
29
|
+
def callsite_hash
|
30
|
+
caller[ScroogeCallsiteSample].hash
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
# Generate a key string - uses the portion of the query before the WHERE
|
35
|
+
# together with the callsite_hash generated by callsite_hash.c
|
36
|
+
#
|
37
|
+
def callsite_key(sql)
|
38
|
+
callsite_hash + sql.gsub(/\sWHERE.*/i, "").hash
|
39
|
+
end
|
40
|
+
|
41
|
+
# Create a new callsite
|
42
|
+
#
|
43
|
+
def create(sql, callsite_key, name)
|
44
|
+
begin
|
45
|
+
model_class = name.split.first.constantize
|
46
|
+
rescue NameError, NoMethodError
|
47
|
+
add_callsite(callsite_key, nil)
|
48
|
+
else
|
49
|
+
add_callsite(callsite_key, Callsite.make_callsite(model_class, sql))
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
# Add a new callsite, wrap in a mutex for safety
|
54
|
+
#
|
55
|
+
def add_callsite(callsite_key, callsite)
|
56
|
+
CallsitesMutex.synchronize do
|
57
|
+
@@callsites[callsite_key] = callsite
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
# Record that a column was accessed, wrap in a mutex for safety
|
62
|
+
#
|
63
|
+
def add_seen_column(callsite, seen_column)
|
64
|
+
CallsitesMutex.synchronize do
|
65
|
+
callsite.seen_columns << seen_column
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|
70
|
+
end
|
@@ -0,0 +1,103 @@
|
|
1
|
+
# Author: Stephen Sykes
|
2
|
+
|
3
|
+
module SlimScrooge
|
4
|
+
# A MonitoredHash allows us to return only some columns into the @attributes
|
5
|
+
# of an ActiveRecord model object, but to notice when an attribute that
|
6
|
+
# wasn't fetched is accessed.
|
7
|
+
#
|
8
|
+
# Also, when a result is first fetched for a particular callsite, we monitor
|
9
|
+
# all the columns so that we can immediately learn which columns are needed.
|
10
|
+
#
|
11
|
+
class MonitoredHash < Hash
|
12
|
+
attr_accessor :callsite, :result_set, :monitored_columns
|
13
|
+
|
14
|
+
# Create a monitored hash. The unmonitored_columns are accessed like a regular
|
15
|
+
# hash. The monitored columns kept separately, and new_column_access is called
|
16
|
+
# before they are returned.
|
17
|
+
#
|
18
|
+
def self.[](monitored_columns, unmonitored_columns, callsite)
|
19
|
+
hash = MonitoredHash.new {|hash, key| hash.new_column_access(key)}
|
20
|
+
hash.monitored_columns = monitored_columns
|
21
|
+
hash.merge!(unmonitored_columns)
|
22
|
+
hash.callsite = callsite
|
23
|
+
hash
|
24
|
+
end
|
25
|
+
|
26
|
+
# Called when an unknown column is requested, through the default proc.
|
27
|
+
# If the column requested is valid, and the result set is not completely
|
28
|
+
# loaded, then we reload. Otherwise just note the column with add_seen_column.
|
29
|
+
#
|
30
|
+
def new_column_access(name)
|
31
|
+
if @callsite.columns_hash.has_key?(name)
|
32
|
+
@result_set.reload! if @result_set && name != @callsite.primary_key
|
33
|
+
Callsites.add_seen_column(@callsite, name)
|
34
|
+
end
|
35
|
+
@monitored_columns[name]
|
36
|
+
end
|
37
|
+
|
38
|
+
# Reload if needed before allowing assignment
|
39
|
+
#
|
40
|
+
def []=(name, value)
|
41
|
+
if has_key?(name)
|
42
|
+
return super
|
43
|
+
elsif @result_set && @callsite.columns_hash.has_key?(name)
|
44
|
+
@result_set.reload!
|
45
|
+
Callsites.add_seen_column(@callsite, name)
|
46
|
+
end
|
47
|
+
@monitored_columns[name] = value
|
48
|
+
end
|
49
|
+
|
50
|
+
# Returns the column names
|
51
|
+
#
|
52
|
+
def keys
|
53
|
+
@result_set ? @callsite.columns_hash.keys : super | @monitored_columns.keys
|
54
|
+
end
|
55
|
+
|
56
|
+
# Check for a column name
|
57
|
+
#
|
58
|
+
def has_key?(name)
|
59
|
+
@result_set ? @callsite.columns_hash.has_key?(name) : super || @monitored_columns.has_key?(name)
|
60
|
+
end
|
61
|
+
|
62
|
+
alias_method :include?, :has_key?
|
63
|
+
|
64
|
+
# Called by Hash#update when reload is called on an ActiveRecord object
|
65
|
+
#
|
66
|
+
def to_hash
|
67
|
+
@result_set.reload! if @result_set
|
68
|
+
@monitored_columns.merge(self)
|
69
|
+
end
|
70
|
+
|
71
|
+
def freeze
|
72
|
+
@result_set.reload! if @result_set
|
73
|
+
@monitored_columns.freeze
|
74
|
+
super
|
75
|
+
end
|
76
|
+
|
77
|
+
# Marshal
|
78
|
+
# Dump a real hash - can't dump a monitored hash due to default proc
|
79
|
+
#
|
80
|
+
def _dump(depth)
|
81
|
+
Marshal.dump(to_hash)
|
82
|
+
end
|
83
|
+
|
84
|
+
def self._load(str)
|
85
|
+
Marshal.load(str)
|
86
|
+
end
|
87
|
+
end
|
88
|
+
end
|
89
|
+
|
90
|
+
# We need to change the update method of Hash so that it *always* calls
|
91
|
+
# to_hash. This is because it normally checks if other_hash is a kind of
|
92
|
+
# Hash, and doesn't bother calling to_hash if so. But we need it to call
|
93
|
+
# to_hash, because otherwise update will not get the complete columns
|
94
|
+
# from a MonitoredHash
|
95
|
+
#
|
96
|
+
# This is not harmful - to_hash in a regular Hash just returns self.
|
97
|
+
#
|
98
|
+
class Hash
|
99
|
+
alias_method :c_update, :update
|
100
|
+
def update(other_hash, &block)
|
101
|
+
c_update(other_hash.to_hash, &block)
|
102
|
+
end
|
103
|
+
end
|