nano-codec / inference.py
taresh18's picture
Upload inference.py with huggingface_hub
c82ec83 verified
"""
nano-codec inference: reconstruct audio through the codec.
Usage:
python inference.py --input input.wav --output reconstructed.wav
Downloads model weights from HuggingFace on first run.
"""
import argparse
import torch
import soundfile as sf
import torchaudio
import yaml
from huggingface_hub import hf_hub_download
from model import RVQCodec
REPO_ID = "taresh18/nano-codec"
def load_model(device="cpu"):
model_path = hf_hub_download(REPO_ID, "model.pt")
config_path = hf_hub_download(REPO_ID, "config.yaml")
with open(config_path) as f:
cfg = yaml.safe_load(f)
model = RVQCodec(
in_ch=1,
latent_ch=cfg['latent_dim'],
K=cfg['codebook_size'],
num_rvq_levels=cfg['num_rvq_levels'],
codebook_dim=cfg.get('codebook_dim', 8),
)
state = torch.load(model_path, map_location=device, weights_only=True)
model.load_state_dict(state)
model = model.to(device)
model.eval()
return model, cfg
def reconstruct(model, audio_path, output_path, sample_rate=16000, chunk_size=16384, device="cpu"):
audio, sr = sf.read(audio_path, dtype='float32')
if audio.ndim > 1:
audio = audio.mean(axis=1)
waveform = torch.from_numpy(audio).unsqueeze(0)
if sr != sample_rate:
waveform = torchaudio.functional.resample(waveform, sr, sample_rate)
waveform = waveform / waveform.abs().max().clamp(min=1e-8)
total_samples = waveform.shape[1]
pad_len = (chunk_size - total_samples % chunk_size) % chunk_size
if pad_len > 0:
waveform = torch.nn.functional.pad(waveform, (0, pad_len))
recon_chunks = []
with torch.no_grad():
for start in range(0, waveform.shape[1], chunk_size):
chunk = waveform[:, start:start + chunk_size].unsqueeze(0).to(device)
recon, _, _, _ = model(chunk)
recon = recon[..., :chunk_size]
recon_chunks.append(recon.cpu())
recon_full = torch.cat(recon_chunks, dim=-1)
recon_full = recon_full[0, :, :total_samples]
sf.write(output_path, recon_full[0].float().numpy(), sample_rate)
print(f"saved: {output_path} ({total_samples / sample_rate:.2f}s)")
def main():
parser = argparse.ArgumentParser(description="nano-codec inference")
parser.add_argument("--input", required=True, help="input wav file")
parser.add_argument("--output", default="reconstructed.wav", help="output wav file")
parser.add_argument("--device", default="cuda" if torch.cuda.is_available() else "cpu")
args = parser.parse_args()
model, cfg = load_model(device=args.device)
reconstruct(
model,
args.input,
args.output,
sample_rate=cfg['sample_rate'],
chunk_size=cfg['chunk_size'],
device=args.device,
)
if __name__ == "__main__":
main()