Compare commits

...

3 Commits

Author SHA1 Message Date
Kain344 4577d5f7e2 feat: implement FireflyModMenu for automatic asset synchronization and server initialization
Build / build (push) Successful in 3m51s
2026-06-11 14:02:23 +07:00
Kain344 c3acbb5453 feat: fix bug
Build / build (push) Successful in 7m46s
2026-06-10 10:39:09 +07:00
Kain344 15b84e0bc7 UPDATE: bump version to 4.3.2-01 and update firefly-go.aar
Build / build (push) Successful in 3m41s
2026-06-09 19:32:36 +07:00
6 changed files with 280 additions and 36 deletions
-1
View File
@@ -1,4 +1,3 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4"> <project version="4">
<component name="ExternalStorageConfigurationManager" enabled="true" /> <component name="ExternalStorageConfigurationManager" enabled="true" />
<component name="ProjectRootManager" version="2" languageLevel="JDK_17" project-jdk-name="jbr-21" project-jdk-type="JavaSDK"> <component name="ProjectRootManager" version="2" languageLevel="JDK_17" project-jdk-name="jbr-21" project-jdk-type="JavaSDK">
Binary file not shown.
BIN
View File
Binary file not shown.
+42
View File
@@ -6,6 +6,13 @@
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd", "lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff" "ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
}, },
"CNBETAAndroid4.3.52": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"ex_resource_url": "https://autopatchos.starrails.com/design_data/BetaLive/output_15318724_a7af31327e74_b3328eb95329b2",
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15305751_09783637ccc4_d16f9c81138ab3",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
},
"CNBETAWin4.3.51": { "CNBETAWin4.3.51": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf", "asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3", "asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3",
@@ -13,6 +20,13 @@
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd", "lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff" "ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
}, },
"CNBETAWin4.3.52": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"ex_resource_url": "https://autopatchos.starrails.com/design_data/BetaLive/output_15318724_a7af31327e74_b3328eb95329b2",
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15305751_09783637ccc4_d16f9c81138ab3",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
},
"CNBETAiOS4.3.51": { "CNBETAiOS4.3.51": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf", "asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3", "asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3",
@@ -20,6 +34,13 @@
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd", "lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff" "ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
}, },
"CNBETAiOS4.3.52": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"ex_resource_url": "https://autopatchos.starrails.com/design_data/BetaLive/output_15318724_a7af31327e74_b3328eb95329b2",
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15305751_09783637ccc4_d16f9c81138ab3",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
},
"OSBETAAndroid4.3.51": { "OSBETAAndroid4.3.51": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf", "asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3", "asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3",
@@ -27,6 +48,13 @@
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd", "lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff" "ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
}, },
"OSBETAAndroid4.3.52": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"ex_resource_url": "https://autopatchos.starrails.com/design_data/BetaLive/output_15318724_a7af31327e74_b3328eb95329b2",
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15305751_09783637ccc4_d16f9c81138ab3",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
},
"OSBETAWin4.3.51": { "OSBETAWin4.3.51": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf", "asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3", "asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3",
@@ -34,11 +62,25 @@
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd", "lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff" "ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
}, },
"OSBETAWin4.3.52": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"ex_resource_url": "https://autopatchos.starrails.com/design_data/BetaLive/output_15318724_a7af31327e74_b3328eb95329b2",
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15305751_09783637ccc4_d16f9c81138ab3",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
},
"OSBETAiOS4.3.51": { "OSBETAiOS4.3.51": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf", "asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15261247_f6e6db2125cf_369da465b36faf",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3", "asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15235885_6091fd15561a_83828f542dc1f3",
"ex_resource_url": "https://autopatchos.starrails.com/design_data/BetaLive/output_15265964_c2fbb1eb05fc_f2238199ee2b6e", "ex_resource_url": "https://autopatchos.starrails.com/design_data/BetaLive/output_15265964_c2fbb1eb05fc_f2238199ee2b6e",
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd", "lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15242148_d40f856defc0_599b68a0adf7bd",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff" "ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
},
"OSBETAiOS4.3.52": {
"asset_bundle_url": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"asset_bundle_url_b": "https://autopatchos.starrails.com/asb/BetaLive/output_15305566_fea016d35145_54bbf8ab4009f5",
"ex_resource_url": "https://autopatchos.starrails.com/design_data/BetaLive/output_15318724_a7af31327e74_b3328eb95329b2",
"lua_url": "https://autopatchos.starrails.com/lua/BetaLive/output_15305751_09783637ccc4_d16f9c81138ab3",
"ifix_url": "https://autopatchos.starrails.com/ifix/BetaLive/output_15265964_799df4f0ecef_5a94550ba64cff"
} }
} }
@@ -2,11 +2,11 @@ package com.example.firefly_go_android
import android.app.Activity import android.app.Activity
import android.content.Context import android.content.Context
import android.os.Build
import android.util.Log import android.util.Log
import libandroid.Libandroid import libandroid.Libandroid
import java.io.File import java.io.File
import java.io.FileOutputStream import java.io.FileOutputStream
import java.util.zip.ZipFile
object FireflyModMenu { object FireflyModMenu {
@@ -15,73 +15,277 @@ object FireflyModMenu {
@JvmStatic @JvmStatic
fun init(activity: Activity) { fun init(activity: Activity) {
val modPackageName = "com.kain344.firefly_go_android"
Log.d("FireflyMod", "Initializing FireflyModMenu for module: $modPackageName")
val appDataPath = File(activity.getExternalFilesDir(null), "FireflyGo").absolutePath val appDataPath = File(activity.getExternalFilesDir(null), "FireflyGo").absolutePath
val dataDir = File("$appDataPath/data") val dataDir = File("$appDataPath/data")
if (!dataDir.exists()) dataDir.mkdirs() if (!dataDir.exists()) dataDir.mkdirs()
val sharedPrefs = activity.getSharedPreferences("FireflyModPrefs", Context.MODE_PRIVATE) val sharedPrefs = activity.getSharedPreferences("FireflyModPrefs", Context.MODE_PRIVATE)
val currentVersion = try {
val packageInfo = activity.packageManager.getPackageInfo("com.kain344.firefly_go_android", 0) val apkPath = getModuleApkPath(FireflyModMenu::class.java.classLoader)
if (Build.VERSION.SDK_INT >= 33) {
packageInfo.longVersionCode val currentUpdateTime = if (apkPath != null) {
} else { try {
@Suppress("DEPRECATION") val lastModified = File(apkPath).lastModified()
packageInfo.versionCode.toLong() Log.d("FireflyMod", "Module APK path: $apkPath, Last modified: $lastModified")
} lastModified
} catch (e: Exception) { } catch (e: Exception) {
1L Log.e("FireflyMod", "Lỗi khi lấy last modified của APK: ${e.message}")
0L
}
} else {
// Fallback to PackageManager if APK path could not be resolved
try {
val packageInfo = activity.packageManager.getPackageInfo(modPackageName, 0)
Log.d("FireflyMod", "Module package found. Last update: ${packageInfo.lastUpdateTime}")
packageInfo.lastUpdateTime
} catch (e: Exception) {
Log.e("FireflyMod", "Module package $modPackageName not found: ${e.message}")
0L
}
}
val lastUpdateTime = sharedPrefs.getLong("last_update_time", 0L)
val shouldOverride = currentUpdateTime != 0L && currentUpdateTime != lastUpdateTime
if (shouldOverride) {
Log.d("FireflyMod", "Module version changed (old: $lastUpdateTime, new: $currentUpdateTime). Forcing asset override.")
} }
val lastVersion = sharedPrefs.getLong("last_version_code", 0L)
val shouldOverride = currentVersion > lastVersion
if (!isServerStarted) { if (!isServerStarted) {
Log.d("FireflyMod", "Start Server") Log.d("FireflyMod", "Starting Server thread...")
isServerStarted = true isServerStarted = true
Thread { Thread {
try { try {
val isCopyDone = copyRawFiles(dataDir, shouldOverride) val isCopyDone = if (apkPath != null) {
copyRawFilesFromApk(apkPath, dataDir, shouldOverride)
} else {
// Fallback to createPackageContext
val modContext = try {
activity.createPackageContext(modPackageName, Context.CONTEXT_INCLUDE_CODE or Context.CONTEXT_IGNORE_SECURITY)
} catch (e: Exception) {
Log.e("FireflyMod", "Không thể tạo mod context cho $modPackageName: ${e.message}")
null
}
if (modContext != null) {
copyRawFiles(modContext, dataDir, shouldOverride)
} else {
false
}
}
if (isCopyDone) { if (isCopyDone) {
if (shouldOverride) { if (shouldOverride) {
sharedPrefs.edit().putLong("last_version_code", currentVersion).apply() sharedPrefs.edit().putLong("last_update_time", currentUpdateTime).apply()
} }
Libandroid.setPathDataLocal(appDataPath) Libandroid.setPathDataLocal(appDataPath)
Libandroid.setServerRunning(true) Libandroid.setServerRunning(true)
Log.d("FireflyMod", "Server started successfully.")
} else { } else {
Log.e("FireflyMod", "Lỗi khi copy assets, server không được khởi chạy.")
isServerStarted = false isServerStarted = false
} }
} catch (e: Exception) { } catch (e: Exception) {
Log.e("FireflyMod", "Error starting server: ${e.message}", e) Log.e("FireflyMod", "Error in server thread: ${e.message}", e)
isServerStarted = false isServerStarted = false
} }
}.start() }.start()
} }
} }
private fun copyRawFiles(targetDir: File, override: Boolean = false): Boolean { private fun getModuleApkPath(classLoader: ClassLoader?): String? {
val files = listOf( var cl = classLoader
"assets/data-in-game.json" to "data-in-game.json", while (cl != null) {
"assets/freesr-data.json" to "freesr-data.json", Log.d("FireflyMod", "Checking ClassLoader: ${cl.javaClass.name}")
"assets/version.json" to "version.json" // 1. Try parsing toString() first (reflection-free, safe from hidden API checks)
val pathFromToString = getModuleApkPathFromToString(cl)
if (pathFromToString != null) {
Log.d("FireflyMod", "Found APK path via ClassLoader toString(): $pathFromToString")
return pathFromToString
}
// 2. Try reflection on this class loader
val pathFromReflection = getModuleApkPathViaReflection(cl)
if (pathFromReflection != null) {
Log.d("FireflyMod", "Found APK path via reflection: $pathFromReflection")
return pathFromReflection
}
cl = cl.parent
}
Log.w("FireflyMod", "Could not find module APK path in ClassLoader hierarchy.")
return null
}
private fun getModuleApkPathFromToString(classLoader: ClassLoader): String? {
val clString = classLoader.toString()
val regexes = listOf(
Regex("""zip file "([^"]+)""""),
Regex("""zip file '([^']+)'"""),
Regex("""\[([^\]]+\.(apk|zip))\]"""),
Regex("""file:([^:\s]+\.(apk|zip))""")
) )
var fallbackPath: String? = null
for (regex in regexes) {
val matches = regex.findAll(clString)
for (match in matches) {
if (match.groupValues.size > 1) {
val path = match.groupValues[1]
val file = File(path)
if (file.exists() && file.isFile) {
if (path.contains("com.kain344.firefly_go_android") && path.endsWith(".apk")) {
return path
}
if (path.endsWith(".apk") && fallbackPath == null) {
fallbackPath = path
}
}
}
}
}
// General path match for any absolute path ending with .apk or .zip
val pathRegex = Regex("""(/[^:\s"']+\.(apk|zip))""")
val pathMatches = pathRegex.findAll(clString)
for (match in pathMatches) {
val path = match.groupValues[1]
val file = File(path)
if (file.exists() && file.isFile) {
if (path.contains("com.kain344.firefly_go_android") && path.endsWith(".apk")) {
return path
}
if (path.endsWith(".apk") && fallbackPath == null) {
fallbackPath = path
}
}
}
return fallbackPath
}
private fun getModuleApkPathViaReflection(classLoader: ClassLoader): String? {
try {
val baseDexClassLoaderClass = Class.forName("dalvik.system.BaseDexClassLoader")
if (!baseDexClassLoaderClass.isInstance(classLoader)) return null
val pathListField = baseDexClassLoaderClass.getDeclaredField("pathList")
pathListField.isAccessible = true
val pathList = pathListField.get(classLoader) ?: return null
val dexElementsField = pathList.javaClass.getDeclaredField("dexElements")
dexElementsField.isAccessible = true
val dexElements = dexElementsField.get(pathList) as? Array<*> ?: return null
var fallbackApkPath: String? = null
for (element in dexElements) {
if (element == null) continue
val elementClass = element.javaClass
val fileField = try {
elementClass.getDeclaredField("file")
} catch (e: NoSuchFieldException) {
try {
elementClass.getDeclaredField("path")
} catch (ex: NoSuchFieldException) {
null
}
}
if (fileField != null) {
fileField.isAccessible = true
val file = fileField.get(element) as? File
if (file != null && file.exists() && file.isFile) {
val path = file.absolutePath
if (path.contains("com.kain344.firefly_go_android") && path.endsWith(".apk")) {
return path
}
if (path.endsWith(".apk") && fallbackApkPath == null) {
fallbackApkPath = path
}
}
}
}
return fallbackApkPath
} catch (e: Exception) {
Log.d("FireflyMod", "Reflection failed on ClassLoader ${classLoader.javaClass.name}: ${e.message}")
}
return null
}
private fun copyRawFilesFromApk(apkPath: String, targetDir: File, override: Boolean = false): Boolean {
val files = listOf(
"data-in-game.json" to "data-in-game.json",
"freesr-data.json" to "freesr-data.json",
"version.json" to "version.json"
)
return try { return try {
for ((assetPath, name) in files) { ZipFile(apkPath).use { zip ->
val outFile = File(targetDir, name) for ((assetFile, outName) in files) {
if (outFile.exists() && !override) continue val outFile = File(targetDir, outName)
if (outFile.exists() && !override) {
Log.d("FireflyMod", "Bỏ qua $outName (đã tồn tại)")
continue
}
val inputStream = val entryName = "assets/$assetFile"
FireflyModMenu::class.java.classLoader?.getResourceAsStream(assetPath) val entry = zip.getEntry(entryName)
?: return false if (entry == null) {
Log.e("FireflyMod", "Không tìm thấy entry $entryName trong APK")
return false
}
inputStream.use { input -> zip.getInputStream(entry).use { input ->
FileOutputStream(outFile).use { output -> FileOutputStream(outFile).use { output ->
input.copyTo(output) input.copyTo(output)
output.fd.sync() output.fd.sync()
} }
} }
Log.d("FireflyMod", "Đã copy $assetFile từ APK thành công (override=$override)")
}
} }
true true
} catch (e: Exception) { } catch (e: Exception) {
Log.e("FireflyMod", "Error copying file: ${e.message}", e) Log.e("FireflyMod", "Lỗi khi copy file từ APK: ${e.message}", e)
false
}
}
private fun copyRawFiles(context: Context, targetDir: File, override: Boolean = false): Boolean {
val files = listOf(
"data-in-game.json" to "data-in-game.json",
"freesr-data.json" to "freesr-data.json",
"version.json" to "version.json"
)
return try {
// Kiểm tra danh sách assets có sẵn để debug
val availableAssets = context.assets.list("")?.toList() ?: emptyList()
Log.d("FireflyMod", "Assets có sẵn trong module: $availableAssets")
for ((assetFile, outName) in files) {
val outFile = File(targetDir, outName)
if (outFile.exists() && !override) {
Log.d("FireflyMod", "Bỏ qua $outName (đã tồn tại)")
continue
}
if (!availableAssets.contains(assetFile)) {
Log.w("FireflyMod", "Cảnh báo: Không tìm thấy $assetFile trong danh sách assets của module!")
}
context.assets.open(assetFile).use { input ->
FileOutputStream(outFile).use { output ->
input.copyTo(output)
output.fd.sync()
}
}
Log.d("FireflyMod", "Đã copy $assetFile thành công (override=$override)")
}
true
} catch (e: Exception) {
Log.e("FireflyMod", "Lỗi khi copy file từ assets của ${context.packageName}: ${e.message}", e)
false false
} }
} }
+2 -3
View File
@@ -1,5 +1,4 @@
{ {
"tag": "4.3.1-02", "tag": "4.3.2-03",
"title": "PreBuild Version 4.3.51 - 02" "title": "PreBuild Version 4.3.52 - 03"
} }