# Copyright (c) Meta Platforms, Inc. and affiliates.
# All rights reserved.

# This source code is licensed under the license found in the
# LICENSE file in the root directory of this source tree.
"""
Run watermark generation and evaluation:

python -m evals.main_wm \
    --output_dir outputs/ \
    --nsamples 16 \
    --batch_size 1 \
    --temperature 1.0 \
    --steps 200 \
    --wm_method maryland \
    --wm_streams 1 2 3 4 \
    --wm_delta 2.0 \
    --wm_ngram 0 \
    --wm_seed 0 \
    --audio_dir audio_prompts/ \
    --mimi_weight checkpoints/finetunes/mimi_ft.pth
"""

import os
import torch
import numpy as np
import pandas as pd
import argparse
from huggingface_hub import hf_hub_download
import sentencepiece
import sphn
import glob
from tqdm import tqdm
import time
import random
import pickle
from collections import defaultdict
from scipy import stats, special

from models.moshi.models import loaders, LMGen
from models.moshi.utils import bool_inst

from training import get_validation_augs, get_dummy_augs
from watermark.engine import get_wm_window_hash, GENERATOR
from watermark.sync import SyncPattern

def get_binomial_pval(x, n, p):
    """
    Calculates the p-value for a one-sided binomial test (greater).
    Args:
        x: The number of successes (e.g., number of matching bits).
        n: The number of trials (e.g., total number of bits).
        p: The hypothesized probability of success under the null hypothesis (e.g., 0.5 for random chance).
    Returns:
        The p-value.
    """
    # p_value = stats.binomtest(x,n,p=p,alternative='greater').pvalue
    p_value = special.betainc(x, 1 + n - x, p)
    return p_value

def seed_all(seed):
    torch.manual_seed(seed)
    if torch.cuda.is_available():
        torch.cuda.manual_seed(seed)
        torch.cuda.manual_seed_all(seed)  # for multi-GPU setups
    random.seed(seed)
    np.random.seed(seed)
    torch.backends.cudnn.deterministic = True
    torch.backends.cudnn.benchmark = False

def load_clustering_maps(clustering_dir, target_min_count=None, device="cpu", vocab_size=2048):
    """
    Load clustering maps from pickle files based on specific channel suffixes.
    Returns: maps[channel][method][min_count] = (k_actual, map_tensor)
    """
    maps = defaultdict(lambda: defaultdict(dict))
    channel_suffixes = {0: "first_0", 1: "rest_0", 2: "rest_1", 3: "rest_2"}

    for c, suffix in channel_suffixes.items():
        # for method in ["leiden", "louvain"]:
        for method in ["leiden"]:
            files = sorted(glob.glob(os.path.join(clustering_dir, f"*{method}*{suffix}*.pkl")))
            if not files: continue

            try:
                with open(files[0], "rb") as f:
                    data = pickle.load(f)
                
                # Determine keys to load: specific target or all available
                keys_to_load = []
                if target_min_count is not None:
                    if target_min_count in data:
                        keys_to_load.append(target_min_count)
                    else:
                        # Fallback logic if target missing
                        available_keys = sorted(data.keys())
                        if available_keys:
                            mid = len(available_keys) // 2
                            fb_key = available_keys[mid]
                            print(f"  > Ch {c} {method}: Target {target_min_count} missing. Using {fb_key}")
                            keys_to_load.append(fb_key)
                else:
                    keys_to_load = sorted(data.keys())

                for key in keys_to_load:
                    cmap = torch.as_tensor(data[key], device=device, dtype=torch.long)
                    
                    # Create full map: copy existing clusters, assign unique IDs to singletons
                    full_map = torch.full((vocab_size,), -1, device=device, dtype=torch.long)
                    limit = min(len(cmap), vocab_size)
                    full_map[:limit] = cmap[:limit]
                    
                    unmapped = (full_map == -1)
                    k_clusters = int(cmap.max().item()) + 1
                    if unmapped.any():
                        full_map[unmapped] = torch.arange(k_clusters, k_clusters + unmapped.sum(), device=device)
                    
                    maps[c][method][key] = (int(full_map.max().item()) + 1, full_map)
                
            except Exception as e:
                print(f"Failed to load {files[0]}: {e}")

    return maps

