each_batched 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- data/MIT-LICENSE +20 -0
- data/README.rdoc +88 -0
- data/Rakefile +55 -0
- data/lib/each_batched/version.rb +3 -0
- data/lib/each_batched.rb +65 -0
- data/lib/tasks/each_batched_tasks.rake +4 -0
- data/test/dummy/Rakefile +7 -0
- data/test/dummy/app/controllers/application_controller.rb +3 -0
- data/test/dummy/app/helpers/application_helper.rb +2 -0
- data/test/dummy/app/models/company.rb +7 -0
- data/test/dummy/app/models/customer.rb +6 -0
- data/test/dummy/app/models/product.rb +5 -0
- data/test/dummy/app/models/purchase.rb +5 -0
- data/test/dummy/app/views/layouts/application.html.erb +14 -0
- data/test/dummy/config/application.rb +45 -0
- data/test/dummy/config/boot.rb +10 -0
- data/test/dummy/config/database.yml +25 -0
- data/test/dummy/config/environment.rb +5 -0
- data/test/dummy/config/environments/development.rb +30 -0
- data/test/dummy/config/environments/production.rb +60 -0
- data/test/dummy/config/environments/test.rb +42 -0
- data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
- data/test/dummy/config/initializers/inflections.rb +10 -0
- data/test/dummy/config/initializers/mime_types.rb +5 -0
- data/test/dummy/config/initializers/secret_token.rb +7 -0
- data/test/dummy/config/initializers/session_store.rb +8 -0
- data/test/dummy/config/locales/en.yml +5 -0
- data/test/dummy/config/routes.rb +2 -0
- data/test/dummy/config.ru +4 -0
- data/test/dummy/db/development.sqlite3 +0 -0
- data/test/dummy/db/migrate/0001_initialize_models.rb +36 -0
- data/test/dummy/db/schema.rb +48 -0
- data/test/dummy/db/test.sqlite3 +0 -0
- data/test/dummy/log/development.log +408 -0
- data/test/dummy/log/test.log +44659 -0
- data/test/dummy/public/404.html +26 -0
- data/test/dummy/public/422.html +26 -0
- data/test/dummy/public/500.html +26 -0
- data/test/dummy/public/favicon.ico +0 -0
- data/test/dummy/script/rails +6 -0
- data/test/each_batched_test.rb +235 -0
- data/test/test_helper.rb +24 -0
- 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
|
data/lib/each_batched.rb
ADDED
@@ -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
|
data/test/dummy/Rakefile
ADDED
@@ -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,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,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,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,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
|
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
|