Rails Rails 路由 - 解决多子域名问题

zamia · 2015年11月15日 · 最后由 zamia 回复于 2017年09月14日 · 5481 次阅读

rails 多子域名问题

Rails 项目发展到较大规模的时候、或者为了其他各种原因,一定会遇到多子域名的问题。目前网上的很多资料只是简单的介绍了利用 constraints 进行操作的方法,并没有系统的解决多子域名实操的时候会遇到的各种问题。比如:

  1. 多个子域名下,路由和控制器的设计?
  2. 多个 routes 文件如何拆分?
  3. url helper 使用的时候的注意事项?

这篇文章算是对上述问题进行的一个较深入的总结和实操,请阅读之前需要 Rails 路由先有个大概的了解,期望大家读完之后对 Rails 路由理解的更加深入。

多子域名问题解决

constraint 是基础

Rails 提供了 constraints 方法来对一组路由进行限制,比如官方文档(Rails 3.2)中提供的例子:

constraints(:id => /\d+\.\d+/) do
  resources :orders
end

上面的例子中只有 /posts/123.456 这样的 id 是允许的,而 "/posts/123" 就是无效的。这也是 constraint(限制)的意思。

那么,根据这个思路,我们可以限制一组路由只在某个子域名下生效,也就达到了多子域名的目的:

constraints :subdomain => "m" do
  resources :orders # mobile下
end
constraints :subdomain => "www" do
  resources :orders # pc下
end

另外,不要在 routes 文件里面 hard code 一些配置,因为一般不同的环境的子域名可能不一样,比如你的测试环境可能需要 alpha.m.example.com 这样的域名,而正式环境才是 m.example.com 这样的域名。我们稍微修改的好一点(示例代码也要写好啊,否则新手容易跟着画瓢):

### config/environments/development.rb
config.mobile_subdomain = "m"
config.main_subdomain = "www"

### config/routes.rb
constraints :subdomain => Rails.configuration.mobile_subdomain do
  resources :orders
end
constraints :subdomain => Rails.configuration.main_subdomain do
  resources :orders
  # 下面是其他路由,比如
  resources :users
end

但是这样显然是不 work 的(手册跟实际工作是有差距的),因为 constraints 只是做了『限制』,路由指向的 controller 并不会改变,也就是说上面的 mobile 子域名中的 orders 也是同样指向了 ::OrdersController 这个控制器,实际工作中,我们一般是期望 m.example.com/orders 路由指向 ::Mobile::USersController 这个控制器的。

那应该如何解决呢?这个时候就需要使用 scope 了。

先提下 namespace

scope 日常不太常使用,因为大家一般都是使用 namespace 就够了,比如最常见的写法:

namespace :admin do
  resources :orders
end

这样会把 /admin/orders 路由指向 ::Admin::PostsController,很方便吧?

scope 来解决

但是在子域名环境下,这样是达不到目的的。因为我们期望 "m.example.com/orders" 能指向 ::Mobile::OrdersController,那么该如何搞呢,这个时候就需要 scope 了。scope 提供了比较 namepace 更细粒度的控制参数,完全可以满足我们的需求。

下面的代码来自于官方文档,稍微翻译了下:

### 把 url "/posts" (不包含/admin前缀) 指向Admin::PostsController
scope :module => "admin" do
  resources :posts
end 

### 把 posts相关路由添加 "/admin/" 前缀
scope :path => "/admin" do
  resources :posts
end

### 修改 url helper,用 +sekret_posts_path+ 替代 +posts_path+
scope :as => "sekret" do
  resources :posts
end

所以,这三个参数是可以独立使用的,那么上面提到的子域名的问题的解决方案也就有了:

constraints :subdomain => Rails.configuration.mobile_subdomain do
  scope module: 'mobile' do 
    resources :orders
  end
end
constraints :subdomain => Rails.configuration.main_subdomain do
  resources :orders
end

这样的话,mobile 子域名下的/orders 会路由到 "::Mobile::OrdersController",目标达成!

优化一下

这样好像还不够好,这样写有一个小小的问题,就是你在 mobile 下面引用一个 url helper 的时候,比如:

### app/mobile/orders/show.html.erb
<%= link_to order_path(@order), order.id %>

读代码的人比较难以直观的知道这个 order_path 是指的哪个域名下的 url,而且如果多个子域名下有 url 路径重复的话,一旦写错,rails 不会提示错误,只有访问的时候才会报错。所以,最好给 scope 加一个 as 参数,把 mobile 下的 url helper 独立出来:

### 添加as参数会修改url helper
constraints :subdomain => Rails.configuration.mobile_subdomain do
  scope module: 'mobile', as: 'mobile' do 
    resources :orders
  end
end

### 调用的时候
<%= link_to mobile_order_path(@order), order.id %>

namespace 和 scope 其实是一个东西

