Compare commits

...

2 Commits

Author SHA1 Message Date
lyt 6f1cc69031 websocket connect 6 months ago
Lei OT e2733bfee7 channel 路由 6 months ago

@ -0,0 +1,55 @@
// const callService = require('../../core/services/call');
// const websocketManager = require('../../core/websocket/manager');
const { callService, websocketManager } = require('../../core'); // Import from core/index.js
exports.getIn = async ctx => {
// Wait for at least one connection to be established (or handle the case where no connections are available)
// await new Promise(resolve => {
// const checkConnections = () => {
// if (websocketManager.getConnections().length > 0) {
// resolve();
// } else {
// setTimeout(checkConnections, 100); // Check again after 100ms
// }
// };
// checkConnections();
// });
const availableWs = websocketManager.getAvailableConnection();
if (!availableWs) {
// ctx.status = 503;
// ctx.body = { message: 'No available connections' };
ctx.assert(availableWs, 503, 'No available connections');
return;
}
// return availableWs;
const { sessionId, url } = callService.createSession(availableWs);
return { sessionId, url, message: 'Connection established' }; // availableWs
};
exports.sendMsg = ctx => {
const { sessionId, message } = ctx.request.body;
if (!sessionId || !message) {
// ctx.status = 400;
// ctx.body = { message: 'Session ID and message are required' };
ctx.assert(sessionId, 400, 'Session ID and message are required');
return;
}
const wsToSend = callService.getSession(sessionId);
if (!wsToSend) {
// ctx.status = 404;
// ctx.body = { message: 'Session not found' };
ctx.assert(wsToSend, 400, 'Session not found');
return;
}
try {
wsToSend.send(message);
ctx.body = { message: 'Message sent' };
} catch (error) {
console.error('Error sending message:', error);
ctx.status = 500;
ctx.body = { message: 'Failed to send message' };
}
};

@ -0,0 +1,52 @@
'use strict';
const generateId = require('../../utils/generateId.util');
/**
* Mock database, replace this with your db models import, required to perform query to your database.
*/
const db = {
channels: [
{
channelId: 1,
phone: 'string',
formattedPhone: 'string',
profilePicture: 'string',
name: 'string',
connectStatus: 'string',
channelStatus: 'string',
createTime: 0,
},
],
};
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
exports.qrcode = async ctx => {
return ``;
};
exports.getOne = async ctx => {
const { id } = ctx.params;
const channel = db.channels.find(channel => channel.channelId === Number(id));
await sleep(500);
ctx.assert(channel, 200, "The requested channel doesn't exist");
return channel;
};
exports.getAll = async ctx => {
return db.channels;
};
exports.createOne = async ctx => {
const { name } = ctx.request.body;
ctx.assert(name, 200, 'The channel info is malformed!');
const id = generateId();
const newChannel = {
id,
name,
timestamp: Date.now(),
};
db.channels.push(newChannel);
const createdChannel = db.channels.find(channel => channel.id === id);
return createdChannel;
};

@ -0,0 +1,21 @@
'use strict';
const controller = require('./channel.controller');
const callController = require('./call.controller');
module.exports = Router => {
const router = new Router({
prefix: `/channels`,
});
router
.get('/qrcode', callController.getIn)
// .get('/qrcode', controller.qrcode)
.get('/:id', controller.getOne)
.get('/', controller.getAll)
.post('/', controller.createOne);
// router.post('/get-in', callController.getIn);
// router.post('/send-msg', callController.sendMsg);
return router;
};

@ -0,0 +1 @@
module.exports = require('./channel.routes');

