mirror of
https://github.com/cainan-c/TaikoPythonTools.git
synced 2024-11-27 17:00:52 +01:00
Upload Files
This commit is contained in:
commit
13073b3957
14
TaikoFumenEncryptDecrypt/README.md
Normal file
14
TaikoFumenEncryptDecrypt/README.md
Normal file
@ -0,0 +1,14 @@
|
||||
# Taiko no Tatsujin - Simple Fumen Encryption and Decryption Tools
|
||||
|
||||
Simple Python 3 script to allows you to quickly decrypt and decompress or re-compress encrypt fumen files.
|
||||
Needed for some newer Taiko no Tatsujin titles.
|
||||
|
||||
Usage: Drag and Drop your Fumen files on `dec.py` to decrypt them
|
||||
Decrypted Files will be output in a folder called `decrypted`
|
||||
Or
|
||||
Drag and Drop your Fumen files on `enc.py` to encrypt them
|
||||
Encrypted files will be output in a folder called `encrypted`
|
||||
|
||||
No Encryption keys will be provided with this tool.
|
||||
|
||||
Define your encryption key in `config.toml`
|
1
TaikoFumenEncryptDecrypt/config.toml
Normal file
1
TaikoFumenEncryptDecrypt/config.toml
Normal file
@ -0,0 +1 @@
|
||||
key = ""
|
86
TaikoFumenEncryptDecrypt/dec.py
Normal file
86
TaikoFumenEncryptDecrypt/dec.py
Normal file
@ -0,0 +1,86 @@
|
||||
import os
|
||||
import sys
|
||||
import toml
|
||||
import gzip
|
||||
from Crypto.Cipher import AES
|
||||
from Crypto.Util.Padding import unpad
|
||||
|
||||
def decrypt_and_rename_to_gz(input_file, key):
|
||||
# Read the encrypted file
|
||||
with open(input_file, 'rb') as file:
|
||||
encrypted_bytes = file.read()
|
||||
|
||||
# Extract IV from the first 16 bytes of the encrypted file
|
||||
iv = encrypted_bytes[:16]
|
||||
|
||||
# Decrypt the rest of the file
|
||||
cipher = AES.new(key, AES.MODE_CBC, iv)
|
||||
decrypted_bytes = unpad(cipher.decrypt(encrypted_bytes[16:]), AES.block_size)
|
||||
|
||||
# Generate the new filename with .gz extension
|
||||
output_file = os.path.splitext(input_file)[0] + ".gz"
|
||||
|
||||
# Write decrypted data to the new .gz file
|
||||
with open(output_file, 'wb') as file:
|
||||
file.write(decrypted_bytes)
|
||||
|
||||
print(f"Decryption successful. Renamed and decrypted file saved as: {output_file}")
|
||||
|
||||
return output_file
|
||||
|
||||
def decompress_gz_file(input_file):
|
||||
# Generate the output filename without the .gz extension
|
||||
output_file = os.path.splitext(input_file)[0]
|
||||
|
||||
# Decompress the .gz file
|
||||
with gzip.open(input_file, 'rb') as f_in, open(output_file, 'wb') as f_out:
|
||||
f_out.write(f_in.read())
|
||||
|
||||
print(f"Decompression successful. Decompressed file saved as: {output_file}")
|
||||
|
||||
return output_file
|
||||
|
||||
def move_to_decrypted_folder(input_file, output_file):
|
||||
# Create the decrypted folder if it doesn't exist
|
||||
decrypted_folder = os.path.join(os.path.dirname(input_file), "decrypted")
|
||||
os.makedirs(decrypted_folder, exist_ok=True)
|
||||
|
||||
# Generate the output filename within the decrypted folder with .bin extension
|
||||
output_bin_file = os.path.join(decrypted_folder, os.path.splitext(os.path.basename(output_file))[0] + ".bin")
|
||||
|
||||
# Move the decrypted and decompressed file to the decrypted folder
|
||||
os.rename(output_file, output_bin_file)
|
||||
|
||||
print(f"Decrypted file moved to folder 'decrypted'. Decrypted file saved as: {output_bin_file}")
|
||||
|
||||
def main():
|
||||
# Check if there are any files dragged and dropped
|
||||
if len(sys.argv) < 2:
|
||||
print("Please drag and drop the AES-256-CBC encrypted .bin file(s) onto this program.")
|
||||
return
|
||||
|
||||
# Load configuration from config.toml
|
||||
config_file = "config.toml"
|
||||
with open(config_file, "r") as file:
|
||||
config = toml.load(file)
|
||||
|
||||
# Get key from configuration and convert it to bytes
|
||||
key_hex = config["key"]
|
||||
key = bytes.fromhex(key_hex)
|
||||
|
||||
for input_file in sys.argv[1:]:
|
||||
# Decrypt the file and get the renamed .gz file
|
||||
gz_file = decrypt_and_rename_to_gz(input_file, key)
|
||||
|
||||
# Decompress the .gz file and get the output filename
|
||||
output_file = decompress_gz_file(gz_file)
|
||||
|
||||
# Move the decrypted and decompressed file to the decrypted folder
|
||||
move_to_decrypted_folder(input_file, output_file)
|
||||
|
||||
# Remove the .gz file
|
||||
os.remove(gz_file)
|
||||
print(f"Removed the .gz file: {gz_file}")
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
87
TaikoFumenEncryptDecrypt/enc.py
Normal file
87
TaikoFumenEncryptDecrypt/enc.py
Normal file
@ -0,0 +1,87 @@
|
||||
import os
|
||||
import sys
|
||||
import toml
|
||||
import gzip
|
||||
from Crypto.Cipher import AES
|
||||
from Crypto.Util.Padding import pad
|
||||
|
||||
def compress_file(input_file):
|
||||
# Generate the output filename with .gz extension
|
||||
output_file = os.path.splitext(input_file)[0] + ".gz"
|
||||
|
||||
# Compress the input file
|
||||
with open(input_file, 'rb') as f_in, gzip.open(output_file, 'wb') as f_out:
|
||||
f_out.write(f_in.read())
|
||||
|
||||
print(f"Compression successful. Compressed file saved as: {output_file}")
|
||||
|
||||
return output_file
|
||||
|
||||
def encrypt_file(input_file, key, iv):
|
||||
# Read the compressed file
|
||||
with open(input_file, 'rb') as file:
|
||||
plaintext = file.read()
|
||||
|
||||
# Encrypt the file
|
||||
cipher = AES.new(key, AES.MODE_CBC, iv)
|
||||
ciphertext = cipher.encrypt(pad(plaintext, AES.block_size))
|
||||
|
||||
# Prepend the IV to the ciphertext
|
||||
encrypted_data = iv + ciphertext
|
||||
|
||||
# Generate the output filename
|
||||
output_file = os.path.splitext(input_file)[0] + ".encrypted.bin"
|
||||
|
||||
# Write the encrypted data to the output file
|
||||
with open(output_file, 'wb') as file:
|
||||
file.write(encrypted_data)
|
||||
|
||||
print(f"Encryption successful. Encrypted file saved as: {output_file}")
|
||||
|
||||
return output_file
|
||||
|
||||
def move_to_encrypted_folder(input_file, output_file):
|
||||
# Create the encrypted folder if it doesn't exist
|
||||
encrypted_folder = os.path.join(os.path.dirname(input_file), "encrypted")
|
||||
os.makedirs(encrypted_folder, exist_ok=True)
|
||||
|
||||
# Generate the output filename within the encrypted folder
|
||||
output_encrypted_file = os.path.join(encrypted_folder, os.path.basename(input_file))
|
||||
|
||||
# Move the encrypted file to the encrypted folder
|
||||
os.rename(output_file, output_encrypted_file)
|
||||
|
||||
print(f"Encrypted file moved to folder 'encrypted'. Encrypted file saved as: {output_encrypted_file}")
|
||||
|
||||
def main():
|
||||
# Check if there are any files dragged and dropped
|
||||
if len(sys.argv) < 2:
|
||||
print("Please drag and drop the files you want to re-compress and re-encrypt onto this program.")
|
||||
return
|
||||
|
||||
# Load configuration from config.toml
|
||||
config_file = "config.toml"
|
||||
with open(config_file, "r") as file:
|
||||
config = toml.load(file)
|
||||
|
||||
# Get key and IV from configuration and convert them to bytes
|
||||
key_hex = config["key"]
|
||||
key = bytes.fromhex(key_hex)
|
||||
iv = bytes.fromhex("FF" * 16) # IV set to FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF
|
||||
|
||||
for input_file in sys.argv[1:]:
|
||||
# Compress the file
|
||||
compressed_file = compress_file(input_file)
|
||||
|
||||
# Encrypt the compressed file
|
||||
encrypted_file = encrypt_file(compressed_file, key, iv)
|
||||
|
||||
# Move the encrypted file to the encrypted folder
|
||||
move_to_encrypted_folder(input_file, encrypted_file)
|
||||
|
||||
# Remove the compressed file
|
||||
os.remove(compressed_file)
|
||||
print(f"Removed the compressed file: {compressed_file}")
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
22
TaikoFumenTimingReplace/README.md
Normal file
22
TaikoFumenTimingReplace/README.md
Normal file
@ -0,0 +1,22 @@
|
||||
# Taiko no Tatsujin - Fumen Timing Window Replace Script
|
||||
|
||||
Simple Python 3 script to replace that can replace the timing windows in a Taiko Gen 3 fumen
|
||||
|
||||
Usage: replace_timing.py inFile outFile timingWindow
|
||||
|
||||
Timing Options: Standard, Hitnarrow, Hitwide or a User Defined Option.
|
||||
|
||||
|
||||
Detects what difficulty timing window to use based on the Input's filename.
|
||||
Easy(\_e) and Normal(\_n) will use the easy timing
|
||||
Hard(\_h), Extreme(\_e) and Ura(\_x) will use hard timing
|
||||
|
||||
If it fails to detect difficulty, it'll use the hard timing values instead.
|
||||
|
||||
To add custom timing values, add a new section to timing.ini and define it's name and add values for each judgement.
|
||||
A template can be found [here](TaikoFumenTimingReplace/resource/template.ini)
|
||||
|
||||
# Credits
|
||||
|
||||
swigz27 - Initial code.
|
||||
Yuki [(Nerdy-boi)](https://github.com/Nerdy-boi) - Code optimisations and tweaks.
|
90
TaikoFumenTimingReplace/replace_timing.py
Normal file
90
TaikoFumenTimingReplace/replace_timing.py
Normal file
@ -0,0 +1,90 @@
|
||||
#Python script that replaces the timing values in a Fumen and spits out a new file.
|
||||
import configparser, struct, sys
|
||||
|
||||
#Function to handle converting a float to a binary array
|
||||
def convert_to_bytearray(f):
|
||||
val = hex(struct.unpack('>I', struct.pack('<f', f))[0])
|
||||
Fixed_hex = val[2:]
|
||||
return bytes.fromhex(Fixed_hex)
|
||||
|
||||
#Load ini file
|
||||
config = configparser.ConfigParser()
|
||||
config.sections()
|
||||
config.read('timing.ini')
|
||||
|
||||
# Define possible endings of file name and their difficulties.
|
||||
easy = '_e'
|
||||
normal = '_n'
|
||||
hard = '_h'
|
||||
extreme = '_m'
|
||||
ura = '_x'
|
||||
|
||||
# Check to see if anything (other than the python file) has been entered.
|
||||
# If two files/the timing window haven't been entered, it'll just throw a normal Python error.
|
||||
if len(sys.argv) > 1:
|
||||
|
||||
#Check if standard/hitwide/hitnarrow/custom was typed in the console
|
||||
if sys.argv[3].lower() == 'standard':
|
||||
GOOD = config.getfloat('standard', 'good')
|
||||
OK = config.getfloat('standard', 'ok')
|
||||
BAD = config.getfloat('standard', 'bad')
|
||||
GOOD_EASY = config.getfloat('standard', 'good_easy')
|
||||
OK_EASY = config.getfloat('standard', 'ok_easy')
|
||||
BAD_EASY = config.getfloat('standard', 'bad_easy')
|
||||
|
||||
else:
|
||||
try:
|
||||
GOOD = config.getfloat(sys.argv[3].lower(), 'good')
|
||||
OK = config.getfloat(sys.argv[3].lower(), 'ok')
|
||||
BAD = config.getfloat(sys.argv[3].lower(), 'bad')
|
||||
GOOD_EASY = config.getfloat(sys.argv[3].lower(), 'good_easy')
|
||||
OK_EASY = config.getfloat(sys.argv[3].lower(), 'ok_easy')
|
||||
BAD_EASY = config.getfloat(sys.argv[3].lower(), 'bad_easy')
|
||||
|
||||
except:
|
||||
print("Invalid Input")
|
||||
exit()
|
||||
|
||||
#Convert the floats to a binary array
|
||||
GOOD_BYTES = (convert_to_bytearray(GOOD))
|
||||
OK_BYTES = (convert_to_bytearray(OK))
|
||||
BAD_BYTES = (convert_to_bytearray(BAD))
|
||||
|
||||
GOOD_EASY_BYTES = (convert_to_bytearray(GOOD_EASY))
|
||||
OK_EASY_BYTES = (convert_to_bytearray(OK_EASY))
|
||||
BAD_EASY_BYTES = (convert_to_bytearray(BAD_EASY))
|
||||
|
||||
#Define binary arrays
|
||||
timing_window_hard = (GOOD_BYTES + OK_BYTES + BAD_BYTES) * 36
|
||||
timing_window_easy = (GOOD_EASY_BYTES + OK_EASY_BYTES + BAD_EASY_BYTES) * 36
|
||||
|
||||
# Define the input and output files.
|
||||
inFile = sys.argv[1]
|
||||
outFile = sys.argv[2]
|
||||
|
||||
# Opens the input file, saves it as the output,
|
||||
# This was the only way I managed was able to acomplish the right output.
|
||||
with open(inFile, "rb") as old, open(outFile, "wb") as new:
|
||||
old.seek(0)
|
||||
new.write(old.read())
|
||||
|
||||
# Actually replaces the timing windows
|
||||
chart_new = open(outFile, "rb+")
|
||||
chart_new.seek(0)
|
||||
|
||||
# Checks if _e, _n, etc is present in the file name:
|
||||
# If it is, it'll use timing_easy,
|
||||
# If it isn't present, it will use timing_hard instead.
|
||||
if easy in inFile:
|
||||
chart_new.write(timing_window_easy)
|
||||
chart_new.close()
|
||||
elif normal in inFile:
|
||||
chart_new.write(timing_window_easy)
|
||||
chart_new.close()
|
||||
else:
|
||||
chart_new.write(timing_window_hard)
|
||||
chart_new.close()
|
||||
|
||||
else:
|
||||
print("TaikoFumenTimingReplace\nUsage:",sys.argv[0], "inFile outFile timingWindow",
|
||||
"\n\nTiming Options: Standard, Hitnarrow, Hitwide or a User Defined Option.")
|
7
TaikoFumenTimingReplace/resource/template.ini
Normal file
7
TaikoFumenTimingReplace/resource/template.ini
Normal file
@ -0,0 +1,7 @@
|
||||
[Template]
|
||||
good = 0.0
|
||||
ok = 0.0
|
||||
bad = 0.0
|
||||
good_easy = 0.0
|
||||
ok_easy = 0.0
|
||||
bad_easy = 0.0
|
23
TaikoFumenTimingReplace/timing.ini
Normal file
23
TaikoFumenTimingReplace/timing.ini
Normal file
@ -0,0 +1,23 @@
|
||||
[standard]
|
||||
good = 25.0250015258789
|
||||
ok = 75.0750045776367
|
||||
bad = 108.441665649414
|
||||
good_easy = 41.7083358764648
|
||||
ok_easy = 108.441665649414
|
||||
bad_easy = 125.125
|
||||
|
||||
[hitnarrow]
|
||||
good = 25.0250015258789
|
||||
ok = 58.3916664123535
|
||||
bad = 108.441665649414
|
||||
good_easy = 41.7083358764648
|
||||
ok_easy = 75.0750045776367
|
||||
bad_easy = 125.125
|
||||
|
||||
[hitwide]
|
||||
good = 75.0750045776367
|
||||
ok = 91.7583312988281
|
||||
bad = 108.441665649414
|
||||
good_easy = 91.7583312988281
|
||||
ok_easy = 108.441665649414
|
||||
bad_easy = 125.125
|
Loading…
Reference in New Issue
Block a user