Multiple stores using a single Spree application instance.
Uses request subdomain to swap database, Rails cache (preferences), image paths.
Add to your Gemfile
:
gem 'spree_shared', github: 'spree-contrib/spree_shared', branch: 'master'
Make sure your config/database.yml
has valid db connection.
Create an initializer config/initializers/apartment.rb
with the following command:
bundle exec rails generate apartment:install
Search for following line:
Apartment.configure do |config|
...
# supply list of database names for migrations to run on
config.tenant_names = lambda { ToDo_Tenant_Or_User_Model.pluck :database }
end
And change it to include two sample subdomains:
Apartment.configure do |config|
...
config.tenant_names = %w(store1 store2)
end
Add the following line to host application.rb
config.middleware.use 'Apartment::Elevators::Subdomain'
Change file paths and urls by adding to config/initializers/spree.rb
following:
Spree::Image.attachment_definitions[:attachment][:url] = '/spree/products/:tenant/:id/:style/:basename.:extension'
Spree::Image.attachment_definitions[:attachment][:path] = ':rails_root/public/spree/products/:tenant/:id/:style/:basename.:extension'
Then allow Paperclip to access tenant from Spree::Image by adding following to Spree initializer:
Paperclip.interpolates :tenant do |attachment, _style|
attachment.instance.tenant
end
By default tenant will resolve to Apartment::Tenant.current_tenant
but you can change it - eg. suppose you use databases like tenant_12345 and only want tenant id in file path, then add following line to config/initializers/apartment.rb
Spree::Image.tenant_proc = -> { Apartment::Tenant.current_tenant.match(/(\d+)/)[1] }
Bootstrap sample stores:
bundle exec rake spree_shared:bootstrap['store1']
bundle exec rake spree_shared:bootstrap['store2']
Setup local subdomains for sample stores, as spree_shared uses by default subdomain routing you need to confirm some local domains such as:
store1.spree.dev
store2.spree.dev
This can be done using Pow or editing your local /etc/hosts
file.
Set namespace for cache engine in development.rb
and/or production.rb
config.cache_store = :memory_store, { namespace: lambda { Apartment::Tenant.current_tenant } }
If you'd like to set preferences for every store you can do so in your config/initializers/spree.rb
initializer by iterating over each store, and then setting it's preference. Since this is multi-tenant with each store having their own database the usual Spree.config block can't be used as it only sets the preference for a single database.
Here is an example:
Apartment.tenant_names.each do |store|
begin
Apartment::Tenant.switch store
Spree::Config.auto_capture = true
rescue
puts " Failed to set up config for store '#{store}'"
end
end
See corresponding guidelines
Copyright (c) 2013-2015 Spree Commerce Inc, and other contributors, released under the New BSD License