GC-Proto/op.js

929 lines
22 KiB
JavaScript

// Removed useless files, to speed up build process and make it clearer.
const path = require("path");
const fs = require("fs");
const readline = require("readline");
// cmdid current version
const read_cmdid_last = "cmdid_last.csv";
const read_cmdid = "cmdid.csv";
const read_cmdid_latest_output = "cmdid.json";
const read_cmdid_ht_output = "cmdid_ht_40.json";
const read_cmdid_gc_output = "cmdid_gc.json";
const read_cmdid_output_gc_full = "cmdid_gc_full.json";
const read_cmdid_output_gc_update = "cmdid_gc_update.json";
const read_cmdid_output_gc_nofound = "cmdid_gc_nofound.json";
const file_gc_needed = "gc_needed.json";
const write_op = "PacketOpcodes.java";
console.log(process.cwd());
const json_gc_needed = read_json(file_gc_needed);
//(TODO: add input file)
// folder gc auto-generated proto
const folder_proto_gc_gen =
"../GSServer-GC/src/generated/main/java/emu/grasscutter/net/proto/";
// file PacketOpcodes currently in use
const read_cmdid_gc =
"../GSServer-GC/src/main/java/emu/grasscutter/net/packet/PacketOpcodes.java";
const folder_packet_gc =
"../GSServer-GC/src/main/java/emu/grasscutter/server/packet/";
const folder_gc_scan = "../GSServer-GCOriginal/src/main/java/emu/grasscutter/";
//const read_cmdid = fs.readFileSync("cmdid.csv");
//const read_packetopcodes = fs.readFileSync("PacketOpcodes.java");
var data = [];
var data_gc = [];
var data_gc2 = [];
var index_file_gen = 0;
var index_file_cmdid = 0;
var index_cmdid_gc = 0;
var index_cmdid_gc_out = 0;
// ???
function check_gen() {
fs.readdir(folder_proto_gc_gen, function (err, files) {
//handling error
if (err) {
return console.log("Unable to scan directory: " + err);
}
files.forEach(function (file) {
index_file_gen++;
});
console.log("file proto gen: " + index_file_gen);
});
}
// ???
function get_cmdid_csvtojson() {
const inputStreamcmdid = fs.createReadStream(read_cmdid);
var lineReadercmdid = readline.createInterface({
input: inputStreamcmdid,
terminal: false,
});
lineReadercmdid.on("line", function (line) {
var config = line.split(",");
var subdata = new Object();
subdata["name"] = config[0];
subdata["id"] = parseInt(config[1]);
data.push(subdata);
index_file_cmdid++;
});
lineReadercmdid.on("close", function () {
console.log("found cmd id " + index_file_cmdid);
//save_json(data, read_cmdid_latest_output);
});
}
// ???
var data_gen = [];
var index_file_cmdid_gen = 0;
function read_cmdid_ht_json() {
const k = read_json(read_cmdid_ht_output);
// console.log(k);
for (const key in k) {
var name = k[key];
var id = parseInt(key);
//console.log(id);
var subdata = new Object();
subdata["name"] = name;
subdata["id"] = id;
data_gen.push(subdata);
index_file_cmdid_gen++;
}
console.log("found cmd id " + index_file_cmdid_gen);
//save_json(data_gen, read_cmdid_latest_output);
}
// op java to json
var tes = 50000;
function get_cmdid_gc() {
const inputStreamcmdid = fs.createReadStream(read_cmdid_gc);
var lineReadercmdid = readline.createInterface({
input: inputStreamcmdid,
terminal: false,
});
//41528
lineReadercmdid.on("line", function (line) {
var config = line.split(" = ");
var name = config[0];
var id = parseInt(config[1]);
if (name.includes("public static final int")) {
name = name.replace(" public static final int ", "");
// skip 0 ?
if (id == 1) {
return;
}
if (id == 8888888) {
id = tes++;
}
var subdata = new Object();
subdata["name"] = name;
subdata["id"] = id;
data_gc.push(subdata);
var found = json_gc_needed.find((j) => name.match(j.name));
if (!found) {
// skip bad
return;
}
data_gc2.push(subdata);
//console.log(name);
index_cmdid_gc++;
} else {
index_cmdid_gc_out++;
}
});
lineReadercmdid.on("close", function () {
console.log(
"found cmd id " + index_cmdid_gc + " | no need " + index_cmdid_gc_out
);
save_json(data_gc2, read_cmdid_gc_output);
save_json(data_gc, read_cmdid_output_gc_full);
});
}
// Update cmdid gc
var found_cmdid_new = 0;
var nofound_cmdid_new = 0;
var rename_name_cmdid = 0;
var noneed_rename_name_cmdid = 0;
var data_gc_cmdid_nofound = [];
var check_dunp_id = [];
var unkId = 50000;
function update_cmdid_gc() {
let melon = "";
const json_cmdid_last = read_json(read_cmdid_latest_output);
const json_cmdid_old = read_json(read_cmdid_gc_output);
json_cmdid_old.forEach(function (s, index) {
var id = s.id;
var name = s.name.trim();
// switch to name mode?
var found_id = json_cmdid_last.find((j) => j.name == name);
if (found_id) {
found_cmdid_new++;
if (id == found_id.id) {
noneed_rename_name_cmdid++;
} else {
rename_name_cmdid++;
//console.log("Wow rename -> ID: "+id+" > "+found_id.id);
s.replace = s.id; // old
s.id = found_id.id; // rename id
}
} else {
unkId++
console.log("No found :( -> ID OLD: " + id + " > "+unkId+" | Name: " + name);
s.id = unkId;
data_gc_cmdid_nofound.push(s);
nofound_cmdid_new++;
}
melon += `${name},${id}\n`;
// find dump by id
/*
var found_id = check_dunp_id.find((j) => j.id == id);
if (found_id) {
console.log("Wow dup -> ID: " +id +" (ADD " +found_id.id +") | Name Remove: " +name +" (ADD " +found_id.name +")");
// remove bad
json_cmdid_old.splice(index, 1);
} else {
check_dunp_id.push(s);
}
*/
// find dump by name?
});
check_dunp_id = []; // clear
// I don't know why this happened but make sure to check again
/*
var check_dunp_name = [];
json_cmdid_old.forEach(function (s, index) {
var id = s.id;
var name = s.name.trim();
var found_name = check_dunp_name.find((j) => j.name === name);
if (found_name) {
console.log(
"Wow dup -> ID: " +
id +
" (ADD " +
found_name.id +
") | Name Remove: " +
name +
" (ADD " +
found_name.name +
")"
);
// remove bad
json_cmdid_old.splice(index, 1);
} else {
check_dunp_name.push(s);
}
var found_id = json_cmdid_last.find((j) => j.id == id);
if (found_id) {
if (name != found_id.name) {
console.log(
"Wow why? -> ID: " + id + " | Name: " + name + " > " + found_id.name
);
s.name = found_id.name;
}
}
});
*/
//check_dunp_name = []; // clear
console.log(
"found " +
found_cmdid_new +
" | no found " +
nofound_cmdid_new +
" | rename " +
rename_name_cmdid +
" | noneed rename " +
noneed_rename_name_cmdid
);
// const csvRows = Object.entries(read_cmdid_output_gc_update).map(([name, id]) => `${name},${id}`);
// const csvData = csvRows.join("\n");
fs.writeFileSync(read_cmdid_last, melon);
save_json(json_cmdid_old, read_cmdid_output_gc_update);
save_json(data_gc_cmdid_nofound, read_cmdid_output_gc_nofound);
}
function read_json(file) {
return JSON.parse(fs.readFileSync(file));
}
// save json
function save_json(raw, file) {
var j = JSON.stringify(raw, null, 4);
save(j, file);
}
function save(raw, file) {
fs.writeFile(file, raw, "utf8", function (err) {
if (err) {
console.log("An error occured while writing to File.");
return console.log(err);
}
console.log("File has been saved: " + file);
});
}
var dup_name = [];
var count_dup = 0;
var count_nodup = 0;
function cmdid_to_op() {
let melon =
"\
package emu.grasscutter.net.packet;\
\n\
\npublic class PacketOpcodes {\
\n// Empty\
\npublic static final int NONE = 0;\
\n\
\n// Opcodes\
";
const cmdidfix_raw = fs.readFileSync(read_cmdid_gc_output);
//const json_cmdidfix_raw = JSON.parse(cmdidfix_raw);
const json_cmdidfix_raw = read_json(read_cmdid_output_gc_update);
json_cmdidfix_raw.sort((a, b) => a.id - b.id);
json_cmdidfix_raw.forEach(function (s) {
var found_id = dup_name.find((j) => j.name == s.name);
if (!found_id) {
melon += "\npublic static final int " + s.name + " = " + s.id + ";";
dup_name.push(s);
count_nodup++;
} else {
count_dup++;
if (s.replace) {
console.log("DUP: " + found_id.id + " > " + s.id + " ");
}
}
//console.log(s);
/*
var found_new = json_cmdidfix_raw.find((j) => j.name == s.name);
if(found_new){
}
*/
});
console.log("done no dup " + count_nodup + " | dup " + count_dup);
melon += "\n}";
save(melon, write_op); // use "npx prettier --write PacketOpcodes.java" for better Formatter
}
var index_file_packet = 0;
var file_gc_need = [];
function fix_packet() {
const files = getAllFiles(folder_packet_gc);
files.forEach(function (file) {
const read = fs.readFileSync(file);
var real = read.toString();
const names = getPacketOpcodes(real); // Adjusted to get an array of names
if (names.length === 0) {
return; // Skip if no PacketOpcodes.* found
}
// Iterate through the names and add them to the array
names.forEach(name => {
const subdata = { name };
file_gc_need.push(subdata);
});
index_file_packet++;
});
save_json(file_gc_need, file_gc_needed);
console.log(
"Index file: " +
index_file_packet
);
}
function isBlank(str) {
return !!!str || /^\s*$/.test(str);
}
function getAllFiles(dirPath, arrayOfFiles) {
files = fs.readdirSync(dirPath);
arrayOfFiles = arrayOfFiles || [];
files.forEach(function (file) {
if (fs.statSync(dirPath + "/" + file).isDirectory()) {
arrayOfFiles = getAllFiles(dirPath + "/" + file, arrayOfFiles);
} else {
arrayOfFiles.push(path.join(__dirname, dirPath, "/", file));
}
});
return arrayOfFiles;
}
function getPacketOpcodes(raw) {
const packetOpcodesRegex = /PacketOpcodes\.\w+/g;
const packetOpcodesMatches = raw.match(packetOpcodesRegex);
if (packetOpcodesMatches) {
return packetOpcodesMatches.map(match => match.replace('PacketOpcodes.', ''));
} else {
console.log('No PacketOpcodes.* found in the file.', raw);
return [];
}
}
// C:\Users\Administrator\Desktop\Projek\Docker\GS\gs\GSServer-GC\src\main\java\emu\grasscutter
var file_proto = [];
var file_proto_more = [];
var found_noclean = 0;
var found_needclean = 0;
var found_maybe_related = 0;
const regex_import = /\import "(.*?)\.proto"/g;
// find import in file
function find_import(file) {
var torequire = [];
var dir = path.parse(file).dir;
if (file.match("ModifierDurability")) {
//console.log("fff");
}
var rd;
// read file
try {
const read = fs.readFileSync(file);
rd = read.toString();
} catch (error) {
return;
}
while ((m = regex_import.exec(rd)) !== null) {
// This is necessary to avoid infinite loops with zero-width matches
if (m.index === regex_import.lastIndex) {
regex_import.lastIndex++;
}
// The result can be accessed through the `m`-variable.
m.forEach((match, groupIndex) => {
// only index 1 grup
if (groupIndex == 1) {
var found_rt = torequire.find((j) => j === match);
if (found_rt) {
//console.log(`Skip ${match}`);
} else {
if (match.match("Unk")) {
//console.log(file + " require " + match);
}
//var subdata = new Object();
//subdata["name"] = match;
var sub = find_import(dir + "/" + match + ".proto");
if (sub) {
sub.forEach(function (k) {
var v = torequire.find((j) => j === k);
if (!v) {
torequire.push(k);
}
});
}
torequire.push(match);
}
}
});
}
return torequire;
}
// Read the skip list
const skipFilePath = './skip.txt';
let skipList;
try {
const skipFileContent = fs.readFileSync(skipFilePath, 'utf-8');
skipList = skipFileContent.split('\n').map(line => line.trim()).filter(line => line.length > 0);
} catch (err) {
console.error('Error reading skip file:', err);
skipList = [];
}
function clean_proto_gen() {
const files = getAllFiles("./proto");
const unimplemented = "./unimplemented/";
console.log(
"File proto: " + files.length + " | Need " + json_gc_needed.length
);
// find all file import
files.forEach(function (file) {
var name_file = path.parse(file).name;
var toaddfile = new Object();
toaddfile["file"] = name_file;
toaddfile["import"] = find_import(file);
file_proto.push(toaddfile);
});
//console.log(file_proto);
function find_json_proto(name_file, newnew = null) {
var noe = [];
if (newnew) {
noe = newnew;
}
var found_proto = file_proto.find((j) => name_file === j.file); // must same file
if (found_proto) {
// if found import
if (found_proto.import) {
found_proto.import.forEach(function (s) {
var f = noe.find((j) => j === s);
if (!f) {
noe.push(s);
// check sub
var sub = find_json_proto(s, noe);
sub.forEach(function (k) {
var v = noe.find((j) => j === k);
if (!v) {
noe.push(k);
}
});
}
});
}
var ss = noe.find((j) => j === name_file);
if (!ss) {
noe.push(name_file);
}
} else {
// skip
}
return noe;
}
// try with proto json
var filedonotdelete = [];
files.forEach(function (file) {
var name_file = path.parse(file).name;
var findme = false;
// find main file
var found = json_gc_needed.find((j) => name_file.match(j.name));
if (found) {
findme = true;
} else {
// skip
}
// main file
if (findme) {
var tosub = find_json_proto(name_file);
tosub.forEach(function (k) {
var oo = filedonotdelete.find((j) => j === k);
if (!oo) {
filedonotdelete.push(k);
}
});
} else {
// not main file
/*
if (name_file.match("ModifierDurability")) {
console.log("3");
}
*/
if (file.match("ChannelerSlabChallenge")) {
console.log(file);
}
}
});
// ModifierDurability
//console.log(filedonotdelete);
// last
files.forEach(function (file) {
var name_file = path.parse(file).name;
var toskip = filedonotdelete.find((j) => name_file === j);
if (toskip) {
found_noclean++;
} else {
if (skipList.includes(name_file)) {
console.log('Skipping file:', file);
found_noclean++;
return;
}
found_needclean++;
console.log("Move file: " + file);
const destinationFile = path.join(unimplemented, path.basename(file));
try {
fs.renameSync(file, destinationFile);
} catch (err) {
console.error(err);
}
}
});
//console.log(filedonotdelete);
console.log(
"No clean: " +
found_noclean +
" | Need to clean: " +
found_needclean +
" | Related " +
found_maybe_related
);
}
function clean_proto_gen_v2() {
const files = getAllFiles("./proto");
console.log("File proto: " + files.length);
// find all file import
files.forEach(function (file) {
// Get the file name from the file path
const fileName = path.basename(file);
// Use regular expressions to extract the desired part
const match = fileName.match(/^(.*)\.proto$/);
if (match == null) {
return;
}
const messageName = match[1];
// Check if the string is all uppercase
if (messageName === messageName.toUpperCase()) {
console.log("Remove: " + file);
try {
//fs.unlinkSync(file);
//file removed
} catch (err) {
console.error(err);
}
} else {
// console.log("String contains non-uppercase letters");
}
});
}
var g_todump = [];
function scan_gc() {
const files = getAllFiles(folder_gc_scan);
console.log("index file: " + files.length);
const regex = /import emu.grasscutter.net.proto.(.*?);/g;
files.forEach(function (file) {
// read file
const read = fs.readFileSync(file);
var rd = read.toString();
// find import
while ((m = regex.exec(rd)) !== null) {
// This is necessary to avoid infinite loops with zero-width matches
if (m.index === regex.lastIndex) {
regex.lastIndex++;
}
// The result can be accessed through the `m`-variable.
m.forEach((match, groupIndex) => {
// only index 1 grup
if (groupIndex == 1) {
if (match === "*") {
return;
}
if (match.match(".")) {
match = match.split(".")[0];
}
match = match.replace("OuterClass", "");
var found_rt = g_todump.find((j) => j.name === match);
if (found_rt) {
//found_maybe_related++;
//console.log(`Skip ${match}`);
} else {
var subdata = new Object();
subdata["name"] = match;
g_todump.push(subdata);
}
}
});
}
});
//save_json(g_todump, file_gc_needed); // This only applies to PacketOpcodes
//console.log(g_todump);
}
function clean_proto_event() {
var torequire = [
"ActivityPushTipsData",
"ActivityWatcherInfo",
"MusicGameActivityDetailInfo",
];
var filedonotdelete = [];
let dir = "./proto";
let file = "./proto/ActivityInfo.proto";
var rd;
const files = getAllFiles(dir);
// read file
try {
const read = fs.readFileSync(file);
rd = read.toString();
} catch (error) {
console.log(error);
return;
}
//console.log(rd);
while ((m = regex_import.exec(rd)) !== null) {
// This is necessary to avoid infinite loops with zero-width matches
if (m.index === regex_import.lastIndex) {
regex_import.lastIndex++;
}
// The result can be accessed through the `m`-variable.
m.forEach((match, groupIndex) => {
// only index 1 grup
if (groupIndex == 1) {
//console.log(`t: ${match}`);
var found_rt = torequire.find((j) => j === match);
if (found_rt) {
//console.log(`Skip ${match}`);
} else {
//console.log(`add ${match}`);
var sub = find_import(dir + "/" + match + ".proto");
//console.log(sub);
if (sub) {
sub.forEach(function (k) {
var v = torequire.find((j) => j === k);
if (!v) {
torequire.push(k);
}
});
}
filedonotdelete.push(match);
torequire.push(match);
}
}
});
}
// last
files.forEach(function (file) {
var name_file = path.parse(file).name;
var toskip = filedonotdelete.find((j) => name_file === j);
if (toskip) {
found_needclean++;
//console.log(name_file);
try {
//fs.unlinkSync(file);
console.log("Remove file: " + file);
//file removed
} catch (err) {
//console.error(err);
}
} else {
found_noclean++;
}
});
console.log(
"No clean: " +
found_noclean +
" | Need to clean: " +
found_needclean +
" | Related " +
found_maybe_related
);
}
function clean_proto_event_v2() {
var torequire = [
"ParamList",
"StrengthenPointData",
"TowerLevelEndNotify",
"TrialAvatarFirstPassDungeonNotify",
];
var filedonotdelete = [];
let dir = "./proto";
let file = "./proto/DungeonSettleNotify.proto";
var rd;
const files = getAllFiles(dir);
// read file
try {
const read = fs.readFileSync(file);
rd = read.toString();
} catch (error) {
console.log(error);
return;
}
//console.log(rd);
while ((m = regex_import.exec(rd)) !== null) {
// This is necessary to avoid infinite loops with zero-width matches
if (m.index === regex_import.lastIndex) {
regex_import.lastIndex++;
}
// The result can be accessed through the `m`-variable.
m.forEach((match, groupIndex) => {
// only index 1 grup
if (groupIndex == 1) {
//console.log(`t: ${match}`);
var found_rt = torequire.find((j) => j === match);
if (found_rt) {
//console.log(`Skip ${match}`);
} else {
//console.log(`add ${match}`);
var sub = find_import(dir + "/" + match + ".proto");
//console.log(sub);
if (sub) {
sub.forEach(function (k) {
var v = torequire.find((j) => j === k);
if (!v) {
torequire.push(k);
}
});
}
filedonotdelete.push(match);
torequire.push(match);
}
}
});
}
console.log(filedonotdelete);
// last
files.forEach(function (file) {
var name_file = path.parse(file).name;
var toskip = filedonotdelete.find((j) => name_file === j);
if (toskip) {
found_needclean++;
console.log(name_file);
try {
//fs.unlinkSync(file);
console.log("Remove file: " + file);
//file removed
} catch (err) {
//console.error(err);
}
} else {
found_noclean++;
//console.log("skip "+name_file);
}
});
console.log(
"No clean: " +
found_noclean +
" | Need to clean: " +
found_needclean +
" | Related " +
found_maybe_related
);
}
function cmdid_json_to_csv() {
// read_cmdid_last
const k = read_json(read_cmdid_ht_output);
//console.log(k);
const csvHeaders = "ID,Name\n";
const csvRows = Object.entries(k).map(([name, id]) => `${id},${name}`);
const csvData = csvRows.join("\n");
fs.writeFileSync(read_cmdid_last, csvData);
console.log("CSV file written successfully");
}
// cmdid_json_to_csv();
// Update GC Proto
//read_cmdid_ht_json(); // 2 or
// get_cmdid_json(); // 2. get last cmdid.csv to json
// now we have cmdid_gc.json and cmdid.json
// update_cmdid_gc(); // 3. update gc cmdid (mode by id)
// npx prettier --write PacketOpcodes.java
// scan_gc(); // 5. scan gc
// clean_proto_event(); // 6. clean event, need manual
// clean_proto_gen_v2();
//console.log(skipList)
// TODO: clean DungeonSettleNotify
// clean_proto_event_v2();
// 1
//fix_packet();
// 2
//get_cmdid_gc();
// 3
//update_cmdid_gc()
cmdid_to_op();
// 4
//clean_proto_gen();
// 5