each_batched 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (43) hide show
  1. data/MIT-LICENSE +20 -0
  2. data/README.rdoc +88 -0
  3. data/Rakefile +55 -0
  4. data/lib/each_batched/version.rb +3 -0
  5. data/lib/each_batched.rb +65 -0
  6. data/lib/tasks/each_batched_tasks.rake +4 -0
  7. data/test/dummy/Rakefile +7 -0
  8. data/test/dummy/app/controllers/application_controller.rb +3 -0
  9. data/test/dummy/app/helpers/application_helper.rb +2 -0
  10. data/test/dummy/app/models/company.rb +7 -0
  11. data/test/dummy/app/models/customer.rb +6 -0
  12. data/test/dummy/app/models/product.rb +5 -0
  13. data/test/dummy/app/models/purchase.rb +5 -0
  14. data/test/dummy/app/views/layouts/application.html.erb +14 -0
  15. data/test/dummy/config/application.rb +45 -0
  16. data/test/dummy/config/boot.rb +10 -0
  17. data/test/dummy/config/database.yml +25 -0
  18. data/test/dummy/config/environment.rb +5 -0
  19. data/test/dummy/config/environments/development.rb +30 -0
  20. data/test/dummy/config/environments/production.rb +60 -0
  21. data/test/dummy/config/environments/test.rb +42 -0
  22. data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
  23. data/test/dummy/config/initializers/inflections.rb +10 -0
  24. data/test/dummy/config/initializers/mime_types.rb +5 -0
  25. data/test/dummy/config/initializers/secret_token.rb +7 -0
  26. data/test/dummy/config/initializers/session_store.rb +8 -0
  27. data/test/dummy/config/locales/en.yml +5 -0
  28. data/test/dummy/config/routes.rb +2 -0
  29. data/test/dummy/config.ru +4 -0
  30. data/test/dummy/db/development.sqlite3 +0 -0
  31. data/test/dummy/db/migrate/0001_initialize_models.rb +36 -0
  32. data/test/dummy/db/schema.rb +48 -0
  33. data/test/dummy/db/test.sqlite3 +0 -0
  34. data/test/dummy/log/development.log +408 -0
  35. data/test/dummy/log/test.log +44659 -0
  36. data/test/dummy/public/404.html +26 -0
  37. data/test/dummy/public/422.html +26 -0
  38. data/test/dummy/public/500.html +26 -0
  39. data/test/dummy/public/favicon.ico +0 -0
  40. data/test/dummy/script/rails +6 -0
  41. data/test/each_batched_test.rb +235 -0
  42. data/test/test_helper.rb +24 -0
  43. metadata +176 -0
