Source: Commands/Main/Events/index.js

/* eslint-disable indent */
/* eslint-disable default-case */
/* eslint-disable max-len */
/* Requires */
const fs = require('fs');

/* Importa módulos, ajuste o local conforme onde usar esse sistema */
const Indexer = require('../../../index');

/* JSON's | Utilidades */
const envInfo = JSON.parse(fs.readFileSync(`${__dirname}/utils.json`));

/**
 * Retorna todos os detalhes do ambiente (`envInfo`).
 *
 * @returns {Object} O objeto `envInfo`, que contém os detalhes do ambiente da execução.
 */
function ambientDetails() {
    /* Retorna a envData */
    return envInfo;
}

/* Cria a função de comando */
async function eventRunner(
    kill = envInfo.functions.exec.arguments.kill.value,
    data = envInfo.functions.exec.arguments.data.value,
) {
    /* Define um resultado padrão */
    envInfo.results.value = false;

    /* Define o sucesso */
    envInfo.results.success = false;

    /* Try-Catch para casos de erro */
    try {
        /* Se recebeu tudo corretamente, se der ruim, não fará nada */
        if (typeof kill === 'object' && typeof data === 'object') {
            /* Define o timestamp */
            const timestamp = typeof data.messageTimestamp === 'string' ? data.messageTimestamp : data.messageTimestamp?.low || data.messageTimestamp?.high || Date.now();

            /* Define quem foi o alvo do evento */
            const participants = data?.messageStubParameters || [];
            const formattedPP = participants.map((usr) => usr.replace(/@s.whatsapp.net/gi, ''));

            /* Define o chat do evento em ID */
            const chatId = data?.key?.remoteJid || data?.key?.participant || irisNumber || config.owner.value[0];

            /* Define o participante pela key | "Causador do evento" */
            const user = data?.participant || data?.key?.participant || irisNumber;
            const userFormatted = user.replace(/@s.whatsapp.net/gi, '');

            /* Determina se é um grupo */
            const isGroup = chatId.includes('@g.us') || false;

            /* Define os dados de database do grupo e usuário */
            const groupDB = isGroup ? Indexer('sql').get('groups', user, chatId).value : 'PV';
            const personalDB = Indexer('sql').get('personal', user, chatId).value;

            /* Se tiver apenas um participante, puxa o nome dele na DB */
            const pName = participants.length === 1 ? Indexer('sql').get('personal', participants[0], chatId).value : 'N/A';

            /* Define o StupType */
            const stubType = data?.messageStubType || 0;

            /* Faz um switch com os tipos de função stub */
            switch (stubType) {
                /* Promote & Demote */
                case 'GROUP_PARTICIPANT_PROMOTE':
                case 'GROUP_PARTICIPANT_DEMOTE':
                case 29:
                case 30:
                    /* Se permitido fazer essa função */
                    if (groupDB?.spy?.enable === true) {
                        /* Define o tipo */
                        const isDemote = stubType === 30 || stubType === 'GROUP_PARTICIPANT_DEMOTE';

                        /* Define o tipo de promote/demote na key */
                        const keyTexts = isDemote ? 'demoteText' : 'promoteText';

                        /* Faz o aviso na tela */
                        console.log(Indexer('color').echo(`[ ${isDemote ? 'DEMOTE' : 'PROMOTE'} ~ ${groupDB.name.text || 'GP'} ] →`, config.colorSet.value[0]).value, Indexer('color').echo(`'${formattedPP} (${pName.name.text})' ${envInfo.parameters.admins.value[keyTexts]} '${userFormatted} (${personalDB.name.text})' as ${new Date(timestamp * 1000).toLocaleString()}\n`, config.colorSet.value[1]).value);

                        /* Avisa o grupo */
                        await kill.sendMessage(chatId, {
                            text: Indexer('sql').languages(region, 'Events', keyTexts, true, true, {
                                admin: `@${userFormatted}`,
                                user: `@${formattedPP.join(', @')}`,
                            }).value,
                            mentions: [user, ...participants],
                        });

                        /* Define que deve parar a execução */
                        envInfo.results.value = 'STOP';
                    }
                break;

                /* Approval (Admin permitiu a entrada) */
                case 27:
                case 'GROUP_PARTICIPANT_ADD':
                    /* Apenas se permitido fazer */
                    if (groupDB?.spy?.enable === true) {
                        /* Faz o aviso na tela */
                        console.log(Indexer('color').echo(`[ APPROVAL ~ ${groupDB.name.text || 'GP'} ] →`, config.colorSet.value[0]).value, Indexer('color').echo(`'${userFormatted} (${personalDB.name.text})' ${envInfo.parameters.admins.value.approvedText} '${formattedPP}' as ${new Date(timestamp * 1000).toLocaleString()}\n`, config.colorSet.value[1]).value);

                        /* Avisa o grupo */
                        await kill.sendMessage(chatId, {
                            text: Indexer('sql').languages(region, 'Events', 'approvedText', true, true, {
                                admin: `@${userFormatted}`,
                                user: `@${formattedPP.join(', @')}`,
                            }).value,
                            mentions: [user, ...participants],
                        });

                        /* Define que deve parar a execução */
                        envInfo.results.value = 'STOP';
                    }
                break;
            }
        }

        /*
            Define o sucesso, se seu comando der erro isso jamais será chamado
            Então o success automaticamente será false em falhas
        */
        envInfo.results.success = true;

        /* Caso de algum erro */
    } catch (error) {
        /* Insere tudo na envInfo */
        logging.echoError(error, envInfo, __dirname);
    }

    /* Retorna os resultados */
    return logging.postResults(envInfo);
}

/**
 * Restaura o ambiente e atualiza as exportações do módulo com a funcionalidade principal
 * @param {Object} [changeKey={}] - Chaves personalizadas para atualizar o envInfo
 * @param {Object} [envFile=envInfo] - Objeto com informações do ambiente
 * @param {string} [dirname=__dirname] - Caminho do diretório atual
 * @returns {Object} Exportações do módulo com todas as funções configuradas
 */
/* eslint-disable-next-line no-return-assign */
const resetLocal = (
    changeKey = {},
    envFile = envInfo,
    dirname = __dirname,
) => module.exports = logging.resetAmbient({
    functions: {
        [envInfo.exports.env]: { value: ambientDetails },
        [envInfo.exports.messedup]: { value: logging.echoError },
        [envInfo.exports.poswork]: { value: logging.postResults },
        [envInfo.exports.reset]: { value: resetLocal },
        [envInfo.exports.exec]: { value: eventRunner },
    },
    parameters: {
        location: { value: __filename },
    },
}, envFile, changeKey, dirname);
resetLocal();