Rails routing to handle multiple domains on single application
Solution 1
It's actually simpler in Rails 3, as per http://guides.rubyonrails.org/routing.html#advanced-constraints:
1) define a custom constraint class in lib/domain_constraint.rb
:
class DomainConstraint
def initialize(domain)
@domains = [domain].flatten
end
def matches?(request)
@domains.include? request.domain
end
end
2) use the class in your routes with the new block syntax
constraints DomainConstraint.new('mydomain.com') do
root :to => 'mydomain#index'
end
root :to => 'main#index'
or the old-fashioned option syntax
root :to => 'mydomain#index', :constraints => DomainConstraint.new('mydomain.com')
Solution 2
In Rails 5, you can simply do this in your routes:
constraints subdomain: 'blogs' do
match '/' => 'blogs#show'
end
Comments
-
Aaron Vegh almost 4 years
I've been unable to find a workable solution to this problem, despite several similar questions here and elsewhere. It seems likely that this question hasn't been answered for Rails 3, so here goes:
I have an application that currently allows users to create their own subdomain that contains their instance of the application. While in Rails 2 you were best served using the subdomain-fu gem, in version 3 it's dramatically simpler, as per the Railscast -- http://railscasts.com/episodes/221-subdomains-in-rails-3.
That's good stuff, but I also want to provide the option for users to associate their own domain name with their account. So while they might have http://userx.mydomain.com, I'd like them to choose to have http://userx.com associated as well.
I found a few references to doing this in Rails 2, but those techniques don't appear to work anymore (particularly this one: https://feefighters.com/blog/hosting-multiple-domains-from-a-single-rails-app/).
Can anyone recommend a way to use routes to accept an arbitrary domain and pass it along to a controller so I can show the appropriate content?
Update: I've gotten most of an answer now, thanks to Leonid's timely response, and a fresh look at the code. It ultimately required an addition to the existing Subdomain code that I was using (from the Railscast solution) and then adding a bit to routes.rb. I'm not all the way there yet but I want to post what I have so far.
In lib/subdomain.rb:
class Subdomain def self.matches?(request) request.subdomain.present? && request.subdomain != "www" end end class Domain def self.matches?(request) request.domain.present? && request.domain != "mydomain.com" end end
I've added the second class in imitation of the first, which is known working. I simply add a condition that ensures that the incoming domain is not the one for which I'm hosting the main site.
This class is used in routes.rb:
require 'subdomain' constraints(Domain) do match '/' => 'blogs#show' end constraints(Subdomain) do match '/' => 'blogs#show' end
Here, I'm prepending the existing subdomain code (again, it's working fine) with a stanza to check for the Domain. If this server responds to that domain and it's not the one under which the main site operates, forward to the specified controller.
And while that appears to be working, I don't quite have the whole thing working yet, but I think this particular problem has been solved.
-
Aaron Vegh over 13 yearsThanks for your answer... I had given up hope! But looking at this code, it appears that I need to anticipate the domain being used. I want to use any arbitrary domain. But I've now discovered the answer (mostly), so I'll edit my question above.
-
ybakos about 13 yearsWhy would you declare the class in an initializer? Shouldn't that be in lib?
-
Jared almost 12 yearsThis answer seems much simpler to me.
-
superluminary over 11 yearsThis is a great solution. How does it work with a development environment?
-
Leonid Shevtsov over 11 years@superluminary it works perfectly fine if you set up local domains for development (for example, via
/etc/hosts
). -
Eric Muyser over 11 yearsNote: if you use Pow locally and have mydomain.com.dev, then
request.domain
returns .com.dev. Changerequest.domain
torequest.host
and it works perfectly. -
Just Lucky Really about 9 yearsI've found that I have to create unnamed routes for this to work, otherwise I get the
Invalid route name, already in use: 'root'
error ... To do this, I changed the route toroot :to => 'mydomain#index', as: nil
-
Andrew Wei almost 8 yearsI made one minor change to def matches?, from "request.domain" to "request.host" to be specific to the subdomain.