RSS Git Download  Clone
Raw Blame History 5kB 129 lines
import fs from 'fs'
import path from 'path'
import { fileURLToPath } from 'url'
import * as socketIoShared from './shared.mjs'
import { isSnapshot, version } from '../../lib/resolve-version.mjs'

// Auto-discover request handlers from request/$area/$function.mjs
const __dirname = path.dirname(fileURLToPath(import.meta.url))
const requestDir = path.join(__dirname, 'request')
const validActions = new Set()
for (const entry of fs.readdirSync(requestDir, { withFileTypes: true })) {
    if (!entry.isDirectory()) continue
    for (const file of fs.readdirSync(path.join(requestDir, entry.name), { withFileTypes: true })) {
        if (!file.isFile() || !file.name.endsWith('.mjs')) continue
        validActions.add(`${entry.name}/${file.name.slice(0, -4)}`)
    }
}
console.info(`socket.io discovered ${validActions.size} request handlers:`, [...validActions].sort().join(', '))

export default (io) => {

    io.on('connect', function (socket) {

        //const token = socket.handshake.query.token;
        socket.p3xrs = {
            address: socket.handshake.headers.origin,
            connectedAt: new Date(),
            connectionId: undefined,
            io: io,
            ioredis: undefined,
            ioredisSubscriber: undefined,
            tunnels: [],
            sshClient: undefined,
            readonly: undefined,
            // commands: undefined,
            subsciber: false,
        }

        console.info(`socket.io connected ${socket.id}`);

        socket.on('disconnect', function () {
            console.warn('socket.p3xrs.connectionId', socket.p3xrs.connectionId)
            if (socket.p3xrs.connectionId !== undefined) {
                const connectionId = socket.p3xrs.connectionId;
                if (p3xrs.redisConnections.hasOwnProperty(connectionId)) {
                    const redisConnectionIndex = p3xrs.redisConnections[connectionId].clients.indexOf(socket.id);
                    if (redisConnectionIndex !== -1) {
                        p3xrs.redisConnections[connectionId].clients.splice(redisConnectionIndex, 1);
                    }
                    if (p3xrs.redisConnections[connectionId].clients.length === 0) {
                        delete p3xrs.redisConnections[connectionId]

                    }
                    socketIoShared.disconnectRedisIo({
                        socket: socket,
                    })
                }
            }

            // Stop MONITOR if active
            if (socket.p3xrs.ioredisMonitor) {
                for (const monitor of socket.p3xrs.ioredisMonitor) {
                    try { monitor.disconnect() } catch {}
                }
                socket.p3xrs.ioredisMonitor = undefined
            }

            // Call on disconnect.
            console.info('socket.io disconnected %s', socket.id);
            socketIoShared.sendStatus({
                socket: socket,
            })
            socketIoShared.disconnectRedis({
                socket: socket,
            })
        });

        socket.on('p3xr-request', (options) => {
            options.socket = socket;
            options.responseEvent = `p3xr-response-${options.requestId}`
            if (options?.action && typeof options.action === 'string' && validActions.has(options.action)) {
                import(`./request/${options.action}.mjs`).then(mod => mod.default(options)).catch(err => {
                    console.error('failed to load request handler', options.action, err)
                    socket.emit(options.responseEvent, {
                        status: 'error',
                        error: err.message,
                    })
                })
            } else {
                console.warn('trying bad action socket.on p3xr-request with options', options)
            }
        })


        let dividers = [
            ":",
            "/",
            "|",
            "-",
            "@"
        ]
        if (p3xrs.cfg.hasOwnProperty('treeDividers') && Array.isArray(p3xrs.cfg.treeDividers)) {
            dividers = p3xrs.cfg.treeDividers
        }
        socket.emit('configuration', {
            readonlyConnections: p3xrs.cfg.readonlyConnections === true,
            snapshot: isSnapshot,
            treeDividers: dividers,
            version: version,
            hasGroqApiKey: !!(p3xrs.cfg.groqApiKey && p3xrs.cfg.groqApiKey.startsWith('gsk_') && p3xrs.cfg.groqApiKey.length > 20),
            groqApiKeyMasked: p3xrs.cfg.groqApiKey && p3xrs.cfg.groqApiKey.length > 8 ? `${p3xrs.cfg.groqApiKey.slice(0, 4)}...${p3xrs.cfg.groqApiKey.slice(-4)}` : '',
            groqApiKeyReadonly: p3xrs.cfg.groqApiKeyReadonly === true,
            aiEnabled: p3xrs.cfg.aiEnabled !== false,
            aiUseOwnKey: p3xrs.cfg.aiUseOwnKey === true,
            groqMaxTokens: p3xrs.cfg.groqMaxTokens || 16384,
        })

        socketIoShared.sendStatus({
            socket: socket,
        })
        socketIoShared.sendConnections({
            socket: socket,
        })


    });

}