feat(BotCallsigns)

This commit is contained in:
GetParanoid 2025-07-18 17:40:38 -05:00
parent 41f21421fa
commit 279a0fe8ce
8 changed files with 34773 additions and 0 deletions

View file

@ -0,0 +1,34 @@
University of Illinois/NCSA Open Source License
Copyright (c) 2022 Chief. All rights reserved.
Developed by: SPT-AKI
Chief
https://www.sp-tarkov.com/
Permission is hereby granted, free of charge, to any person
obtaining a copy of this software and associated documentation files
(the "Software"), to deal with the Software without restriction,
including without limitation the rights to use, copy, modify, merge,
publish, distribute, sublicense, and/or sell copies of the Software,
and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:
* Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimers.
* Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimers in the
documentation and/or other materials provided with the distribution.
* Neither the names of Chief, SPT-AKI nor the names of its
contributors may be used to endorse or promote products derived from
this Software without specific prior written permission.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
CONTRIBUTORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS WITH
THE SOFTWARE.

View file

@ -0,0 +1,8 @@
{
"validateNames": false,
"useCustomScavNames": false,
"useBEARCyrillicNames": false,
"useUSECEnglishNames": false,
"addExtraNames": false,
"junklessLogging": true
}

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,8 @@
{
"name": "Bot Callsigns",
"version": "1.5.3",
"main": "src/mod.js",
"license": "MIT",
"author": "Helldiver, harmony",
"sptVersion": "3.11"
}

View file

