我们使用 socket.io 进行移动服务器通信。由于我们不能强制升级用户的设备,如果我们想升级到版本 1(非向后兼容),我们必须在服务器上处理这两个版本一段时间。
有哪些选项?
我目前最喜欢的是将旧版本和新版本都包装在一个多路复用器中。它根据 header 和查询参数检测传入请求的版本,从而知道要调用哪些函数。
另一个(更糟糕的)选项是将新版本包装在一个模块中,该模块可以在必要时将协议(protocol)的旧版本转换为新版本(并再次转换回来)。这有一个严重的缺点。确保我正确确定并处理所有微小差异将是一项耗时且不确定的工作。有些差异可能需要一些认真的按摩。
(如果您感到好奇或者知道这对您有帮助,我们在 Go 中就是这样做的。)
最佳答案
看来您可以在服务器上运行两个不同版本的 socket.io。由于这两个版本没有唯一的模块文件名,您可能需要从不同的路径加载一个版本。而且,很明显,在加载模块并初始化它们时,您会将它们分配给不同名称的变量。例如:
var io_old = require('old/socket.io');
var io = require('socket.io);
一旦您将两个版本加载到服务器上,我认为有两种不同的方法可以运行它们。
1) 为每个版本使用不同的端口。旧版本将使用与 node.js 网络服务器共享的默认端口 80(不需要为此更改配置)。较新的版本将在不同的端口(例如端口 3000)上运行。然后,您会将每个版本的 socket.io 初始化为它自己的端口。然后,您的较新版本客户端将连接到运行较新版本的端口。
对于在端口 80 上运行的旧 socket.io 服务器,您将使用您已经拥有的任何可能挂接到现有 http 服务器的初始化。
对于在其他端口上运行的新 socket.io 服务器,您可以像这样单独初始化它:
var io_old = require('old/socket.io')(server);
var io = require('socket.io')(3000);
那么,在新版客户端中,连接时需要指定3000端口。
var socket = io("http://yourdomain.com:3000");
2) 为每个版本使用不同的 HTTP 请求路径。默认情况下,每个 socket.io 连接都以如下所示的 HTTP 请求开始:http://yourdomain.com/socket.io?EIO=xx&transport=xxx?t=xxx。但是,该请求的 /socket.io 部分是可配置的,并且两个不同版本的 socket.io 可以各自使用不同的路径名。在服务器上,启动 socket.io 监听的 .listen() 方法采用可选的选项对象,可以使用自定义路径配置该对象,如 path: "/socket.io-v2 “。同样,客户端中的 .connect() 方法也接受该选项对象。有点难找the documentation对于此选项,因为它实际上是一个 engine.io 选项(socket.io 使用),但 socket.io 将选项传递给 engine.io。
我自己都没有尝试过,但我研究了 socket.io 连接是如何从客户端和服务器发起的,看起来底层引擎支持这种能力,我看不出它为什么不能工作。
以下是更改服务器路径的方法:
var io = require('socket.io')(server, {path: "/socket.io.v1"});
然后,在新版本的客户端代码中,您将像这样连接:
var socket = io({path: "/socket.io.v1"});
这将导致对 HTTP URL 发出初始连接请求,如下所示:
http://yourdomain.com/socket.io.v1?EIO=xx&transport=xxx?t=xxx
这将由您的 HTTP 服务器上的不同请求处理程序处理,从而将两个版本分开。
仅供引用,也有可能 socket.io 连接 URL 中的 EIO=3 查询参数实际上是一个 engine.io 版本号,也可用于识别客户端版本和“做正确的事”基于这个值(value)观。我没有找到任何关于它如何工作的文档,甚至找不到在 engine.io 或 socket.io 源代码中查看该查询参数的位置,因此需要更多调查作为另一种可能性。
关于sockets - 让 socket.io 客户端版本落后于服务器版本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29618018/
我在使用omniauth/openid时遇到了一些麻烦。在尝试进行身份验证时,我在日志中发现了这一点:OpenID::FetchingError:Errorfetchinghttps://www.google.com/accounts/o8/.well-known/host-meta?hd=profiles.google.com%2Fmy_username:undefinedmethod`io'fornil:NilClass重要的是undefinedmethodio'fornil:NilClass来自openid/fetchers.rb,在下面的代码片段中:moduleNetclass
我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..
我在我的Rails项目中使用Pow和powifygem。现在我尝试升级我的ruby版本(从1.9.3到2.0.0,我使用RVM)当我切换ruby版本、安装所有gem依赖项时,我通过运行railss并访问localhost:3000确保该应用程序正常运行以前,我通过使用pow访问http://my_app.dev来浏览我的应用程序。升级后,由于错误Bundler::RubyVersionMismatch:YourRubyversionis1.9.3,butyourGemfilespecified2.0.0,此url不起作用我尝试过的:重新创建pow应用程序重启pow服务器更新战俘
最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru
我正在尝试修改当前依赖于定义为activeresource的gem:s.add_dependency"activeresource","~>3.0"为了让gem与Rails4一起工作,我需要扩展依赖关系以与activeresource的版本3或4一起工作。我不想简单地添加以下内容,因为它可能会在以后引起问题:s.add_dependency"activeresource",">=3.0"有没有办法指定可接受版本的列表?~>3.0还是~>4.0? 最佳答案 根据thedocumentation,如果你想要3到4之间的所有版本,你可以这
在Rails4.0.2中,我使用s3_direct_upload和aws-sdkgems直接为s3存储桶上传文件。在开发环境中它工作正常,但在生产环境中它会抛出如下错误,ActionView::Template::Error(noimplicitconversionofnilintoString)在View中,create_cv_url,:id=>"s3_uploader",:key=>"cv_uploads/{unique_id}/${filename}",:key_starts_with=>"cv_uploads/",:callback_param=>"cv[direct_uplo
这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下
如果我使用ruby版本2.5.1和Rails版本2.3.18会怎样?我有基于rails2.3.18和ruby1.9.2p320构建的rails应用程序,我只想升级ruby的版本,而不是rails,这可能吗?我必须面对哪些挑战? 最佳答案 GitHub维护apublicfork它有针对旧Rails版本的分支,有各种变化,它们一直在运行。有一段时间,他们在较新的Ruby版本上运行较旧的Rails版本,而不是最初支持的版本,因此您可能会发现一些关于需要向后移植的有用提示。不过,他们现在已经有几年没有使用2.3了,所以充其量只能让更
我安装了ruby版本管理器,并将RVM安装的ruby实现设置为默认值,这样'哪个ruby'显示'~/.rvm/ruby-1.8.6-p383/bin/ruby'但是当我在emacs中打开inf-ruby缓冲区时,它使用安装在/usr/bin中的ruby。有没有办法让emacs像shell一样尊重ruby的路径?谢谢! 最佳答案 我创建了一个emacs扩展来将rvm集成到emacs中。如果您有兴趣,可以在这里获取:http://github.com/senny/rvm.el
我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的