mirror of
https://github.com/skyline-emu/skyline.git
synced 2025-01-01 10:05:29 +03:00
Improve robustness of KeyReader.import
* Close the input and output file streams before moving the output file to the final destination * Clean up the destination path before moving the new file * Introduce a `ImportResult` return value to differentiate between the possible causes of import errors * Display more meaningful error messages in the UI
This commit is contained in:
parent
38129d9dc3
commit
ab6c5f4c50
@ -14,6 +14,14 @@ import java.io.File
|
|||||||
object KeyReader {
|
object KeyReader {
|
||||||
private val Tag = KeyReader::class.java.simpleName
|
private val Tag = KeyReader::class.java.simpleName
|
||||||
|
|
||||||
|
enum class ImportResult {
|
||||||
|
Success,
|
||||||
|
InvalidInputPath,
|
||||||
|
InvalidKeys,
|
||||||
|
DeletePreviousFailed,
|
||||||
|
MoveFailed,
|
||||||
|
}
|
||||||
|
|
||||||
enum class KeyType(val keyName : String, val fileName : String) {
|
enum class KeyType(val keyName : String, val fileName : String) {
|
||||||
Title("title_keys", "title.keys"), Prod("prod_keys", "prod.keys");
|
Title("title_keys", "title.keys"), Prod("prod_keys", "prod.keys");
|
||||||
|
|
||||||
@ -39,20 +47,23 @@ object KeyReader {
|
|||||||
/**
|
/**
|
||||||
* Reads keys file, trims and writes to internal app data storage, it makes sure file is properly formatted
|
* Reads keys file, trims and writes to internal app data storage, it makes sure file is properly formatted
|
||||||
*/
|
*/
|
||||||
fun import(context : Context, uri : Uri, keyType : KeyType) : Boolean {
|
fun import(context : Context, uri : Uri, keyType : KeyType) : ImportResult {
|
||||||
Log.i(Tag, "Parsing ${keyType.name} $uri")
|
Log.i(Tag, "Parsing ${keyType.name} $uri")
|
||||||
|
|
||||||
if (!DocumentFile.isDocumentUri(context, uri))
|
if (!DocumentFile.isDocumentUri(context, uri))
|
||||||
return false
|
return ImportResult.InvalidInputPath
|
||||||
|
|
||||||
val outputDirectory = File("${context.filesDir.canonicalFile}/keys/")
|
val outputDirectory = File("${context.filesDir.canonicalFile}/keys/")
|
||||||
if (!outputDirectory.exists())
|
if (!outputDirectory.exists())
|
||||||
outputDirectory.mkdirs()
|
outputDirectory.mkdirs()
|
||||||
val tmpOutputFile = File(outputDirectory, "${keyType.fileName}.tmp")
|
|
||||||
|
val outputFile = File(outputDirectory, keyType.fileName)
|
||||||
|
val tmpOutputFile = File("${outputFile}.tmp")
|
||||||
|
var valid = false
|
||||||
|
|
||||||
context.contentResolver.openInputStream(uri).use { inputStream ->
|
context.contentResolver.openInputStream(uri).use { inputStream ->
|
||||||
tmpOutputFile.bufferedWriter().use { writer ->
|
tmpOutputFile.bufferedWriter().use { writer ->
|
||||||
val valid = inputStream!!.bufferedReader().useLines {
|
valid = inputStream!!.bufferedReader().useLines {
|
||||||
for (line in it) {
|
for (line in it) {
|
||||||
if (line.startsWith(";") || line.isBlank()) continue
|
if (line.startsWith(";") || line.isBlank()) continue
|
||||||
|
|
||||||
@ -81,11 +92,32 @@ object KeyReader {
|
|||||||
}
|
}
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (valid) tmpOutputFile.renameTo(File("${tmpOutputFile.parent}/${keyType.fileName}"))
|
val cleanup = {
|
||||||
return valid
|
try {
|
||||||
|
tmpOutputFile.delete()
|
||||||
|
} catch (_ : Exception) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!valid) {
|
||||||
|
cleanup()
|
||||||
|
return ImportResult.InvalidKeys
|
||||||
|
}
|
||||||
|
|
||||||
|
if (outputFile.exists() && !outputFile.delete()) {
|
||||||
|
cleanup()
|
||||||
|
return ImportResult.DeletePreviousFailed
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!tmpOutputFile.renameTo(outputFile)) {
|
||||||
|
cleanup()
|
||||||
|
return ImportResult.MoveFailed
|
||||||
|
}
|
||||||
|
|
||||||
|
return ImportResult.Success
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun isHexString(str : String) : Boolean {
|
private fun isHexString(str : String) : Boolean {
|
||||||
|
@ -24,10 +24,18 @@ class KeyPickerPreference @JvmOverloads constructor(context : Context, attrs : A
|
|||||||
|
|
||||||
context.getSettings().refreshRequired = true
|
context.getSettings().refreshRequired = true
|
||||||
|
|
||||||
val success = KeyReader.import(context, uri, KeyReader.KeyType.parse(key))
|
val result = KeyReader.import(context, uri, KeyReader.KeyType.parse(key))
|
||||||
Snackbar.make((context as SettingsActivity).binding.root, if (success) R.string.import_keys_success else R.string.import_keys_failed, Snackbar.LENGTH_LONG).show()
|
Snackbar.make((context as SettingsActivity).binding.root, resolveImportResultString(result), Snackbar.LENGTH_LONG).show()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onClick() = documentPicker.launch(arrayOf("*/*"))
|
override fun onClick() = documentPicker.launch(arrayOf("*/*"))
|
||||||
|
|
||||||
|
private fun resolveImportResultString(result : KeyReader.ImportResult) = when (result) {
|
||||||
|
KeyReader.ImportResult.Success -> R.string.import_keys_success
|
||||||
|
KeyReader.ImportResult.InvalidInputPath -> R.string.import_keys_invalid_input_path
|
||||||
|
KeyReader.ImportResult.InvalidKeys -> R.string.import_keys_invalid_keys
|
||||||
|
KeyReader.ImportResult.DeletePreviousFailed -> R.string.import_keys_delete_previous_failed
|
||||||
|
KeyReader.ImportResult.MoveFailed -> R.string.import_keys_move_failed
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -53,7 +53,10 @@
|
|||||||
<string name="prod_keys">Production Keys</string>
|
<string name="prod_keys">Production Keys</string>
|
||||||
<string name="title_keys">Title Keys</string>
|
<string name="title_keys">Title Keys</string>
|
||||||
<string name="import_keys_success">Successfully imported keys</string>
|
<string name="import_keys_success">Successfully imported keys</string>
|
||||||
<string name="import_keys_failed">Failed to import keys</string>
|
<string name="import_keys_invalid_input_path">The path to the provided keys is invalid</string>
|
||||||
|
<string name="import_keys_invalid_keys">The keys you tried to import are invalid</string>
|
||||||
|
<string name="import_keys_delete_previous_failed">Failed to delete the currently installed keys</string>
|
||||||
|
<string name="import_keys_move_failed">Failed to move the keys to the internal directory</string>
|
||||||
<!-- Settings - Display -->
|
<!-- Settings - Display -->
|
||||||
<string name="display">Display</string>
|
<string name="display">Display</string>
|
||||||
<string name="screen_orientation">Screen orientation</string>
|
<string name="screen_orientation">Screen orientation</string>
|
||||||
|
Loading…
Reference in New Issue
Block a user