@ -0,0 +1,166 @@
"use strict";
const fs = require('fs');
class BotNames {
CFG = require("../config/config.json");
constructor() {
this.bearNames = this.CFG.useBEARCyrillicNames
? require("../names/bear_cyrillic.json")
: require("../names/bear.json");
this.usecNames = this.CFG.useUSECEnglishNames
? require("../names/usec_en.json")
: require("../names/usec.json");
}
postDBLoad(container) {
const logger = container.resolve("WinstonLogger");
const db = container.resolve("DatabaseServer");
const bot = db.getTables().bots.types;
const config = this.CFG;
const extraUSECNamesPath = "./user/mods/BotCallsigns/config/usec_extra_names.json";
const extraBEARNamesPath = "./user/mods/BotCallsigns/config/bear_extra_names.json";
const scavNames = "./user/mods/BotCallsigns/config/scav_names.json";
const pathToTwitchPlayers = "./user/mods/TwitchPlayers";
// Check for extra name files and create them
function createFileIfNotExists(path, scavFile = false, ModReady = false) {
if (!fs.existsSync(path) && scavFile == false) {
const defaultStructure = { "Names": ["extra", "names", "go", "here"] };
try {
fs.writeFileSync(path, JSON.stringify(defaultStructure, null, 2));
logger.log(`[BotCallsigns] Created missing extra names file: ${path}`, "cyan");
} catch (error) {
logger.log(`[BotCallsigns] Failed to create file: ${path} $`, "red");
}
} else if (!fs.existsSync(path) && scavFile == true) {
const defaultStructure = { "firstNames": ["put", "your", "first names", "here", "or", "add", "more"], "lastNames": ["put", "your", "last names", "here", "or", "add", "more"] };
try {
fs.writeFileSync(path, JSON.stringify(defaultStructure, null, 2));
logger.log(`[BotCallsigns] Created missing extra names file: ${path}`, "cyan");
} catch (error) {
logger.log(`[BotCallsigns] Failed to create file: ${path} $`, "red");
}
}
// Create mod.ready file for Twitch Players signaling that we're done with file creation
if (ModReady == true) {
modReadyTTV();
}
}
createFileIfNotExists(extraUSECNamesPath, false, false);
createFileIfNotExists(extraBEARNamesPath, false, false);
createFileIfNotExists(scavNames, true, true);
// Mod is ready to work with Twitch Players
function modReadyTTV() {
if(fs.existsSync(pathToTwitchPlayers)){
const pathToFlag = "./user/mods/TwitchPlayers/temp/mod.ready";
fs.writeFileSync(pathToFlag, '', 'utf-8');
}
}
// Name validation
function validateNames(names, type) {
const validNamePattern = /^[\p{L}\p{N}\-_!@ #]+(?:\.[\p{L}\p{N}\-_!@ ]+)*$/u;
const validNames = [];
const invalidNames = [];
names.forEach(name => {
if (validNamePattern.test(name)) {
validNames.push(name);
} else {
invalidNames.push(name);
}
});
if (invalidNames.length > 0) {
logger.log(`[BotCallsigns] ${type} names contain invalid name(s): ${invalidNames.join(", ")} | The mod will not use them. You can either fix, or ignore this.`, "yellow");
} else {
if (!config.junklessLogging)
logger.log(`[BotCallsigns] ${type} names passed name validation`, "green");
}
return validNames;
}
// Extra names loading
function loadExtraNames(path, defaultNames, type) {
if (fs.existsSync(path)) {
try {
const fileContent = fs.readFileSync(path, "utf-8");
const extraNames = JSON.parse(fileContent).Names;
logger.log(`[BotCallsigns] Loaded extra ${type} names from ${path}`, "green");
return [...defaultNames, ...extraNames];
} catch (error) {
logger.log(`[BotCallsigns] Failed to load extra ${type} names from ${path}`, "red");
}
}
return defaultNames;
}
if (config.addExtraNames) {
this.bearNames.Names = loadExtraNames(extraBEARNamesPath, this.bearNames.Names, "BEAR", logger);
this.usecNames.Names = loadExtraNames(extraUSECNamesPath, this.usecNames.Names, "USEC", logger);
}
// Name Validation if enabled in the config
if (config.validateNames) {
if (!config.junklessLogging)
logger.log("[BotCallsigns] Validating BEAR and USEC names...", "green");
const bearNames = validateNames(this.bearNames.Names, "BEAR", logger);
const usecNames = validateNames(this.usecNames.Names, "USEC", logger);
bot["bear"].firstName = bearNames;
bot["usec"].firstName = usecNames;
if (!config.junklessLogging)
logger.log(`[BotCallsigns] Loaded ${bearNames.length} BEAR and ${usecNames.length} USEC names!`, "green");
} else {
bot["bear"].firstName = this.bearNames.Names;
bot["usec"].firstName = this.usecNames.Names;
if (!config.junklessLogging)
logger.log(`[BotCallsigns] Loaded ${this.bearNames.Names.length} BEAR and ${this.usecNames.Names.length} USEC names!`, "green");
}
// If using SCAV names too
if (config.useCustomScavNames) {
if (config.validateNames) {
if (!config.junklessLogging)
logger.log("[BotCallsigns] Validating SCAV names...", "green");
this.scavNames = require("../config/scav_names.json");
const scavFirstNames = validateNames(this.scavNames['firstNames'], "SCAV First Names", logger);
const scavLastNames = validateNames(this.scavNames['lastNames'], "SCAV Last Names", logger);
Object.assign(bot["assault"], {
firstName: scavFirstNames,
lastName: scavLastNames
});
if (!config.junklessLogging)
logger.log(`[BotCallsigns] Loaded ${scavFirstNames.length} SCAV first names and ${scavLastNames.length} last names`, "green");
} else {
const scavFirstNames = this.scavNames['firstNames'];
const scavLastNames = this.scavNames['lastNames'];
Object.assign(bot["assault"], {
firstName: scavFirstNames,
lastName: scavLastNames
});
if (!config.junklessLogging)
logger.log(`[BotCallsigns] Loaded ${scavFirstNames.length} SCAV first names and ${scavLastNames.length} last names`, "green");
}
}
}
}
module.exports = { mod: new BotNames() };