Xtool -mpng+reflate Now
if chunk_type == CHUNK_TYPE_IEND: break def mpng_reflate_cli(args): input_png = args.input output_png = args.output or input_png.replace('.png', '_reflated.png') level = args.recompress_level or 6 extract_only = args.extract_only replace_data = args.replace
while True: chunk_type, data, _ = read_chunk(f) if chunk_type == CHUNK_TYPE_MPNG: # MPNG chunk structure: [index:4][compressed_data...] idx = struct.unpack('>I', data[:4])[0] compressed = data[4:] streams.append((idx, compressed)) elif chunk_type == CHUNK_TYPE_IEND: break return streams def reflate_stream(compressed_data, level=6, extract_only=False): """Reflate: decompress then recompress zlib stream""" decompressed = zlib.decompress(compressed_data) if extract_only: return decompressed # raw decompressed data recompressed = zlib.compress(decompressed, level) return recompressed xtool -mpng+reflate
def read_chunk(f): """Read PNG chunk: length, type, data, crc""" len_data = struct.unpack('>I', f.read(4))[0] chunk_type = f.read(4) data = f.read(len_data) crc = struct.unpack('>I', f.read(4))[0] return chunk_type, data, crc data[:4])[0] compressed = data[4:] streams.append((idx
if extract_only: for idx, comp in streams: raw = zlib.decompress(comp) out_file = f"{Path(input_png).stem}_stream_{idx}.raw" Path(out_file).write_bytes(raw) print(f"Extracted stream {idx} to {out_file}") return crc""" len_data = struct.unpack('>