node.js - long - socket io query




socket.ioにはhttpとhttpsの両方を使用してください。 (4)

おそらく、HTTPおよびHTTPS用のnode.jsサーバーオブジェクトには、SSLを使用しても使用しなくても、任意の数のポートおよびインターフェイスをリッスンする機能が与えられるべきですが、これは現在実装されていないようです。 (server.listen(handle、[callback])インタフェースへの "handle"引数としてリクエストリスナを持たない2番目のサーバをserver.listen(port)に渡すことで、1つのサーバに2つのポートをリッスンさせることができました。 、[ホスト名]、[バックログ]、[コールバック])、ただし、SSL /非SSLサーバーが混在している場合は機能しませんでした。

すでに言及した風洞の回避策はもちろん実行可能なオプションですが、(node.js以外の依存関係を避けるために)別のソフトウェアをインストールすることが望ましくない場合は、代わりにnode.jsでも同じトンネリングを実現できます。ポート80のHTTPおよびポート443のHTTPS)

var fs = require('fs');
var net = require('net');
var tls = require('tls');
var sslOptions = {
    key: fs.readFileSync('server-key.pem'),
    cert: fs.readFileSync('server-cert.pem')
};
tls.createServer(sslOptions, function (cleartextStream) {
    var cleartextRequest = net.connect({
        port: 80,
        host: '127.0.0.1'
    }, function () {
        cleartextStream.pipe(cleartextRequest);
        cleartextRequest.pipe(cleartextStream);
    });
}).listen(443);

これはstunnelを使うのと同じ効果があります。 つまり、node.jsの "https"モジュールを冗長にしながら、2つの別々のsocket.ioサーバーインスタンスが不要になります。

socket.ioをhttphttps両方の接続で機能させようとしていますが、私の設定ではどちらか一方でしか機能しないようです。

下記の設定オプションを使うと、socket.ioはhttpsを通して私のアプリケーションにアクセスできますが、通常のhttpを通してアクセスしようとするとそれは接続できず、私はエラーを受け取ります。

    var app = express()
  , http= require('http').createServer(app)
  , https = require('https').createServer(options, app)
  , io = require('socket.io').listen(https, { log: false })

そして後で私はこれを持っている

http.listen(80, serverAddress);
https.listen(443, serverAddress);

クライアント側で私はこれを持っています:

<script src='/socket.io/socket.io.js'></script>

var socket = io.connect('https://<%= serverAddress %>', {secure: true, 'sync disconnect on unload' : true});

もちろん、サーバとクライアントの.connect機能と.connect機能でそれぞれhttpsオプションを使用してhttpを切り替えると、逆の結果になります。 Socket.ioはhttps経由ではなくhttp経由でアクセスできます。

どのようにしてこれを達成することが可能ですか? それはFacebookアプリに関するものなので、私はほとんどそれを必要とします、それはそれがFacebookのルールに従ってhttpとhttps両方の接続オプションを提供しなければなりません。

編集:それは問題について役立つ場合には、私が受け取っているエラーは以下の通りです

Failed to load resource: the server responded with a status of 404 (Not Found) http://DOMAIN/socket.io/socket.io.js

そしてこれのために私は他のものを得る:

Uncaught ReferenceError: io is not defined 

クロスオリジンリクエストがあなたがエラーを受けている理由かもしれないと思います。 プロトコルの変更はドメインの変更と見なされます。 そのため、 httpサーバー経由でhttpsサーバーにアクセスしているページ(それに接続されているwebsocketサーバー)では、セキュリティエラーが発生する可能性があります。 ExpressでCORSを有効にする方法については、 hereの例を参照してください。

また、ヘッダーの*http://serverAddress , https://serverAddressます。 すべてのサイトを許可することはお勧めできません。テストに使用してください。

httpサーバーとhttpsサーバーの間でAJAXを実行しようとしている場合も同様です。 念のため、エラーを投稿してください。


