Osz2.py
A python library for working with osz2 & osf2 files
Install / Use
/learn @Lekuruu/Osz2.pyREADME
osz2.py
osz2.py is a Python library for reading osz2 files. It's a direct port of the existing Osz2Decryptor project by xxCherry and osz2-go by me. The Python port itself was done by @ascenttree; all credit goes to them. I took part in code refactoring and optimizing the performance by moving the heavy crypto primitives into a native C extension (osz2.crypto) on top of NumPy, bringing encryption time down to ~100 ms instead of 25 seconds.
This project won't provide beatmap parsing support. You will have to implement that by yourself, if you decide to use this library for implementing the beatmap submission system.
Installation
pip install osz2
Or install from source:
git clone https://github.com/Lekuruu/osz2.py
cd osz2.py
pip install -e .
Usage
This repository provides a command-line interface for easy testing:
python -m osz2 decrypt <input.osz2> <output_directory>
python -m osz2 encrypt <target_directory> <output.osz2>
But that's not all!
Here is an example of how to use osz2.py as a library:
from osz2 import Osz2Package, MetadataType
# Parse package from file
package = Osz2Package.from_file("beatmap.osz2")
# Access metadata
print("Title:", package.metadata.get(MetadataType.Title))
print("Artist:", package.metadata.get(MetadataType.Artist))
print("Creator:", package.metadata.get(MetadataType.Creator))
print("Difficulty:", package.metadata.get(MetadataType.Difficulty))
# Access files
for file in package.files:
print(f"File: {file.filename}, Size: {len(file.content)} bytes")
# Extract specific files
for file in package.files:
if not file.filename.endswith(".osu"):
continue
with open(file.filename, "wb") as f:
f.write(file.content)
# Create a regular .osz package
with open("beatmap.osz", "wb") as f:
f.write(package.create_osz_package())
Metadata-only Mode
If you only need to read metadata without extracting files, you can use the metadata_only parameter:
# Only parse metadata
package = Osz2Package.from_file("beatmap.osz2", metadata_only=True)
# Access metadata
print("Title:", package.metadata.get(MetadataType.Title))
print("BeatmapSet ID:", package.metadata.get(MetadataType.BeatmapSetID))
Alternative Constructors
# From file path
package = Osz2Package.from_file("beatmap.osz2")
# From bytes
with open("beatmap.osz2", "rb") as f:
data = f.read()
package = Osz2Package.from_bytes(data)
# From an io.BufferedReader-like object, e.g. a file stream
with open("beatmap.osz2", "rb") as f:
package = Osz2Package(f)
Exporting an osz2 package
You can initialize and export osz2 packages from a directory:
from osz2 import Osz2Package, MetadataType
# Initialize package from a directory containing beatmap files
package = Osz2Package.from_directory("./my_beatmap_folder")
# Export to osz2 format
osz2_data = package.export()
with open("output.osz2", "wb") as f:
f.write(osz2_data)
# Or save directly to a file
package.save("./output.osz2")
Managing Files
You can add, remove, and modify files within a package:
from osz2 import Osz2Package, MetadataType
# Create a new package
package = Osz2Package()
# Add metadata (required for export)
package.add_metadata(MetadataType.Title, "My Beatmap")
package.add_metadata(MetadataType.Artist, "Artist Name")
package.add_metadata(MetadataType.Creator, "Mapper Name")
package.add_metadata(MetadataType.BeatmapSetID, 123456)
# Add a file from memory
beatmap_content = b"osu file format v14\n..."
package.add_file("my_beatmap.osu", beatmap_content)
# Add a file from disk
package.add_file_from_disk("audio.mp3", "./path/to/audio.mp3")
# Add an entire directory (non-recursive)
package.add_directory("./beatmap_files", recursive=False)
# Add an entire directory (recursive, preserves folder structure)
package.add_directory("./beatmap_folder", recursive=True)
# Remove a file
package.remove_file("old_file.osu")
# Find a file by name
file = package.find_file_by_name("audio.mp3")
if file:
print(f"Found: {file.filename}, size: {file.size}")
# Set beatmap IDs
package.set_beatmap_id("my_beatmap.osu", 789012)
# Export the package
package.save("my_beatmap.osz2")
Managing Metadata
Metadata can be added, retrieved, and removed:
from osz2 import Osz2Package, MetadataType
package = Osz2Package.from_file("beatmap.osz2")
# Add or update metadata
package.add_metadata(MetadataType.Title, "New Title")
package.add_metadata(MetadataType.Artist, "New Artist")
# Get metadata
title = package.get_metadata(MetadataType.Title)
print(f"Title: {title}")
# Remove metadata
package.remove_metadata(MetadataType.Difficulty)
# Convenience method for setting beatmapset ID
package.set_beatmapset_id(999999)
# Save changes
package.save("modified.osz2")
Applying a patch
When developing an implementation of the beatmap submission system, this could come in handy:
# Assuming you have a source osz2 file and a patch file
osz2_file = b"..."
patch_file = b"..."
updated_osz2 = osz2.apply_bsdiff_patch(osz2_file, patch_file)
osz2 = Osz2Package.from_bytes(updated_osz2)
Using osu!stream .osf2 files
I have not tested this, but in theory this should work by passing in KeyType.OSF2 when initializing the osz2 package:
osf2 = Osz2Package.from_file("beatmap.osf2", key_type=KeyType.OSF2)
You can also specify this when using the command-line interface:
python -m osz2 <input.osz2> <output_directory> --key-type osf2
Building the crypto.c extension
If you change any code under osz2/crypto.c, rebuild the module in place before running tests:
python -m venv venv
source venv/bin/activate
pip install -e .
Or, if you only need the extension locally:
python setup.py build_ext --inplace
The resulting shared object is picked up automatically by the package import system.
Related Skills
node-connect
347.0kDiagnose OpenClaw node connection and pairing failures for Android, iOS, and macOS companion apps
frontend-design
107.8kCreate distinctive, production-grade frontend interfaces with high design quality. Use this skill when the user asks to build web components, pages, or applications. Generates creative, polished code that avoids generic AI aesthetics.
openai-whisper-api
347.0kTranscribe audio via OpenAI Audio Transcriptions API (Whisper).
qqbot-media
347.0kQQBot 富媒体收发能力。使用 <qqmedia> 标签,系统根据文件扩展名自动识别类型(图片/语音/视频/文件)。