@ -9,7 +9,7 @@ const baseName = path.basename(__filename);
function applyApiMiddleware(app) {
const router = new Router({
prefix: `/api/${apiVersion}`,
prefix: `/api/${apiVersion}/whatsapp`,
});
// Require all the folders and create a sub-router for each feature api

@ -0,0 +1 @@
module.exports = require('./message.routes');

@ -0,0 +1,54 @@
'use strict';
const generateId = require('../../utils/generateId.util');
/**
* Mock database, replace this with your db models import, required to perform query to your database.
*/
const db = {
users: [
{
id: 'bff28903-042e-47c2-b9ee-07c3954989ec',
name: 'Marco',
created_at: 1558536830937,
},
{
id: 'dca01a32-36e6-4886-af75-8e7caa0162a9',
name: 'Leonardo',
created_at: 1558536843742,
},
{
id: 'dca01a32-36e6-4886-af75-8e7caa0162a9',
name: 'Berta',
created_at: 1558536863550,
},
],
};
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
exports.getOne = async ctx => {
const { id } = ctx.params;
const user = db.users.find(user => user.id === id);
await sleep(500);
ctx.assert(user, 404, "The requested user doesn't exist");
return user;
};
exports.getAll = async ctx => {
return db.users;
};
exports.createOne = async ctx => {
const { name } = ctx.request.body;
ctx.assert(name, 400, 'The user info is malformed!');
const id = generateId();
const newUser = {
id,
name,
timestamp: Date.now(),
};
db.users.push(newUser);
const createdUser = db.users.find(user => user.id === id);
return createdUser;
};

@ -0,0 +1,16 @@
'use strict';
const controller = require('./message.controller');
module.exports = Router => {
const router = new Router({
prefix: `/messages`,
});
router
.get('/:id', controller.getOne)
.get('/', controller.getAll)
.post('/', controller.createOne);
return router;
};

@ -0,0 +1,26 @@
'use strict';
const serverNodes = Array.from({ length: 1 }, (_, i) => ({
name: `wai-server-${i + 1}`,
port: 8080 + i,
// host: `ws://remote-websocket-service-${i + 1}.com`,
host: `wss://p9axztuwd7x8a7.mycht.cn/whatsapp_server?opisn=404&_spam=${Date.now().toString()}`,
protocol: 'WhatsApp',
// path: '/ws',
// protocol: 'ws',
// subprotocol: 'protoo',
// maxPayload: 1048576, // 1MB
// allowRequest: (req, next) => next(null, true),
// compression: true,
// json: true,
// binary: true,
// heartbeatInterval: 5000,
// heartbeatTimeout: 10000,
// closeTimeout: 5000,
}));
const config = {
websockets: serverNodes,
};
module.exports = config;

@ -0,0 +1,12 @@
// core/index.js
const createCallService = require('./services/call');
const createWebsocketManager = require('./websocket/manager');
// Create the instances here
const callService = createCallService();
const websocketManager = createWebsocketManager(callService);
module.exports = {
callService, // Export the INSTANCE
websocketManager, // Export the INSTANCE
};

@ -0,0 +1,31 @@
const generateId = require('../../utils/generateId.util');
module.exports = () => {
const sessions = new Map();
const createSession = ws => {
const sessionId = generateId();
sessions.set(ws, sessionId);
return { sessionId, url: ws.url };
};
const getSession = sessionId => {
for (const [ws, storedSessionId] of sessions) {
if (storedSessionId === sessionId) {
return ws;
}
}
return null;
};
const removeSession = ws => {
sessions.delete(ws);
};
return {
createSession,
getSession,
removeSession,
sessions,
};
};

@ -0,0 +1,60 @@
const WebSocket = require('ws');
const websocketServerNodes = require('../../config').websockets;
module.exports = callService => {
let connections = [];
const connect = () => {
websocketServerNodes.forEach((node, index) => {
createConnection(node, index);
});
};
const createConnection = (node, index) => {
try {
const ws = new WebSocket(node.host, [node.protocol]);
ws.on('open', () => {
console.log(`Connected to WebSocket ${index}: ${node.protocol} ${node.host} ${node.protocol}`);
// connections.push(ws);
connections = [...connections, ws];
console.log('Current connections:', connections.length);
});
ws.on('close', () => {
console.log(`WebSocket ${index} disconnected. Reconnecting...`);
connections.splice(connections.indexOf(ws), 1);
setTimeout(() => createConnection(node, index), 3000);
});
ws.on('error', error => {
console.error(`WebSocket ${index} error:`, error);
});
ws.on('message', message => {
console.log(`Received from WebSocket ${index}: ${message}`);
});
} catch (error) {
console.error(`Error connecting to WebSocket ${index}:`, error);
}
};
const getAvailableConnection = () => {
// console.log('Available connections length:', connections.length);
for (const ws of connections) {
// console.log('Session has ws:', callService.sessions.has(ws));
if (!callService.sessions.has(ws)) {
// console.log('Found available ws:', ws); // Add this line
return ws;
}
}
console.log('No available connections found.');
return null;
};
const getConnections = () => connections; // Add this getter
return {
connect,
getAvailableConnection,
getConnections,
};
};

@ -4,6 +4,7 @@ const http = require('http');
const server = require('./server');
const { port } = require('./config').server;
const { callService, websocketManager } = require('./core');
async function bootstrap() {
/**
@ -15,7 +16,10 @@ async function bootstrap() {
}
bootstrap()
.then(server => console.log(`🚀 Server listening on port ${server.address().port}!`))
.then(server => {
console.log(`🚀 Server listening on port ${server.address().port}!`);
websocketManager.connect(); // connect websocket after server start
})
.catch(err => {
setImmediate(() => {
console.error('Unable to run the server because of the following error:');

@ -11,10 +11,11 @@ module.exports = async (ctx, next) => {
}
} catch (err) {
console.log('Error handler:', err);
ctx.status = err.status || 500;
// ctx.status = 200; // err.status || 500;
ctx.body = {
errcode: 1,
errmsg: err.message || 'Internal server error',
statuscode: err.status,
result: null,
};
}

@ -14,7 +14,7 @@ const applyApiMiddleware = require('./api');
* Add here only development middlewares
*/
if (isDevelopment) {
server.use(logger);
// server.use(logger);
// Logger
server.use(async (ctx, next) => {
const start = new Date();

Loading…
Cancel
Save