私は別の方法で問題を解決し、暗号化されていないトランスポートのみをサポートするようにサーバーを構成し、httpsサポートにはstunnelを使用しました。

stunnelをインストールする方法については、このpostをチェックしてください。

そして、次のようなcon設定を使います。

#/etc/stunnel/stunnel.conf
cert = /var/www/node/ssl/encryption.pem
[node]
accept = 443
connect = 80

最後に、クライアントを接続するために次のものを使用しました。

var socket = that.socket = io.connect('//'+server);

これはブラウザのスキーム自動検出し、それに応じてhttp / httpsを使用して接続します。


私は問題がサーバー側クライアント側でsocket.ioを設定するあなたの方法にあると思います。

これが私がそれを機能させる方法です(あなたのためだけに)。

サーバ:

var debug = require('debug')('httpssetuid');
var app = require('../app');
var http = require('http');
var https = require('https');
var fs = require('fs');
var exec = require('child_process').exec;
var EventEmitter = require('events').EventEmitter;
var ioServer = require('socket.io');

var startupItems = [];
startupItems.httpServerReady = false;
startupItems.httpsServerReady = false;

var ee = new EventEmitter();

ee.on('ready', function(arg) {
  startupItems[arg] = true;
  if (startupItems.httpServerReady && startupItems.httpsServerReady) {
    var id = exec('id -u ' + process.env.SUDO_UID, function(error, stdout, stderr) {
      if(error || stderr) throw new Error(error || stderr);
      var uid = parseInt(stdout);
      process.setuid(uid);
      console.log('de-escalated privileges. now running as %d', uid);
      setInterval(function cb(){
        var rnd = Math.random();
        console.log('emitting update: %d', rnd);
        io.emit('update', rnd);
      }, 5000);
    });
  };
});

app.set('http_port', process.env.PORT || 80);
app.set('https_port', process.env.HTTPS_PORT || 443);

var httpServer = http.createServer(app);

var opts = {
  pfx: fs.readFileSync('httpssetuid.pfx')
};
var httpsServer = https.createServer(opts, app);

var io = new ioServer();

httpServer.listen(app.get('http_port'), function(){
  console.log('httpServer listening on port %d', app.get('http_port'));
  ee.emit('ready', 'httpServerReady');
});

httpsServer.listen(app.get('https_port'), function(){
  console.log('httpsServer listening on port %d', app.get('https_port'));
  ee.emit('ready', 'httpsServerReady');
});

io.attach(httpServer);
io.attach(httpsServer);

io.on('connection', function(socket){
  console.log('socket connected: %s', socket.id);
});

クライアント:

script(src='/socket.io/socket.io.js')
script.
  var socket = io();
  socket.on('update', function(update){
    document.getElementById('update').innerHTML = update;
  });

サーバーのキーポイントは次のとおりです。

  1. socket.ioが必要ですが、まだlistenメソッドを呼び出してはいけません(httpとhttpsがすでに必要であると仮定して)。 代わりに、単に参照を守ってください。 (var ioServer = require( 'socket.io'))
  2. http&httpsサーバーを作成する
  3. ioServerの新しいインスタンスを作成します。
  4. httpサーバーとhttpsサーバーをバインドします(.listen)
  5. httpおよびhttpsサーバーインスタンスをioインスタンスに接続します。 (.listenは.attachの別名です)
  6. ioイベントを設定します。

そしてクライアント(翡翠構文だがあなたはアイデアを得る):

  1. socket.ioスクリプトタグを含める
  2. ioを呼び出して参照をキャプチャする
  3. イベントハンドラを設定する

クライアントではio.connect()を呼び出す必要はありません。 さらに、私はあなたの選択についてそこに確信がありません。 それはあなたがタイプミス(、、)を持っているように見えます、そして私はsecureへの言及を見つけることができません:1.0ドキュメントのtrue。







socket.io