import struct from typing import Dict, List, Optional class TwoDX: """ Packer/unpacker class for a bytestream representing a `.2dx` file. """ def __init__(self, data: Optional[bytes] = None) -> None: self.__name: Optional[str] = None self.__files: Dict[str, bytes] = {} if data is not None: self.__parse_file(data) def __parse_file(self, data: bytes) -> None: # Parse file header (name, headerSize, numfiles) = struct.unpack('<16sII', data[0:24]) self.__name = name.split(b'\x00')[0].decode('ascii') if headerSize != (72 + (4 * numfiles)): raise Exception('Unrecognized 2dx file header!') fileoffsets = struct.unpack('<' + ''.join(['I' for _ in range(numfiles)]), data[72:(72 + (4 * numfiles))]) fileno = 1 for offset in fileoffsets: (magic, headerSize, wavSize, _, track, _, attenuation, loop) = struct.unpack( '<4sIIhhhhi', data[offset:(offset + 24)], ) if magic != b'2DX9': raise Exception('Unrecognized entry in file!') if headerSize != 24: raise Exception('Unrecognized subheader in file!') wavOffset = offset + headerSize wavData = data[wavOffset:(wavOffset + wavSize)] self.__files[f'{self.__name}_{fileno}.wav'] = wavData fileno = fileno + 1 @property def name(self) -> str: return self.__name def set_name(self, name: str) -> None: if len(name) <= 16: self.__name = name else: raise Exception('Name of archive too long!') @property def filenames(self) -> List[str]: return [f for f in self.__files] def read_file(self, filename: str) -> bytes: return self.__files[filename] def write_file(self, filename: str, data: bytes) -> None: self.__files[filename] = data def get_new_data(self) -> bytes: if not self.__files: raise Exception('No files to write!') if not self.__name: raise Exception('2dx archive name not set!') name = self.__name.encode('ascii') while len(name) < 16: name = name + b'\x00' filedata = [self.__files[x] for x in self.__files] # Header length is also the base offset for the first file baseoffset = 72 + (4 * len(filedata)) data = [struct.pack('<16sII', name, baseoffset, len(filedata)) + (b'\x00' * 48)] # Calculate offset this will go to for bytedata in filedata: # Add where this file will go, then calculate the length data.append(struct.pack('