def build_stream_ngrams_from_full_stream(stream_tokens: torch.Tensor, wm_ngram: int, device='cpu'):
    """
    stream_tokens: 1D tensor length T
    returns: tensor shape [T, wm_ngram] dtype long on `device`.
    Each row i contains tokens[start:i+1] left-padded with zeros to length wm_ngram,
    where start = max(0, i+1-wm_ngram).
    If wm_ngram <= 0 returns tensor shape [1, 0] for backward compatibility.
    """
    T = int(stream_tokens.shape[-1])
    n = int(wm_ngram)
    if n <= 0:
        return torch.zeros((1, 0), dtype=torch.long, device=device)
    s = stream_tokens.to(torch.long).to(device)
    rows = []
    for i in range(T):
        start = max(0, i + 1 - n)
        ctx = s[start : i + 1]  # length L <= n
        L = ctx.shape[-1]
        if L < n:
            pad = torch.zeros((n - L,), dtype=torch.long, device=device)
            ctx = torch.cat([pad, ctx], dim=0)
        rows.append(ctx.unsqueeze(0))  # 1 x n
    return torch.cat(rows, dim=0)  # T x n

def compute_watermark_scores(wm_stream, ngrams, audio_vocab_size, gamma, wm_seed, device='cpu', clustering_map=None):
    """Compute watermark scores for a given stream of tokens.
    ngrams may be:
      - shape (1, n): same ngram used for all tokens (backwards compatible)
      - shape (T, n): one ngram per token/time-step (use that per-token hash)
      - shape (1, 0): empty ngram (no context)
    """
    # Track unseen tokens
    seen_tokens = set()
    green_mask = torch.zeros_like(wm_stream, dtype=torch.bool)
    to_score_mask = torch.zeros_like(wm_stream, dtype=torch.bool)

    effective_vocab_size = int(clustering_map.max().item()) + 1 if clustering_map is not None else audio_vocab_size
    T = wm_stream.shape[-1]

    # Precompute window hashes if we have a per-timestep ngrams matrix
    per_timestep_hashes = None
    if ngrams is not None and ngrams.numel() > 0 and ngrams.shape[0] == T:
        # This is executed for h > 0
        # ngrams: T x n
        per_timestep_hashes = get_wm_window_hash(ngrams, wm_seed, clustering_map=clustering_map)
        # ensure it's on CPU / convertible to int per element
        per_timestep_hashes = per_timestep_hashes.cpu()

    # If ngrams is a single-row or empty, compute single hash once
    single_hash_val = None
    if per_timestep_hashes is None:
        # This is executed for h = 0
        # pass through original behaviour: compute single hash for provided ngrams (could be 1 x n or 1 x 0)
        single_hash = get_wm_window_hash(ngrams, wm_seed, clustering_map=clustering_map)
        single_hash_val = int(single_hash[0].item())

    for ii, token in enumerate(wm_stream):
        # Decide which seed to use
        if per_timestep_hashes is not None:
            seed = int(per_timestep_hashes[ii].item())
        else:
            seed = single_hash_val

        GENERATOR.manual_seed(seed)
        vocab_perm = torch.randperm(effective_vocab_size, generator=GENERATOR)
        greenlist = vocab_perm[:int(gamma * effective_vocab_size)]  # list of tokens/clusters
        
        token_val = token.cpu().item()
        
        if clustering_map is not None:
             cluster_id = clustering_map[token.long()].item()
             is_green = cluster_id in greenlist
        else:
             is_green = token_val in greenlist

        green_mask[ii] = is_green
        if token_val not in seen_tokens:
            to_score_mask[ii] = 1
            seen_tokens.add(token_val)
            
    return green_mask, to_score_mask

