为什么我的GraphQL订阅在本地服务器上工作,但在实时服务器上部署时却不能



我正在构建一个GraphQL服务器,并在其中进行订阅。一旦我在本地机器上部署服务器并在Playground中检查订阅,它就会工作,即它会侦听事件并获得新添加的数据。这意味着订阅实现是正确的,它们没有任何问题。当我在本地(即localhost(上运行时,订阅工作正常,但当我在live(Heroku(上部署服务器时,当我在Playground:中侦听订阅时,会出现以下错误

{
"error": "Could not connect to websocket endpoint wss://foodesk.herokuapp.com/graphql. Please check if the endpoint url is correct."
}

只是为了了解更多信息,我的查询和突变也在实时服务器上工作,只是订阅不起作用。

这是我的代码:

/**
* third party libraries
*/
const bodyParser = require('body-parser');
const express = require('express');
const { ApolloServer } = require('apollo-server-express');
const helmet = require('helmet');
const http = require('http');
const mapRoutes = require('express-routes-mapper');
const mkdirp = require('mkdirp');
const shortid = require('shortid');
/**
* server configuration
*/
const config = require('../config/');
const auth = require('./policies/auth.policy');
const dbService = require('./services/db.service');
const { schema } = require('./graphql');
// environment: development, testing, production
const environment = process.env.NODE_ENV;

const graphQLServer = new ApolloServer({
schema,
uploads: false
});
/**
* express application
*/
const api = express();
const server = http.createServer(api);
graphQLServer.installSubscriptionHandlers(server)
const mappedRoutes = mapRoutes(config.publicRoutes, 'api/controllers/');
const DB = dbService(environment, config.migrate).start();
// allow cross origin requests
// configure to allow only requests from certain origins
api.use(function (req, res, next) {
// Website you wish to allow to connect
res.setHeader('Access-Control-Allow-Origin', '*');
// Request methods you wish to allow
res.setHeader('Access-Control-Allow-Methods', '*');
// Request headers you wish to allow
res.setHeader('Access-Control-Allow-Headers', '*');
// Pass to next layer of middleware
next();
});
// secure express app
api.use(helmet({
dnsPrefetchControl: false,
frameguard: false,
ieNoOpen: false,
}));
// parsing the request bodys
api.use(bodyParser.urlencoded({ extended: false }));
api.use(bodyParser.json());
// public REST API
api.use('/rest', mappedRoutes);
// private GraphQL API
api.post('/graphql', (req, res, next) => auth(req, res, next));
graphQLServer.applyMiddleware({
app: api,
cors: {
origin: true,
credentials: true,
methods: ['POST'],
allowedHeaders: [
'X-Requested-With',
'X-HTTP-Method-Override',
'Content-Type',
'Accept',
'Authorization',
'Access-Control-Allow-Origin',
],
},
playground: {
settings: {
'editor.theme': 'light',
},
},
});
server.listen(config.port, () => {
if (environment !== 'production'
&& environment !== 'development'
&& environment !== 'testing'
) {
console.error(`NODE_ENV is set to ${environment}, but only production and development are valid.`);
process.exit(1);
}
return DB;
});

基于您的错误;

{
"error": "Could not connect to websocket endpoint wss://foodesk.herokuapp.com/graphql. Please check if the endpoint url is correct."
}

您的应用程序正在heroku应用程序服务上运行,即";herokuapp.com";。就此服务而言,您无法控制服务器配置。

在开发过程中,您根据";ws://";被转换为安全版本的协议";wss://";当你部署到heroku时。我的解决方案需要您通过ssh访问服务器。这意味着您可以升级您的heroku订阅,以拥有一个具有专用ip地址的vm或任何其他云提供商。如果你已经准备好了,请执行以下操作;

  1. 在vm上托管您的应用程序并为其提供服务,然后编辑apache以代理"yourdomain.com";在3000端口为应用程序提供服务,如下所示
<VirtualHost *:80>
ServerName yourdomain.com
ProxyPreserveHost on
ProxyPass / http://localhost:3000/
ProxyPassReverse / http://localhost:3000/
RewriteEngine on

</VirtualHost>
  1. 转到您的域注册商,添加一个子域,您将通过该子域为您的订阅(也称为网络套接字(提供服务。这是通过添加具有子域名称例如"a"的a记录来完成的;websocket";指向你的vm ip地址。这将使";websocket.yourdomain.com";可供使用。

  2. 在您的服务器上安装apache服务器,并包括如下所示的虚拟主机

<VirtualHost *:80>
ServerName websocket.yourdomain.com
ProxyPreserveHost on
ProxyPass / ws://localhost:3000/
ProxyPassReverse / ws://localhost:3000/
RewriteEngine on

</VirtualHost>
  1. 重新启动apache服务器

此时,您的应用程序正在";yourdomain.com";这也是提供所有graphql突变和查询的地方。但是所有的订阅都将在"0"建立;websocket.yourdomain.com";它将把所有请求代理到";ws://";协议,从而在没有错误的情况下到达您的订阅服务器。

N/B您的客户端代码将使用"wss://websocket.yourdomain.com"用于订阅连接

最新更新