其实事实上,看 rails 源码可以看到,namespace 只不过是包装了一层,底层完全是用 scope 来实现的:

### File actionpack/lib/action_dispatch/routing/mapper.rb, line 679
def namespace(path, options = {})
  path = path.to_s
  options = { :path => path, :as => path, :module => path,
              :shallow_path => path, :shallow_prefix => path }.merge!(options)
  scope(options) { yield }
end

子域名的问题到这里应该已经基本说清楚了,下面讲讲其他容易遇到的问题。

rails routes 文件拆分

一旦你有了多个子域名,可能你的 routes 文件开始变大很难维护了,这个时候把 routes 文件拆分是一个好主意。

routes 文件拆分有两个办法,一种是 rails 内建的,但是 Rails4 已经移除了,一种是不受 rails 版本影响的方法。

这篇英文的文章写得挺清楚了,下面简单说一下。

修改 rails 路径配置参数

通过修改 "config/routes" 配置来解决:

config.paths["config/routes"] += Dir[Rails.root.join('config', 'routes','*.rb')]

如果对加载顺序有依赖(最好别依赖),可以一个文件一个文件的加载:

config.paths["config/routes"] = %w(
      config/routes/admin.rb
      config/routes/api.rb
      config/routes.rb
    ).map { |p| Rails.root.join(p) }

利用 instance_eval

利用 instance_eval 来加载其他路由文件即可:

Example::Application.routes.draw do
  def draw(routes_name)
    instance_eval(File.read(Rails.root.join("config", "routes", "#{routes_name}.rb")))
  end

  # subdomain
  draw :mobile
  draw :api

  ### 下面正常写其他路由就可以了
  resources :users
end

这样把其他 routes 文件放在 config/routes/ 目录下即可。

多个子域名下的_path 和_url 的使用

主要提一个 *_path 和 *_url 的区别,虽然一个是相对地址、一个是绝对地址,但是单个域名下,其实区别不大,所以很多人都是随手用。

但是一旦有了多个子域名,如果还是随手混用就会导致很多问题。所以,多个子域名下应该注意下面的几个约定:

  1. 除非必要,只用 *_path

    多个子域名下,大部分的内链还是在子域名内部的,所以尽量使用*_path 来引用 url。如果不是的话,请考虑产品设计的是否合理、是否本子域名下也需要一个独立的 url 来满足需求。

  2. 如果使用 *_url,那么一定要加上子域名的参数

    如果在跨域名访问的情况下(或者是 mailer 中),使用 *_url 的时候一定要加上 subdomain 的参数:

    <%= link_to mobile_users_path(subdomain: Rails.configuration.mobile_subdomain), users.nickname %>
    
  3. 使用 url helper,而不是字符串来代表地址。

    这一点,初级的 rails 工程师很容易犯,觉得写一个字符串非常简单,干嘛还要搞一个 url helper?可是一旦使用字符串表达 url,一旦需要重构代码、升级产品的时候基本上代码是不可维护的,这时候只能默默流泪了。

  4. javascript 代码中引用 url

    这种情况也不少,可以使用 data-url 的形式,如:

    ### 这样
    <div data-url="<%= mobile_users_path %>"></div>
    ### 或者这样
    <%= content_tag :div, :'data-url' => mobile_users_path do %>
      some content
    <% end %>
    

差不多就是这样

好,差不多就是这样,希望在多子域名的问题上对大家有帮助,欢迎大家提意见~

广告时间: 大鱼自助游还在招 ror 工程师哦,初高级均可,欢迎直接抛简历给我。

mark 正需要相关的资料

#1 楼 @leeboqiang 嗯,可以少走一点坑

赞!现在的项目大概就是这样做。写了别的路由文件,然后加载到 rails 的路由文件里面

#4 楼 @pathbox 怎么样调用别的文件里的路由?这个"别的" 指的是 不在同一个项目里面么?想知道做法。

#5 楼 @yangman_wenzhu 肯定是同一个项目里面的啊,不在同一个项目里面的路由为什么要调用?文章中有说明

备注下:子域名可定义为/(\w+).\w+.\w+$/中捕获的部分。例如 a.b.c.d.e 的子域名为 a.b.c a.b 没有子域名 a.b.c 的子域名为 c

gxlonline 把二级栏目变成二级域名,请问怎么配置好? 提及了此话题。 11月28日 09:50

碰上了一个坑,Google 了半天解决了,补充一下: 在开发环境下,Rails 默认不读取子域名,因此需要在开发环境中设置: config.action_dispatch.tld_length = 0

引用: https://gist.github.com/indiesquidge/b836647f851179589765

Insub 回复

可能是 Rails 5 的问题,这篇文章对应的是 Rails 3,没有这个问题啊

需要 登录 后方可回复, 如果你还没有账号请 注册新账号