def run_watermark_eval(args, clustering_maps=None, config_name="standard"):
    """Generate audio with watermarks and evaluate watermark preservation"""
    # Load models
    text_tokenizer = sentencepiece.SentencePieceProcessor(args.tokenizer)
    mimi = loaders.get_mimi(args.mimi_weight, args.device)
    mimi_ori = loaders.get_mimi(args.mimi_weight_ori, args.device)
    lm = loaders.get_moshi_lm(args.moshi_weight, device=args.device)
    lm_gen = LMGen(lm)

    # Set generation parameters
    lm_gen.temp = args.temperature
    frame_size = int(mimi.sample_rate / mimi.frame_rate)
    
    # Configure watermark
    args.wm_streams = [int(x) for x in args.wm_streams]
    lm_gen.wm = args.wm_method
    lm_gen.wm_ngram = args.wm_ngram
    lm_gen.wm_streams = args.wm_streams
    lm_gen.wm_seed = args.wm_seed
    lm_gen.wm_aux_params["delta"] = args.wm_delta
    lm_gen.wm_aux_params["gamma"] = args.wm_gamma
    
    # Inject clustering maps if available
    if clustering_maps:
        lm_gen.wm_aux_params["clustering_maps"] = clustering_maps
        
    audio_vocab_size = lm_gen.lm_model.card
    print(f"Watermarking config: method={lm_gen.wm}, streams={lm_gen.wm_streams}, "
          f"ngram={lm_gen.wm_ngram}, delta={lm_gen.wm_aux_params['delta']}")
    print(f"--- Running Configuration: {config_name} ---")

    # Configure synchronization if needed
    if args.wm_sync:
        sync_pattern = SyncPattern()
        sync_pattern.to(args.device)

    # Handle prompt preparation if using prompts
    nsamples = args.nsamples
    audio_files = []
    
    if args.use_prompts and args.audio_dir:
        # Find all audio files in directory
        for ext in ['*.wav', '*.mp3', '*.ogg', '*.flac']:
            audio_files.extend(glob.glob(os.path.join(args.audio_dir, ext)))
            if not audio_files:
                audio_files.extend(glob.glob(os.path.join(args.audio_dir, f"*/*/{ext}")))
        
        # Sort and limit number of files if specified
        audio_files = sorted(audio_files)
        if args.nsamples > 0:
            audio_files = audio_files[:args.nsamples]
        
        if len(audio_files) == 0:
            print(f"No audio files found in {args.audio_dir}. Proceeding without prompts.")
        else:
            nsamples = min(nsamples, len(audio_files))
            print(f"Using {nsamples} audio files as prompts")
    
    # Initialize global containers for tokens and watermark results
    tokens_saved = [dict() for _ in range(nsamples)]
    global_watermark_results = []
    
    # Loop over samples in batches
    for batch_start in tqdm(range(0, nsamples, args.batch_size)):
        batch_size = min(args.batch_size, nsamples - batch_start)
        
        # Process prompt codes for current batch if using prompts
        current_prompt_codes = None
        if args.use_prompts and audio_files:
            batch_files = audio_files[batch_start:batch_start + batch_size]
            
            # Process each audio file in current batch
            prompt_codes_list = []
            for audio_path in batch_files:
                # Load and resample audio
                sample_pcm, sample_sr = sphn.read(audio_path, duration_sec=args.duration_sec)
                sample_pcm = sphn.resample(
                    sample_pcm, src_sample_rate=sample_sr, dst_sample_rate=mimi.sample_rate
                )
                sample_pcm = torch.tensor(sample_pcm, device=args.device).unsqueeze(0)
                
                with torch.no_grad():
                    # Encode audio
                    prompt_code = mimi_ori.encode(sample_pcm)
                    prompt_codes_list.append(prompt_code)
            
            # Pad prompts to the same length for current batch
            assert len(prompt_codes_list) == batch_size, "Batch size mismatch"

            len_toks = [code.shape[-1] for code in prompt_codes_list]
            max_len_toks = max(len_toks)
            min_pad = 0
            end_prompt_idx = max_len_toks + min_pad
            start_answer_idx = end_prompt_idx
            pad = torch.tensor(lm.zero_token_id, device=prompt_codes_list[0].device).unsqueeze(0).unsqueeze(0)
            prompt_codes_list = [
                torch.cat([
                    pad.repeat((1, code.shape[-2], end_prompt_idx - code.shape[-1])), 
                    code
                ], dim=-1) 
                for code in prompt_codes_list
            ]
            current_prompt_codes = torch.cat(prompt_codes_list, dim=0)
        
        else:
            raise ValueError("No audio files found in the specified directory.")
        
        batch_all_tokens = []
        batch_all_audios = []
        batch_all_texts = [[] for _ in range(batch_size)]
        
        # Continue with generation using current_prompt_codes
        with torch.no_grad():
            with mimi.streaming(batch_size), lm_gen.streaming(batch_size):
                for step in range(args.steps):
                    if current_prompt_codes is not None and step < current_prompt_codes.shape[-1]:
                        codes = current_prompt_codes[:, :, step:step + 1]
                    else:
                        chunk = torch.zeros((batch_size, 1, frame_size), dtype=torch.float, device=args.device)
                        codes = mimi_ori.encode(chunk)
                    
                    # print(codes.shape)
                    tokens = lm_gen.step(codes[:, :, :1], force_epad= (step==start_answer_idx))
                    # tokens = lm_gen.step(codes[:, :, :1], force_epad=False)
                    if tokens is None:
                        continue
                    if current_prompt_codes is not None and step < start_answer_idx:  # min_pad as before
                        continue
                        
                    batch_all_tokens.append(tokens.detach().cpu())
                    
                    # Process text tokens
                    text_tokens = tokens[:, 0, :]  # b 1
                    for idx in range(batch_size):
                        text_token = text_tokens[idx].item()
                        if text_token not in (0, 3):  # Skip special tokens
                            _text = text_tokenizer.id_to_piece(text_token)
                            _text = _text.replace("▁", " ")
                            batch_all_texts[idx].append(_text)
                    
                    # Decode audio tokens
                    audio_tokens = tokens[:, 1:, :]
                    pcms = mimi.decode(audio_tokens)
                    batch_all_audios.append(pcms)
        
        # Concatenate batch tokens and audio segments
        if len(batch_all_audios) == 0: continue
        batch_all_audio = torch.cat(batch_all_audios, dim=-1)  # b 1 t
        batch_all_tokens_th = torch.cat(batch_all_tokens, dim=-1)  # b 9 s
        
        # Add synchronization watermark if needed
        if args.wm_sync:
            batch_all_audio = sync_pattern.get_sync_wm(batch_all_audio, alpha=0.5)

        # Prepare validation augmentations, similar to before
        augs = get_validation_augs() if args.eval_aug else get_dummy_augs()
        for aug, _ in augs:
            aug.to(args.device)
        batch_audio_saved = batch_all_audio.clone()
        
        for validation_aug, strengths in augs:
            for strength in strengths:
                # Apply augmentation
                batch_aug_audio, _ = validation_aug(batch_audio_saved, None, strength)

                # Use synchronization watermark if needed
                if args.wm_sync:
                    detection_results = sync_pattern.detect_sync_wm(batch_aug_audio) # b s

                for idx in range(batch_size):
                    synced_audio = batch_aug_audio[idx:idx+1]
                    if args.wm_sync:
                        detection_score =  detection_results[idx].mean()
                        threshold = 0.25
                        if np.abs(detection_score - 0.5) < threshold:
                            speedup, shift = sync_pattern.get_speedup_and_shift(detection_results[idx])
                            synced_audio = sync_pattern.invert(synced_audio, speedup, shift)
                            print(f"Sync watermark detected - Score: {detection_score} - Speedup: {speedup}, Shift: {shift}")

                    # Encode augmented audio
                    tokens_roundtrip = mimi.encode(synced_audio)  # 1 1 s -> 1 8 s

                    # Get watermarked streams
                    wm_tokens = batch_all_tokens_th[0, args.wm_streams, :]  # w s
                    new_wm_streams = [stream - 1 for stream in args.wm_streams if stream > 0]
                    wm_tokens_roundtrip = tokens_roundtrip[0, new_wm_streams, :] if new_wm_streams else None
                    
                    # Calculate watermark stats
                    # ngrams = torch.zeros((1, 0), device='cpu')
                    orig_greens, orig_scored = [], []

                    # Precompute the index of the "first audio channel" (stream id == 1) if present in args.wm_streams
                    audio_stream_index = None
                    if 1 in args.wm_streams:
                        audio_stream_index = args.wm_streams.index(1)

                    # audio_stream_index = 0

                    for stream_idx in range(wm_tokens.shape[0]):
                        wm_stream = wm_tokens[stream_idx, :]
                        stream_id = args.wm_streams[stream_idx]
                        s_map = clustering_maps.get(stream_id) if clustering_maps else None

                        # Choose source stream for ngram construction:
                        # - text stream (stream id 0): use the text stream tokens themselves
                        # - depformer streams (stream id > 0): use first audio channel stream (stream id == 1) when available
                        if int(args.wm_ngram) <= 0:
                            ngrams_for_stream = torch.zeros((1, 0), device='cpu')
                        else:
                            if stream_id == 0:
                                # build per-timestep ngrams from the text stream itself
                                ngrams_for_stream = build_stream_ngrams_from_full_stream(wm_stream, args.wm_ngram, device='cpu')
                            else:
                                # use first audio channel as source; fallback to empty if not present
                                if audio_stream_index is not None:
                                    audio_stream = wm_tokens[audio_stream_index, :]
                                    ngrams_for_stream = build_stream_ngrams_from_full_stream(audio_stream, args.wm_ngram, device='cpu')
                                else:
                                    # no first audio channel available in the selected streams; fallback to empty ngram
                                    ngrams_for_stream = torch.zeros((1, 0), device='cpu')

                        green_mask, to_score_mask = compute_watermark_scores(
                            wm_stream,
                            ngrams_for_stream,
                            lm_gen.lm_model.card,
                            args.wm_gamma,
                            args.wm_seed,
                            clustering_map=s_map,
                        )
                        orig_greens.append((green_mask * to_score_mask).float().sum().item())
                        orig_scored.append(to_score_mask.float().sum().item())
                    
                    greens, scored = [], []
                    if wm_tokens_roundtrip is not None:
                        # find index of first audio channel (stream id == 1) among selected streams, if present
                        audio_stream_index = None
                        if 1 in args.wm_streams:
                            audio_stream_index = args.wm_streams.index(1)

                        # audio_stream_index = 0

                        for stream_idx in range(wm_tokens_roundtrip.shape[0]):
                            wm_stream = wm_tokens_roundtrip[stream_idx, :]
                            stream_id = args.wm_streams[stream_idx]
                            s_map = clustering_maps.get(stream_id) if clustering_maps else None

                            # build the ngram matrix for this stream:
                            if int(args.wm_ngram) <= 0:
                                ngrams_for_stream = torch.zeros((1, 0), device='cpu')
                            else:
                                if stream_id == 0:
                                    # text stream: use the text stream itself
                                    ngrams_for_stream = build_stream_ngrams_from_full_stream(wm_stream, args.wm_ngram, device='cpu')
                                else:
                                    # depformer stream: use the first audio channel's tokens as source if available
                                    if audio_stream_index is not None:
                                        audio_stream = wm_tokens_roundtrip[audio_stream_index, :]
                                        ngrams_for_stream = build_stream_ngrams_from_full_stream(audio_stream, args.wm_ngram, device='cpu')
                                    else:
                                        # fallback to empty ngram if first audio channel not present
                                        ngrams_for_stream = torch.zeros((1, 0), device='cpu')

                            green_mask, to_score_mask = compute_watermark_scores(
                                wm_stream,
                                ngrams_for_stream,
                                lm_gen.lm_model.card,
                                args.wm_gamma,
                                args.wm_seed,
                                clustering_map=s_map,
                            )
                            greens.append((green_mask * to_score_mask).float().sum().item())
                            scored.append(to_score_mask.float().sum().item())
                    
                    tot_orig_greens = float(sum(orig_greens))
                    tot_orig_scored = float(sum(orig_scored))
                    orig_pval = get_binomial_pval(tot_orig_greens, tot_orig_scored, args.wm_gamma)
                    if wm_tokens_roundtrip is not None:
                        tot_greens = sum(greens)
                        tot_scored = sum(scored)
                        pval = get_binomial_pval(tot_greens, tot_scored, args.wm_gamma)
                    else:
                        pval = None
                    
                    print(orig_pval, pval)
                    
                    global_idx = batch_start + idx
                    result = {
                        "config": config_name,
                        "idx": global_idx,
                        "aug_name": str(validation_aug),
                        "strength": strength,
                        "original_greens": orig_greens,
                        "original_ntoks": wm_tokens.shape[-1],
                        "original_pval": orig_pval,
                        "greens": greens,
                        "scored": scored,
                        "ntoks": wm_tokens_roundtrip.shape[-1] if wm_tokens_roundtrip is not None else 0,
                        "pval": pval,
                    }
                    # print(orig_pval, pval)
                    global_watermark_results.append(result)
                    
                    # Save generated audio if within limit
                    if args.save_audio > 0 and global_idx < args.save_audio:
                        audio_output_dir = os.path.join(args.output_dir, f"audio_{config_name}")
                        os.makedirs(audio_output_dir, exist_ok=True)
                        aug_audio = batch_aug_audio[idx, 0].detach().cpu().numpy().astype(np.float32)
                        sphn.write_wav(
                            os.path.join(audio_output_dir, f'{validation_aug}_{strength}_{global_idx:03d}.wav'),
                            aug_audio,
                            mimi.sample_rate,
                        )
                        if args.wm_sync:
                            synced_audio = synced_audio[0, 0].detach().cpu().numpy().astype(np.float32)
                            sphn.write_wav(
                                os.path.join(audio_output_dir, f'{validation_aug}_{strength}_{global_idx:03d}_synced.wav'),
                                synced_audio,
                                mimi.sample_rate,
                            )

        with open(os.path.join(args.output_dir, f"generated_texts_{config_name}.txt"), "a", encoding="utf-8") as f:
            for idx in range(batch_size):
                f.write(f"{idx + batch_start:04d},{''.join(batch_all_texts[idx])}\n")

    # Save only summary results, remove tokens.pt save
    summary = {
        'config': vars(args),
        'results': global_watermark_results
    }
    torch.save(summary, os.path.join(args.output_dir, f'summary_{config_name}.pt'))

    # Calculate statistics and print pandas DataFrame
    # Keep generated_text for the full df, but exclude from mean calculation
    df_data = [
        {
            "idx": wmr["idx"],
            "aug_name": wmr["aug_name"],
            "strength": str(wmr["strength"]),
            "greens": sum(wmr["greens"]),
            "scored": sum(wmr["scored"]),
            "ntoks": wmr["ntoks"],
            "pval": wmr["pval"],
            "logpval": -np.log10(wmr["pval"]) if wmr["pval"] is not None and wmr["pval"] > 0 else None, # ensure pval > 0 for log
        }
        for wmr in global_watermark_results
    ]

    df = pd.DataFrame(df_data)
    # Select columns for mean aggregation, excluding 'generated_text' and 'idx'
    numeric_cols_for_mean = ["greens", "scored", "ntoks", "pval", "logpval"]
    # Filter df for numeric_cols_for_mean, handling cases where some might be missing
    cols_to_aggregate = [col for col in numeric_cols_for_mean if col in df.columns]
    
    mean_df = df.groupby(["aug_name", "strength"])[cols_to_aggregate].agg("mean")
    mean_df.to_csv(os.path.join(args.output_dir, f'summary_{config_name}.csv'))
        
    pd.set_option('display.max_rows', None)
    print(mean_df)
    
    # Optionally, save the full DataFrame with texts to a CSV
    df.to_csv(os.path.join(args.output_dir, f'results_{config_name}.csv'), index=False)

