Я сталкиваюсь с проблемой только на стороне удаленного сервера, это означает, что локально работает нормально даже через самозаверяющий сертификат через https. Но когда я перемещаю код на сервер, он также работает локально, но он не работает удаленно.
Я создал приложение узла, которое размещается на сервере через порт https 3000. Это приложение имеет Socket IO lib, который я подключил к тому же серверу https. Я не использую nodeiis, потому что я использую правила перезаписи для передачи его через Windows IIS.
Я уже установил модуль websocket в IIS. На самом деле я уже использовал websockets вместо Socket.IO, и он работал нормально с той же конфигурацией, что и сейчас, я просто заменил его на Socket.IO, потому что он лучше для того, что мне нужно.
Теперь мой код
Html Страница
<script src="/socket.io/socket.io.js"></script>
<script src="/js/Client.js"></script>
Клиент. js
$(document).ready(function() {
var address = window.location.protocol + '//' + window.location.host;
var details = {
resource: (window.location.pathname.split('/').slice(0, -1).join('/') + '/socket.io').substring(1)
};
const socket = io.connect(address, details);
socket.on('connect', function () {
console.log('Client Connected');
socket.emit('ping', 'hi server ping sent!');
});
socket.on('error', function (reason){
console.log('Connection failed', reason); //Here is where it launch the error
});
});
Приложение. js
....
const https = require('https');
var socketlib = require('./socketLib');
const fs = require('fs');
const app = express();
var cookieParser = require('cookie-parser');
app.use(sessionParser);
var expiryDate = new Date( Date.now() + 60 * 60 * 1000 );
const sessionParser = session({
secret: 'secret', resave: true, cookieName: 'sessionName',
name: 'sessionId', saveUninitialized: true,
ephemeral: true,
cookie: { secure: true, expires: expiryDate, SameSite : 'None',
maxAge: 24000 * 60 * 60, // One hour
}
});
//// HTTPS Server ////
const options = {
key: fs.readFileSync(config.certKey),
cert: fs.readFileSync(config.certCert)
};
var httpsServer = https.createServer(options, app, function (req, res) {
console.log('request starting...https');
});
httpsServer.listen(3000, function(req, res) {
console.log('Server is running at', config.nodeApiUrl + ':', port)
});
socketlib(httpsServer, sessionParser);
app.all('/*', function (req, res, next) {
res.header('Access-Control-Allow-Origin', 'https://localhost:3000')
res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE')
res.header("Access-Control-Allow-Credentials", "true");
res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept, X-Access-Token,X-Key");
return next();
})
socketLib. js
module.exports = async function(httpsServer, sessionParser) {
var io = require("socket.io")(httpsServer);
io.use(function(socket, next) {
sessionParser(socket.request, socket.request.res, next);
});
io.use((socket, next) => {
if (socket.request.headers.cookie)
return next();
next(new Error('Authentication error'));
});
io.sockets.on('connection', function(socket) {
console.log(`New connection from: ${socket.handshake.address}`)
});
}
iis webconfig
<rule name="Websocket" stopProcessing="true" enabled="true">
<match url="socket.io" ignoreCase="true" />
<action type="Rewrite" url="https://localhost:3000/{R:0}/socket.io.js" appendQueryString="false" redirectType="Found" />
<conditions>
<add input="{HTTP_HOST}" pattern="domainxxx.com" />
</conditions>
</rule>
И эту ошибку я вижу в консоли браузера
socketio.js:57 Error Error: server error
at r.onPacket (socket.io.js:7)
at o.<anonymous> (socket.io.js:7)
at o.r.emit (socket.io.js:6)
at o.r.onPacket (socket.io.js:7)
at n (socket.io.js:7)
at Object.e.decodePayload (socket.io.js:7)
at o.r.onData (socket.io.js:7)
at i.<anonymous> (socket.io.js:7)
at i.r.emit (socket.io.js:6)
at i.onData (socket.io.js:7)
, если я нажимаю в любой, если те линии, вот где мне нужно:
switch(a('socket receive: type "%s", data "%s"',t.type,t.data),this.emit("packet",t),this.emit("heartbeat"),t.type){case"open":this.onHandshake(JSON.parse(t.data));break;case"pong":this.setPing(),this.emit("pong");break;case"error":var e=new Error("server error");e.code=t.data,this.onError(e);break;
Вы увидите:
var e=new Error("server error")
Наконец, в основном, когда я пытаюсь получить доступ через domainxxx.com, это как что-то происходит где сокет не может быть подключен только удаленно, локально работает нормально, так что, я думаю, единственная проблема - после правила перезаписи. Насколько я вижу, правило перезаписи является правильным на 100%, но, очевидно, я что-то упустил.
Надеюсь, вы мне поможете! Я провел много исследований и не смог найти лекарство от этого!
Заранее спасибо!