diff --git a/wai-server/api/channels/call.controller.js b/wai-server/api/channels/call.controller.js new file mode 100644 index 0000000..2b19fa7 --- /dev/null +++ b/wai-server/api/channels/call.controller.js @@ -0,0 +1,67 @@ +const { websocketService, websocketManager } = require('../../core'); // Import from core/index.js +const generateId = require('../../utils/generateId.util'); +// const { createConnection, getConnection } = require('../../core/whatsapp/connection'); +// const { getAvailableConnection } = require('../../core/whatsapp/sessionStore'); + +exports.newConnect = async ctx => { + // try { + // const socketId = generateId(); + // const { qr } = await createConnection(socketId); // Await the Promise here + // return { socketId, qr, message: 'Connection initiated. QR code provided.' }; // Return the QR code + // } catch (error) { + // console.error('create connection error', error); + // ctx.assert(null, 500, 'Failed to create connection or generate QR code.'); + // } +}; + +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 } = websocketService.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 = websocketService.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' }; + } +}; diff --git a/wai-server/api/channels/channel.controller.js b/wai-server/api/channels/channel.controller.js new file mode 100644 index 0000000..2c250b6 --- /dev/null +++ b/wai-server/api/channels/channel.controller.js @@ -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; +}; diff --git a/wai-server/api/channels/channel.routes.js b/wai-server/api/channels/channel.routes.js new file mode 100644 index 0000000..9bfb760 --- /dev/null +++ b/wai-server/api/channels/channel.routes.js @@ -0,0 +1,22 @@ +'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.newConnect) + .get('/qrcode0', 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; +}; diff --git a/wai-server/api/channels/index.js b/wai-server/api/channels/index.js new file mode 100644 index 0000000..c358233 --- /dev/null +++ b/wai-server/api/channels/index.js @@ -0,0 +1 @@ +module.exports = require('./channel.routes'); diff --git a/wai-server/api/contacts/contact.controller.js b/wai-server/api/contacts/contact.controller.js new file mode 100644 index 0000000..ad9a93a --- /dev/null +++ b/wai-server/api/contacts/contact.controller.js @@ -0,0 +1 @@ +'use strict'; diff --git a/wai-server/api/contacts/contact.routes.js b/wai-server/api/contacts/contact.routes.js new file mode 100644 index 0000000..819110d --- /dev/null +++ b/wai-server/api/contacts/contact.routes.js @@ -0,0 +1,20 @@ +'use strict'; + +// const controller = require('./contact.controller'); + +module.exports = Router => { + const router = new Router({ + prefix: `/contacts`, + }); + + // 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; +}; diff --git a/wai-server/api/contacts/index.js b/wai-server/api/contacts/index.js new file mode 100644 index 0000000..ca67d2a --- /dev/null +++ b/wai-server/api/contacts/index.js @@ -0,0 +1 @@ +module.exports = require('./contact.routes'); diff --git a/wai-server/api/group/group.controller.js b/wai-server/api/group/group.controller.js new file mode 100644 index 0000000..ad9a93a --- /dev/null +++ b/wai-server/api/group/group.controller.js @@ -0,0 +1 @@ +'use strict'; diff --git a/wai-server/api/group/group.routes.js b/wai-server/api/group/group.routes.js new file mode 100644 index 0000000..7050e93 --- /dev/null +++ b/wai-server/api/group/group.routes.js @@ -0,0 +1,20 @@ +'use strict'; + +// const controller = require('./channel.controller'); + +module.exports = Router => { + const router = new Router({ + prefix: `/group`, + }); + + // 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; +}; diff --git a/wai-server/api/group/index.js b/wai-server/api/group/index.js new file mode 100644 index 0000000..1c4317e --- /dev/null +++ b/wai-server/api/group/index.js @@ -0,0 +1 @@ +module.exports = require('./group.routes'); diff --git a/wai-server/api/index.js b/wai-server/api/index.js index 67a8a19..06822fa 100644 --- a/wai-server/api/index.js +++ b/wai-server/api/index.js @@ -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 diff --git a/wai-server/api/messages/index.js b/wai-server/api/messages/index.js new file mode 100644 index 0000000..163eadf --- /dev/null +++ b/wai-server/api/messages/index.js @@ -0,0 +1 @@ +module.exports = require('./message.routes'); diff --git a/wai-server/api/messages/message.controller.js b/wai-server/api/messages/message.controller.js new file mode 100644 index 0000000..47c553b --- /dev/null +++ b/wai-server/api/messages/message.controller.js @@ -0,0 +1,62 @@ +'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; +}; + +exports.sendText = async ctx => { + const { type } = ctx.params; + const body = ctx.request.body; + ctx.assert(null, 400, `debug: content: {${type}} {${body.content}}`); + ctx.assert(null, 400, 'The message info is malformed!'); + return body; +}; diff --git a/wai-server/api/messages/message.routes.js b/wai-server/api/messages/message.routes.js new file mode 100644 index 0000000..0561e27 --- /dev/null +++ b/wai-server/api/messages/message.routes.js @@ -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('/:type', controller.sendText); + + return router; +}; diff --git a/wai-server/config/components/websocket.config.js b/wai-server/config/components/websocket.config.js new file mode 100644 index 0000000..ee873da --- /dev/null +++ b/wai-server/config/components/websocket.config.js @@ -0,0 +1,26 @@ +'use strict'; + +const serverNodes = Array.from({ length: 2 }, (_, 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=${i ? '383' : '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; diff --git a/wai-server/config/components/whatsapp.baileys.config.js b/wai-server/config/components/whatsapp.baileys.config.js new file mode 100644 index 0000000..b031af3 --- /dev/null +++ b/wai-server/config/components/whatsapp.baileys.config.js @@ -0,0 +1,12 @@ +'use strict'; + +const serverNodes = Array.from({ length: 2 }, (_, i) => ({ + name: `wai-server-${i + 1}`, + // host: `ws://remote-websocket-service-${i + 1}.com`, +})); + +const config = { + whatsappBaileys: serverNodes, +}; + +module.exports = config; diff --git a/wai-server/core/index.js b/wai-server/core/index.js new file mode 100644 index 0000000..2e41585 --- /dev/null +++ b/wai-server/core/index.js @@ -0,0 +1,12 @@ +// core/index.js +const websocketServicesI = require('./websocket/services/session'); +const websocketConnectionI = require('./websocket/connection'); + +// Create the instances here +const websocketService = websocketServicesI(); +const websocketManager = websocketConnectionI(websocketService); + +module.exports = { + websocketService, + websocketManager, +}; diff --git a/wai-server/core/websocket/connection.js b/wai-server/core/websocket/connection.js new file mode 100644 index 0000000..3961253 --- /dev/null +++ b/wai-server/core/websocket/connection.js @@ -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}`); + // 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, + }; +}; diff --git a/wai-server/core/websocket/services/session.js b/wai-server/core/websocket/services/session.js new file mode 100644 index 0000000..b3744ee --- /dev/null +++ b/wai-server/core/websocket/services/session.js @@ -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, + }; +}; diff --git a/wai-server/index.js b/wai-server/index.js index 15176f2..e33169a 100644 --- a/wai-server/index.js +++ b/wai-server/index.js @@ -4,6 +4,7 @@ const http = require('http'); const server = require('./server'); const { port } = require('./config').server; +const { websocketService, 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:'); diff --git a/wai-server/middleware/request.middleware.js b/wai-server/middleware/request.middleware.js index 374cc24..074f295 100644 --- a/wai-server/middleware/request.middleware.js +++ b/wai-server/middleware/request.middleware.js @@ -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, }; } diff --git a/wai-server/server.js b/wai-server/server.js index a06b9cf..e927807 100644 --- a/wai-server/server.js +++ b/wai-server/server.js @@ -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();