data/MIT-LICENSE ADDED
@@ -0,0 +1,20 @@
1
+ Copyright 2011 David Burry
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.rdoc ADDED
@@ -0,0 +1,88 @@
1
+ = EachBatched
2
+
3
+ More grouping/batching logic options than what's included in Rails.
4
+
5
+ Ever since Rails 2.3, it has had ActiveRecord::Batches#find_in_batches[http://api.rubyonrails.org/classes/ActiveRecord/Batches.html#method-i-find_in_batches] (and its cousin ActiveRecord::Batches#find_each[http://api.rubyonrails.org/classes/ActiveRecord/Batches.html#method-i-find_each]) as a great resource saver, because it allowed you to run through larger data sets in batches, instead of loading everything under the sun into memory first.
6
+
7
+ But it has some gotchas. First, the algorithm it uses tries to keep from messing up during concurrent inserts/deletes, during the loop. This is a good thing, except it does it by fixing the order to an id order, then grabbing each successive batch (with limit) that's greater than your last primary key id. So there's no way to limit it to a subset of your data, and there's no way to order anything any differently than by primary key id order. This can be rather limiting sometimes. Also it uses finders in kind of an old fashioned way compared to Rails 3... instead of scopes.
8
+
9
+ So this library attempts to address these, by providing two additional algorithms you can choose from, depending on your needs.
10
+
11
+ == Dependencies
12
+
13
+ * Ruby 1.9.2 - does not support Ruby 1.8!
14
+ * Rails 3.x - does not support Rails 2!
15
+
16
+ == Features
17
+
18
+ * Saves memory by not needing to load all records into memory at once, just one batch at a time, looping over them all.
19
+ * You can specify an order for your results, using the standard Rails 3 arel/scoped way.
20
+ * You can specify an offset and/or limit to only grab some results, using the standard Rails 3 arel/scoped way.
21
+ * Two different algorithms provided, to fit your different needs.
22
+ * Includes variants that yields groups (each as a scope), or yields individual rows, depending on your needs.
23
+
24
+ == Installation
25
+
26
+ Add to your <tt>Gemfile</tt>:
27
+
28
+ # Gemfile
29
+
30
+ gem 'each_batched'
31
+
32
+ and run:
33
+
34
+ $ bundle install
35
+
36
+ == Usage
37
+
38
+ First, let's explain the "range" algorithm:
39
+ * It simply uses offset/limit internally to run through each batch.
40
+ * Simple obvious approach, few queries.
41
+ * Could work on primary-key-less data.
42
+ * Does NOT work well with data that could have inserts/deletes while you're looping (it might miss or duplicate random rows at the boundaries of batches)! So it should only be used on data that you are sure will not change (such as locked data, or static data, etc).
43
+
44
+ YourModel.each_by_ranges do |record|
45
+ # Do something with this model record
46
+ end
47
+ YourModel.batches_by_ranges do |batch|
48
+ # Do something with this batch
49
+ # It's a standard model scope that's already been loaded
50
+ # and can act like an array of records
51
+ end
52
+
53
+ Next, the "ids" algorithm:
54
+ * Grabs a list of all selected primary keys in one query, then loops through them all, grabbing the row data in batches.
55
+ * Works with simultaneously changing data nicely (might miss added/deleted rows themselves of course).
56
+ * For complicated queries, it could be faster than other approaches too.
57
+ * May generate really long queries if you're doing a lot of rows in each batch.
58
+
59
+ YourModel.each_by_ids do |record|
60
+ # Do something with this model record
61
+ end
62
+ YourModel.batches_by_ids do |batch|
63
+ # Do something with this batch
64
+ # It's a standard model scope that has NOT been lazy loaded yet
65
+ # but will be as soon as you access its records
66
+ end
67
+
68
+ All the above can take an optional parameter: the size of the batch to use (defaults to 1000).
69
+
70
+ == Contributing
71
+
72
+ If you think you found a bug or want a feature, get involved at http://github.com/dburry/each_batched/issues If you'd then like to contribute a patch, use Github's wonderful fork and pull request features.
73
+
74
+ To set up a full development environment:
75
+ * <tt>git clone</tt> the repository,
76
+ * have RVM[https://rvm.beginrescueend.com/] and Bundler[http://gembundler.com/] installed,
77
+ * then cd into your repo (follow any RVM prompts if this is your first time using that),
78
+ * and run <tt>bundle install</tt> to pull in all the rest of the development dependencies.
79
+ * After that point, <tt>rake -T</tt> should be fairly self-explanatory.
80
+
81
+ == Alternatives
82
+
83
+ * ActiveRecord::Batches#find_in_batches[http://api.rubyonrails.org/classes/ActiveRecord/Batches.html#method-i-find_in_batches] and
84
+ * ActiveRecord::Batches#find_each[http://api.rubyonrails.org/classes/ActiveRecord/Batches.html#method-i-find_each] built into rails 2.3 and above.
85
+
86
+ == License
87
+
88
+ This library is distributed under the MIT license. Please see the LICENSE file.
data/Rakefile ADDED
@@ -0,0 +1,55 @@
1
+ #!/usr/bin/env rake
2
+ begin
3
+ require 'bundler/setup'
4
+ rescue LoadError
5
+ puts 'You must `gem install bundler` and `bundle install` to run rake tasks'
6
+ end
7
+ begin
8
+ require 'rdoc/task'
9
+ rescue LoadError
10
+ require 'rdoc/rdoc'
11
+ require 'rake/rdoctask'
12
+ RDoc::Task = Rake::RDocTask
13
+ end
14
+
15
+ RDoc::Task.new(:rdoc) do |rdoc|
16
+ rdoc.rdoc_dir = 'rdoc'
17
+ rdoc.title = 'EachBatched'
18
+ rdoc.options << '--line-numbers'
19
+ rdoc.rdoc_files.include('README.rdoc')
20
+ rdoc.rdoc_files.include('lib/**/*.rb')
21
+ end
22
+
23
+
24
+
25
+ Bundler::GemHelper.install_tasks
26
+
27
+ require 'rake/testtask'
28
+
29
+ Rake::TestTask.new(:test) do |t|
30
+ t.libs << 'lib'
31
+ t.libs << 'test'
32
+ t.pattern = 'test/**/*_test.rb'
33
+ t.verbose = false
34
+ end
35
+
36
+ begin
37
+ require 'simplecov'
38
+ rescue LoadError
39
+ puts 'WARNING: missing simplecov library, some tasks are not available'
40
+ else
41
+ desc 'Analyze code coverage with tests'
42
+ task :coverage => 'coverage:clobber' do
43
+ ENV['USE_SIMPLECOV'] = :true.to_s
44
+ Rake::Task['test'].invoke
45
+ # see top of test helper for the rest of this task, which is triggered with the environment var....
46
+ end
47
+ namespace :coverage do
48
+ desc 'Remove output directory generated by simplecov'
49
+ task :clobber do
50
+ FileUtils.rm_rf(File.expand_path('../coverage', __FILE__))
51
+ end
52
+ end
53
+ end
54
+
55
+ task :default => :test
@@ -0,0 +1,3 @@
1
+ module EachBatched
2
+ VERSION = "0.1.0"
3
+ end
@@ -0,0 +1,65 @@
1
+ #
2
+ # More grouping/batching logic options than what's included in Rails.
3
+ #
4
+
5
+ module EachBatched
6
+
7
+ # Default batch size to use, if none is specified (defaults to 1000)
8
+ DEFAULT_BATCH_SIZE = 1_000
9
+
10
+ # Yields batches of records from the current scope.
11
+ # Uses offset/limit internally to run through each batch,
12
+ # and can be further restricted by in-scope offset/limit/order (it doesn't just toss them out!).
13
+ #
14
+ # * This algorithm does NOT work well with data that may have inserts/deletes while you're looping,
15
+ # so if that's a problem, then you should either lock the table or rows first or use a different algorithm
16
+ # (like ActiveRecord::Batches#find_in_batches or #batches_by_ids).
17
+ # * This algorithm may be slower than #batches_by_ids if your query doesn't execute very quickly.
18
+ # * This algorithm can't be lazily loaded, because it checks for empty results to see when it's done.
19
+ def batches_by_range(batch_size=DEFAULT_BATCH_SIZE)
20
+ start_offset = scoped.offset_value || 0
21
+ end_limit = scoped.limit_value # || nil
22
+ group_number = 0
23
+ processed_number = 0
24
+ # This giant while condition (with multiple assignments in it) is a mess, isn't it!
25
+ # But simplifying it means I have to repeat most of it multiple times!
26
+ # And putting it into a subroutine doesn't really save space either, with lots of parameters and/or return values!
27
+ while (length = (records = offset(start_offset + batch_size * group_number).
28
+ limit(asked_limit = end_limit.nil? || processed_number + batch_size < end_limit ?
29
+ batch_size : end_limit - processed_number)).length) > 0
30
+ yield records
31
+ processed_number += length
32
+ break if length < asked_limit || (! end_limit.nil? && processed_number >= end_limit)
33
+ group_number += 1
34
+ end
35
+ end
36
+
37
+ # Loops through each individual row found by #batches_by_range, instead of each batch
38
+ # see #batches_by_range for an explanation of its algorithm
39
+ def each_by_range(batch_size=DEFAULT_BATCH_SIZE)
40
+ batches_by_range(batch_size) { |batch| batch.each { |row| yield row } }
41
+ end
42
+
43
+ # Yields batches of records from the current scope
44
+ # Snapshots the primary key ids in scope, then loops through grabbing the rows, one chunk of ids at a time.
45
+ #
46
+ # * You should explicitly set an order if you want the same order as #batches_by_range, or it may be different.
47
+ # * The yielded scope can be lazily loaded (though the id selection query has already run obviously)
48
+ def batches_by_ids(batch_size=DEFAULT_BATCH_SIZE)
49
+ reduced_scope = scoped.tap { |s| s.where_values = [] }.offset(nil).limit(nil)
50
+ select("#{table_name}.#{primary_key}").collect(&(primary_key.to_sym)).in_groups_of(batch_size, false) do |group_ids|
51
+ # keeps select/group/joins/includes, inside inner batched scope
52
+ yield reduced_scope.where(primary_key => group_ids)
53
+ end
54
+ end
55
+
56
+ # Loops through each individual row found by #batches_by_ids, instead of each batch
57
+ # see #batches_by_ids for an explanation of its algorithm
58
+ def each_by_ids(batch_size=DEFAULT_BATCH_SIZE)
59
+ batches_by_ids(batch_size) { |batch| batch.each { |row| yield row } }
60
+ end
61
+
62
+ end
63
+
64
+ # add all this functionality to ActiveRecord for all models to use
65
+ ActiveRecord::Base.extend EachBatched
@@ -0,0 +1,4 @@
1
+ # desc "Explaining what the task does"
2
+ # task :each_batched do
3
+ # # Task goes here
4
+ # end
@@ -0,0 +1,7 @@
1
+ #!/usr/bin/env rake
2
+ # Add your own tasks in files placed in lib/tasks ending in .rake,
3
+ # for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.
4
+
5
+ require File.expand_path('../config/application', __FILE__)
6
+
7
+ Dummy::Application.load_tasks
@@ -0,0 +1,3 @@
1
+ class ApplicationController < ActionController::Base
2
+ protect_from_forgery
3
+ end
@@ -0,0 +1,2 @@
1
+ module ApplicationHelper
2
+ end
@@ -0,0 +1,7 @@
1
+
2
+ class Company < ActiveRecord::Base
3
+ has_many :customers
4
+ has_many :products
5
+ has_many :customer_purchases, :through => :customers, :source => :purchases
6
+ has_many :purchased_products, :through => :customer_purchases, :source => :product, :uniq => true
7
+ end
@@ -0,0 +1,6 @@
1
+
2
+ class Customer < ActiveRecord::Base
3
+ belongs_to :company
4
+ has_many :purchases
5
+ has_many :purchased_products, :through => :purchases, :source => :products, :uniq => true
6
+ end
@@ -0,0 +1,5 @@
1
+
2
+ class Product < ActiveRecord::Base
3
+ belongs_to :company
4
+ has_many :purchases
5
+ end
@@ -0,0 +1,5 @@
1
+
2
+ class Purchase < ActiveRecord::Base
3
+ belongs_to :product
4
+ belongs_to :customer
5
+ end
@@ -0,0 +1,14 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Dummy</title>
5
+ <%= stylesheet_link_tag "application" %>
6
+ <%= javascript_include_tag "application" %>
7
+ <%= csrf_meta_tags %>
8
+ </head>
9
+ <body>
10
+
11
+ <%= yield %>
12
+
13
+ </body>
14
+ </html>
@@ -0,0 +1,45 @@
1
+ require File.expand_path('../boot', __FILE__)
2
+
3
+ require 'rails/all'
4
+
5
+ Bundler.require
6
+ require "each_batched"
7
+
8
+ module Dummy
9
+ class Application < Rails::Application
10
+ # Settings in config/environments/* take precedence over those specified here.
11
+ # Application configuration should go into files in config/initializers
12
+ # -- all .rb files in that directory are automatically loaded.
13
+
14
+ # Custom directories with classes and modules you want to be autoloadable.
15
+ # config.autoload_paths += %W(#{config.root}/extras)
16
+
17
+ # Only load the plugins named here, in the order given (default is alphabetical).
18
+ # :all can be used as a placeholder for all plugins not explicitly named.
19
+ # config.plugins = [ :exception_notification, :ssl_requirement, :all ]
20
+
21
+ # Activate observers that should always be running.
22
+ # config.active_record.observers = :cacher, :garbage_collector, :forum_observer
23
+
24
+ # Set Time.zone default to the specified zone and make Active Record auto-convert to this zone.
25
+ # Run "rake -D time" for a list of tasks for finding time zone names. Default is UTC.
26
+ # config.time_zone = 'Central Time (US & Canada)'
27
+
28
+ # The default locale is :en and all translations from config/locales/*.rb,yml are auto loaded.
29
+ # config.i18n.load_path += Dir[Rails.root.join('my', 'locales', '*.{rb,yml}').to_s]
30
+ # config.i18n.default_locale = :de
31
+
32
+ # Configure the default encoding used in templates for Ruby 1.9.
33
+ config.encoding = "utf-8"
34
+
35
+ # Configure sensitive parameters which will be filtered from the log file.
36
+ config.filter_parameters += [:password]
37
+
38
+ # Enable the asset pipeline
39
+ config.assets.enabled = false
40
+
41
+ # Version of your assets, change this if you want to expire all your assets
42
+ config.assets.version = '1.0'
43
+ end
44
+ end
45
+
@@ -0,0 +1,10 @@
1
+ require 'rubygems'
2
+ gemfile = File.expand_path('../../../../Gemfile', __FILE__)
3
+
4
+ if File.exist?(gemfile)
5
+ ENV['BUNDLE_GEMFILE'] = gemfile
6
+ require 'bundler'
7
+ Bundler.setup
8
+ end
9
+
10
+ $:.unshift File.expand_path('../../../../lib', __FILE__)
@@ -0,0 +1,25 @@
1
+ # SQLite version 3.x
2
+ # gem install sqlite3
3
+ #
4
+ # Ensure the SQLite 3 gem is defined in your Gemfile
5
+ # gem 'sqlite3'
6
+ development:
7
+ adapter: sqlite3
8
+ database: db/development.sqlite3
9
+ pool: 5
10
+ timeout: 5000
11
+
12
+ # Warning: The database defined as "test" will be erased and
13
+ # re-generated from your development database when you run "rake".
14
+ # Do not set this db to the same as development or production.
15
+ test:
16
+ adapter: sqlite3
17
+ database: db/test.sqlite3
18
+ pool: 5
19
+ timeout: 5000
20
+
21
+ production:
22
+ adapter: sqlite3
23
+ database: db/production.sqlite3
24
+ pool: 5
25
+ timeout: 5000
@@ -0,0 +1,5 @@
1
+ # Load the rails application
2
+ require File.expand_path('../application', __FILE__)
3
+
4
+ # Initialize the rails application
5
+ Dummy::Application.initialize!
@@ -0,0 +1,30 @@
1
+ Dummy::Application.configure do
2
+ # Settings specified here will take precedence over those in config/application.rb
3
+
4
+ # In the development environment your application's code is reloaded on
5
+ # every request. This slows down response time but is perfect for development
6
+ # since you don't have to restart the web server when you make code changes.
7
+ config.cache_classes = false
8
+
9
+ # Log error messages when you accidentally call methods on nil.
10
+ config.whiny_nils = true
11
+
12
+ # Show full error reports and disable caching
13
+ config.consider_all_requests_local = true
14
+ config.action_controller.perform_caching = false
15
+
16
+ # Don't care if the mailer can't send
17
+ config.action_mailer.raise_delivery_errors = false
18
+
19
+ # Print deprecation notices to the Rails logger
20
+ config.active_support.deprecation = :log
21
+
22
+ # Only use best-standards-support built into browsers
23
+ config.action_dispatch.best_standards_support = :builtin
24
+
25
+ # Do not compress assets
26
+ config.assets.compress = false
27
+
28
+ # Expands the lines which load the assets
29
+ config.assets.debug = true
30
+ end
@@ -0,0 +1,60 @@
1
+ Dummy::Application.configure do
2
+ # Settings specified here will take precedence over those in config/application.rb
3
+
4
+ # Code is not reloaded between requests
5
+ config.cache_classes = true
6
+
7
+ # Full error reports are disabled and caching is turned on
8
+ config.consider_all_requests_local = false
9
+ config.action_controller.perform_caching = true
10
+
11
+ # Disable Rails's static asset server (Apache or nginx will already do this)
12
+ config.serve_static_assets = false
13
+
14
+ # Compress JavaScripts and CSS
15
+ config.assets.compress = true
16
+
17
+ # Don't fallback to assets pipeline if a precompiled asset is missed
18
+ config.assets.compile = false
19
+
20
+ # Generate digests for assets URLs
21
+ config.assets.digest = true
22
+
23
+ # Defaults to Rails.root.join("public/assets")
24
+ # config.assets.manifest = YOUR_PATH
25
+
26
+ # Specifies the header that your server uses for sending files
27
+ # config.action_dispatch.x_sendfile_header = "X-Sendfile" # for apache
28
+ # config.action_dispatch.x_sendfile_header = 'X-Accel-Redirect' # for nginx
29
+
30
+ # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies.
31
+ # config.force_ssl = true
32
+
33
+ # See everything in the log (default is :info)
34
+ # config.log_level = :debug
35
+
36
+ # Use a different logger for distributed setups
37
+ # config.logger = SyslogLogger.new
38
+
39
+ # Use a different cache store in production
40
+ # config.cache_store = :mem_cache_store
41
+
42
+ # Enable serving of images, stylesheets, and JavaScripts from an asset server
43
+ # config.action_controller.asset_host = "http://assets.example.com"
44
+
45
+ # Precompile additional assets (application.js, application.css, and all non-JS/CSS are already added)
46
+ # config.assets.precompile += %w( search.js )
47
+
48
+ # Disable delivery errors, bad email addresses will be ignored
49
+ # config.action_mailer.raise_delivery_errors = false
50
+
51
+ # Enable threaded mode
52
+ # config.threadsafe!
53
+
54
+ # Enable locale fallbacks for I18n (makes lookups for any locale fall back to
55
+ # the I18n.default_locale when a translation can not be found)
56
+ config.i18n.fallbacks = true
57
+
58
+ # Send deprecation notices to registered listeners
59
+ config.active_support.deprecation = :notify
60
+ end
@@ -0,0 +1,42 @@
1
+ Dummy::Application.configure do
2
+ # Settings specified here will take precedence over those in config/application.rb
3
+
4
+ # The test environment is used exclusively to run your application's
5
+ # test suite. You never need to work with it otherwise. Remember that
6
+ # your test database is "scratch space" for the test suite and is wiped
7
+ # and recreated between test runs. Don't rely on the data there!
8
+ config.cache_classes = true
9
+
10
+ # Configure static asset server for tests with Cache-Control for performance
11
+ config.serve_static_assets = true
12
+ config.static_cache_control = "public, max-age=3600"
13
+
14
+ # Log error messages when you accidentally call methods on nil
15
+ config.whiny_nils = true
16
+
17
+ # Show full error reports and disable caching
18
+ config.consider_all_requests_local = true
19
+ config.action_controller.perform_caching = false
20
+
21
+ # Raise exceptions instead of rendering exception templates
22
+ config.action_dispatch.show_exceptions = false
23
+
24
+ # Disable request forgery protection in test environment
25
+ config.action_controller.allow_forgery_protection = false
26
+
27
+ # Tell Action Mailer not to deliver emails to the real world.
28
+ # The :test delivery method accumulates sent emails in the
29
+ # ActionMailer::Base.deliveries array.
30
+ config.action_mailer.delivery_method = :test
31
+
32
+ # Use SQL instead of Active Record's schema dumper when creating the test database.
33
+ # This is necessary if your schema can't be completely dumped by the schema dumper,
34
+ # like if you have constraints or database-specific column types
35
+ # config.active_record.schema_format = :sql
36
+
37
+ # Print deprecation notices to the stderr
38
+ config.active_support.deprecation = :stderr
39
+
40
+ # Allow pass debug_assets=true as a query parameter to load pages with unpackaged assets
41
+ config.assets.allow_debugging = true
42
+ end
@@ -0,0 +1,7 @@
1
+ # Be sure to restart your server when you modify this file.
2
+
3
+ # You can add backtrace silencers for libraries that you're using but don't wish to see in your backtraces.
4
+ # Rails.backtrace_cleaner.add_silencer { |line| line =~ /my_noisy_library/ }
5
+
6
+ # You can also remove all the silencers if you're trying to debug a problem that might stem from framework code.
7
+ # Rails.backtrace_cleaner.remove_silencers!
@@ -0,0 +1,10 @@
1
+ # Be sure to restart your server when you modify this file.
2
+
3
+ # Add new inflection rules using the following format
4
+ # (all these examples are active by default):
5
+ # ActiveSupport::Inflector.inflections do |inflect|
6
+ # inflect.plural /^(ox)$/i, '\1en'
7
+ # inflect.singular /^(ox)en/i, '\1'
8
+ # inflect.irregular 'person', 'people'
9
+ # inflect.uncountable %w( fish sheep )
10
+ # end
@@ -0,0 +1,5 @@
1
+ # Be sure to restart your server when you modify this file.
2
+
3
+ # Add new mime types for use in respond_to blocks:
4
+ # Mime::Type.register "text/richtext", :rtf
5
+ # Mime::Type.register_alias "text/html", :iphone
@@ -0,0 +1,7 @@
1
+ # Be sure to restart your server when you modify this file.
2
+
3
+ # Your secret key for verifying the integrity of signed cookies.
4
+ # If you change this key, all old signed cookies will become invalid!
5
+ # Make sure the secret is at least 30 characters and all random,
6
+ # no regular words or you'll be exposed to dictionary attacks.
7
+ Dummy::Application.config.secret_token = '34350caf6c17053a9b555856c85a7b3b9da91bc35d6f3362621232d0b18621fd01f1f1cbdac374dc9d8d9425fe50ab82b4069fe4b3b6fefd1832a94a4dffc117'
@@ -0,0 +1,8 @@
1
+ # Be sure to restart your server when you modify this file.
2
+
3
+ Dummy::Application.config.session_store :cookie_store, key: '_dummy_session'
4
+
5
+ # Use the database for sessions instead of the cookie-based default,
6
+ # which shouldn't be used to store highly confidential information
7
+ # (create the session table with "rails generate session_migration")
8
+ # Dummy::Application.config.session_store :active_record_store
@@ -0,0 +1,5 @@
1
+ # Sample localization file for English. Add more files in this directory for other locales.
2
+ # See https://github.com/svenfuchs/rails-i18n/tree/master/rails%2Flocale for starting points.
3
+
4
+ en:
5
+ hello: "Hello world"
@@ -0,0 +1,2 @@
1
+ Dummy::Application.routes.draw do
2
+ end
@@ -0,0 +1,4 @@
1
+ # This file is used by Rack-based servers to start the application.
2
+
3
+ require ::File.expand_path('../config/environment', __FILE__)
4
+ run Dummy::Application
Binary file
@@ -0,0 +1,36 @@
1
+
2
+ class InitializeModels < ActiveRecord::Migration
3
+ def self.up
4
+ create_table :companies do |t|
5
+ t.integer :sort, :null => false, :default => 0
6
+ end
7
+ create_table :customers do |t|
8
+ t.integer :sort , :null => false, :default => 0
9
+ t.references :company, :null => false
10
+ end
11
+ create_table :products do |t|
12
+ t.integer :sort, :null => false, :default => 0
13
+ t.references :company, :null => false
14
+ end
15
+ create_table :purchases do |t|
16
+ t.date :ordered_on, :null => false, :default => Date.parse('Jan 1 2011')
17
+ t.references :product, :null => false
18
+ t.references :customer, :null => false
19
+ end
20
+ add_index :companies, :sort
21
+ add_index :customers, :sort
22
+ add_index :customers, :company_id
23
+ add_index :products, :sort
24
+ add_index :products, :company_id
25
+ add_index :purchases, :ordered_on
26
+ add_index :purchases, :product_id
27
+ add_index :purchases, :customer_id
28
+ end
29
+
30
+ def self.down
31
+ drop_table :purchases
32
+ drop_table :products
33
+ drop_table :customers
34
+ drop_table :companies
35
+ end
36
+ end