def main():
    parser = argparse.ArgumentParser()
    # Basic configuration
    parser.add_argument("--output_dir", type=str, required=True)
    parser.add_argument("--device", type=str, default="cuda" if torch.cuda.device_count() else "cpu")
    parser.add_argument("--seed", type=int, default=42424242)
    parser.add_argument("--hf_repo", type=str, default="kyutai/moshiko-pytorch-bf16")
    parser.add_argument("--steps", type=int, default=200)
    parser.add_argument("--temperature", type=float, default=1.0)
    
    # Prompt configuration
    parser.add_argument("--use_prompts", type=bool_inst, default=True)
    parser.add_argument("--audio_dir", type=str, help="Directory containing audio files for prompts")
    parser.add_argument("--duration_sec", type=float, default=None,
                    help="Maximum duration in seconds for each audio file. None means full length")
    parser.add_argument("--nsamples", type=int, default=-1, help="Number of audio files to process. -1 means all files")
    parser.add_argument("--batch_size", type=int, default=1, help="Batch size for watermark evaluation")
    
    # Watermarking configuration
    parser.add_argument("--wm_method", type=str, default="maryland", 
                    help="Watermarking method to use")
    parser.add_argument("--wm_streams", nargs='+', default=[],
                    help="List of stream indices to apply watermarking")
    parser.add_argument("--wm_delta", type=float, default=8.0,
                    help="Watermark delta parameter")
    parser.add_argument("--wm_gamma", type=float, default=0.25,
                    help="Watermark gamma parameter")
    parser.add_argument("--wm_ngram", type=int, default=0,
                    help="Watermark n-gram parameter")
    parser.add_argument("--wm_seed", type=int, default=0)
    parser.add_argument("--wm_sync", type=bool_inst, default=False,
                    help="Whether to use sync watermarking. Very WIP.")

    # Clustering
    parser.add_argument("--wm_clustering", type=bool_inst, default=False,
                    help="Whether to use vocabulary clustering. Currently, this runs all configurations.")
    
    # Model weights
    parser.add_argument("--tokenizer", type=str)
    parser.add_argument("--moshi_weight", type=str)
    parser.add_argument("--mimi_weight", type=str)
    parser.add_argument("--mimi_weight_ori", type=str)
    
    # Analysis and saving options - update save_audio and add save_tokens
    parser.add_argument("--save_audio", type=int, default=10,
                    help="Number of audio files to save (0 = none)")
    parser.add_argument("--save_tokens", type=int, default=0,
                    help="Number of token files to save (0 = none)")
    
    # Add evaluation under augmentations argument:
    parser.add_argument("--eval_aug", type=bool_inst, default=True,
                        help="Evaluate watermark under validation augmentations")
                        
    args = parser.parse_args()

    if args.mimi_weight is None or args.mimi_weight.lower() == "none":
        args.mimi_weight = None
    
    # Download weights if not provided
    if args.mimi_weight is None:
        args.mimi_weight = hf_hub_download(args.hf_repo, loaders.MIMI_NAME)
    if args.mimi_weight_ori is None:
        args.mimi_weight_ori = hf_hub_download(args.hf_repo, loaders.MIMI_NAME)
    if args.moshi_weight is None:
        args.moshi_weight = hf_hub_download(args.hf_repo, loaders.MOSHI_NAME)
    if args.tokenizer is None:
        args.tokenizer = hf_hub_download(args.hf_repo, loaders.TEXT_TOKENIZER_NAME)
    
    # Create output dir
    os.makedirs(args.output_dir, exist_ok=True)
    
    # Set random seed
    seed_all(args.seed)

    # Determine what to run
    configs_to_run = [{"method": None, "maps": None}] # Default standard
    all_maps = None

    if args.wm_clustering:
        clustering_dir = "models/moshi/embeddings/clusterings"
        
        target_min_count = 1
        target_min_count = None
        all_maps = load_clustering_maps(clustering_dir, target_min_count=target_min_count, device=args.device)
        
        # Build configs list
        if all_maps and 0 in all_maps: # Check channel 0 for available methods
            found_methods = list(all_maps[0].keys())
            
            for m in found_methods:
                # Find all available keys for this method in channel 0
                keys_available = sorted(list(all_maps[0][m].keys()))
                
                for key in keys_available:
                    # Construct map for this specific configuration
                    current_config_maps = {}
                    valid_config = True
                    for s in args.wm_streams:
                        c = int(s) - 1
                        if c in all_maps and m in all_maps[c] and key in all_maps[c][m]:
                            _, tmap = all_maps[c][m][key]
                            current_config_maps[int(s)] = tmap
                        else:
                            valid_config = False # Missing map for a required stream
                            break
                    
                    if valid_config:
                        configs_to_run.append({
                            "method": m,
                            "min_count": key,
                            "maps": current_config_maps
                        })

    print(f"Starting execution for {len(configs_to_run)} configurations")

    # Run watermark evaluation for each method
    for config in configs_to_run:
        if config["method"] is None:
            config_name = "standard"
        else:
            config_name = f"{config['method']}_{config['min_count']}"
        
        run_watermark_eval(args, clustering_maps=config["maps"], config_name=config_name)

if __name__ == "__main__":
    main()
