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
def write_chunk(f, chunk_type, data): """Write PNG chunk with CRC""" f.write(struct.pack('>I', len(data))) f.write(chunk_type) f.write(data) crc = zlib.crc32(chunk_type + data) & 0xffffffff f.write(struct.pack('>I', crc)) xtool -mpng+reflate
def rebuild_png_with_mpng(input_path, output_path, streams, recompress_level=6, replace_map=None): """Rebuild PNG with modified MPNG streams""" with open(input_path, 'rb') as fin, open(output_path, 'wb') as fout: fout.write(fin.read(8)) # PNG signature while True: chunk_type, data, _ = read_chunk(f) if
PNG_SIGNATURE = b'\x89PNG\r\n\x1a\n' CHUNK_TYPE_MPNG = b'mPNg' # Custom MPNG chunk CHUNK_TYPE_IDAT = b'IDAT' CHUNK_TYPE_IEND = b'IEND' while True: chunk_type