3

我让 Action Cable 在本地主机环境中工作,在这种情况下,我使用一个包含以下内容的简单文件启动 Puma 服务器

# /bin/bash
bundle exec puma -p 28080 cable/config.ru

一旦发生这种情况,puma 服务器就会启动并监听这个 28080 端口和本地服务器正在运行的端口。通过在线搜索,我找不到一个地方可以告诉我一种在heroku上模拟它的方法,或者一种让我的服务器始终在同一个端口上启动的方法(尽管我不知道这是否会给我想要的结果)

我设置了一个 javascript 文件来创建与该端口相关的使用者。

//= require cable
//= require_self
//= require_tree .


this.App = {};

App.cable = Cable.createConsumer('ws://127.0.0.1:28080');

我想我还需要更改 127.0.0.1 部分,以便部署到 heroku 也能正常工作,但我不确定。我试图切断 28080 部分并将其替换为 ENV['PORT'] 但它说它是一个未知变量,即使我设置了一个 puma.rb 文件,其端口设置为

... (only part of the file)
rackup      DefaultRackup
port        ENV['PORT']     || 3000
environment ENV['RACK_ENV'] || 'development'
...

所以在我看来 ENV['PORT'] 被定义为当我检查 heroku 日志时 puma 服务器将是

2015-07-26T06:50:25.278030+00:00 heroku[web.1]: Starting process with command `bin/rails server -p 48875 -e production`
2015-07-26T06:50:30.760680+00:00 app[web.1]: => Booting Puma
2015-07-26T06:50:30.760714+00:00 app[web.1]: => Rails 4.2.1 application starting in production on http://0.0.0.0:48875
2015-07-26T06:50:30.760716+00:00 app[web.1]: => Run `rails server -h` for more startup options
2015-07-26T06:50:30.760718+00:00 app[web.1]: => Ctrl-C to shutdown server
2015-07-26T06:50:31.578843+00:00 app[web.1]: Puma 2.12.2 starting...
2015-07-26T06:50:31.578851+00:00 app[web.1]: * Min threads: 0, max threads: 16
2015-07-26T06:50:31.578859+00:00 app[web.1]: * Environment: production
2015-07-26T06:50:31.578861+00:00 app[web.1]: * Listening on tcp://0.0.0.0:48875

如果有任何不清楚的地方,我深表歉意,如果我遗漏了什么,我很乐意提供更多信息。

编辑

这是 /app/assets/javascripts/channels/index.js.erb 中的更新代码

//= require cable
//= require_self
//= require_tree .

this.App = {};

App.cable = Cable.createConsumer('<%= ENV["CABLE_SERVER"] %>');

其中 ENV["CABLE_SERVER"] 指向ws://the-action-cable-server.herokuapp.com. 此变量存储在 rails 服务器环境变量中。

4

1 回答 1

9

问题

Heroku路由器有一些限制:它只能侦听端口 80 和 443。换句话说,您不能在任何 Heroku 应用程序上打开固定端口。在 ActionCable 服务器的情况下,没有办法打开一个固定端口并将 websocket 流量路由到它。所以要么 Heroku 允许这样的事情(我怀疑),要么我们使用一种解决方法。

解决方法

从 actioncable 的 0.0.3 版开始,这是我使用的解决方法。

我们的想法是拥有两个 Heroku 应用程序,而不是一个:一个用于主 rails 服务器,一个用于 ActionCable 服务器。两者都将在端口 80(或 443)上运行。

从一个代码库运行两个不同的服务器只需拥有两个 Procfile:一个用于 rails,一个用于 action cable。有一个buildpack可以处理这个问题。要使用它,您还需要multi buildpack

假设您有两个名为rails和的 Heroku 应用程序actioncable

使用以下命令在项目的根目录中创建一个.buildpacks文件:

https://github.com/cantino/heroku-selectable-procfile
https://github.com/heroku/heroku-buildpack-ruby

railsand上actioncable,创建一个环境变量BUILDPACK_URLhttps://github.com/heroku/heroku-buildpack-multi.git

现在对于 Procfiles,我选择继续Procfile使用 foreman 在本地运行所有内容并创建两个自定义文件:Procfile.railsProcfile.actioncable.

Procfile.rails中,您描述了除了动作电缆服务器之外的所有需要​​的测功机,例如:

web: bundle exec puma -C config/puma/config.rb
clockwork: bundle exec clockwork lib/clockwork.rb
worker: bundle exec rake jobs:work

Procfile.actioncable中,您描述了动作电缆服务器:

web: bundle exec puma -p $PORT cable/config.ru

请注意,我们使用的是web测功机,它将将动作电缆服务器安装在端口 80(或 443)上。

注意您需要将 puma 配置文件移动config/puma.rb到自定义位置。我选择config/puma/config.rbconfig/puma.rb是在没有任何特定配置文件的情况下启动 puma 时的默认位置,这就是我们在Procfile.actioncable. 这可能会导致意外行为(请参阅下面的评论)。

现在,在 上rails,创建一个 env varPROCFILE_PATHProcfile.railson ,actioncable创建一个 env var PROCFILE_PATHProcfile.actioncable

说到环境变量,actioncable需要所有环境变量rails来启动可操作的服务器,如 DATABASE_URL 或任何凭据。

现在是关键的一步:我们如何连接railsactioncable在一起?这只需使用相同的Redis 实例即可完成。rails会将消息放在 Redis 上,并actioncable会听取它们并采取相应的行动。这就是为什么两者都必须针对同一个Redis 实例。如果你使用 Heroku Redis,你只需要在REDIS_URL上设置相同的值。这是有线服务器的配置文件:rails actioncablecable.yml

production: &production
  :url: <%= ENV["REDIS_URL"] %>
  :timeout: 1
development: &development
  :url: <%= ENV["REDIS_URL"] %>
  :timeout: 1
  :inline: true
test: *development

最后一步是更改 javascript 文件,以便我们可以控制 Action Cable 服务器的位置。我们可以通过使用 env var 来做到这一点。

如有必要,将.js后缀更改为。.js.erb

//= require cable
//= require_self
//= require_tree .


this.App = {};

App.cable = Cable.createConsumer('<%= ENV["CABLE_SERVER"] %>');

CABLE_SERVER变量现在可以指向ws://127.0.0.1:28080本地和 on rails,其值将是 的 url actioncable

现在您已准备好在 和 上部署rails代码actioncable

注意事项/缺点

  • on actioncable,如果您有任何客户端身份验证,则不能cookies示例中使用 like 。您现在有两个 Heroku 应用程序,它们不能共享 cookie。我想你仍然可以使用多个子域的 cookie 来解决这个问题。
  • 您现在有两个部署目标需要维护。
  • 你有多个 Procfile 需要维护。
  • 希望随着时间的推移,会有一个更简单的解决方法:)

备择方案

  • 我们可以使用一个简单的中间件让同一台服务器处理正常的 Web 流量websocket 流量。这里是如何。
  • 从 rails 5 beta2 开始,action cable 服务器现在也可以与您的主要 rails 应用程序一起侧面安装。拥有两个不同的服务器仍然有意义:您可以单独扩展它们。

hth

于 2015-07-28T12:48:50.867 回答