entrenar 0.7.10

Training & Optimization library with autograd, LoRA, quantization, and model merging
Documentation
//! Model Configuration
//!
//! Contains model-related configuration types for training manifests.

use serde::{Deserialize, Serialize};
use std::collections::HashMap;

use super::shorthand::deserialize_human_usize_opt;

/// Model configuration
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ModelConfig {
    /// Model source URI (pacha://, hf://, or local path)
    pub source: String,

    /// Model format (safetensors, gguf, apr, pt)
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub format: Option<String>,

    /// Architecture override
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub architecture: Option<ArchitectureConfig>,

    /// Layers to freeze
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub freeze: Option<Vec<String>>,

    /// Device placement (auto, cpu, cuda, cuda:0, mps)
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub device: Option<String>,

    /// Data type (float32, float16, bfloat16)
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub dtype: Option<String>,
}

/// Model architecture configuration
///
/// Supports both preset names and custom architecture parameters.
/// Custom params override values from config.json or preset defaults.
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ArchitectureConfig {
    /// Architecture type (transformer, sequential)
    #[serde(rename = "type")]
    pub arch_type: String,

    /// Hidden size (embedding dimension). Accepts shorthand: `"1K"` = 1024.
    #[serde(
        default,
        skip_serializing_if = "Option::is_none",
        deserialize_with = "deserialize_human_usize_opt"
    )]
    pub hidden_size: Option<usize>,

    /// Number of transformer layers
    #[serde(
        default,
        skip_serializing_if = "Option::is_none",
        alias = "num_hidden_layers",
        deserialize_with = "deserialize_human_usize_opt"
    )]
    pub num_layers: Option<usize>,

    /// Number of attention heads
    #[serde(
        default,
        skip_serializing_if = "Option::is_none",
        alias = "num_attention_heads",
        deserialize_with = "deserialize_human_usize_opt"
    )]
    pub num_heads: Option<usize>,

    /// Number of key-value heads (for grouped-query attention)
    #[serde(
        default,
        skip_serializing_if = "Option::is_none",
        alias = "num_key_value_heads",
        deserialize_with = "deserialize_human_usize_opt"
    )]
    pub num_kv_heads: Option<usize>,

    /// FFN intermediate dimension. Accepts shorthand: `"4K"` = 4096.
    #[serde(
        default,
        skip_serializing_if = "Option::is_none",
        deserialize_with = "deserialize_human_usize_opt"
    )]
    pub intermediate_size: Option<usize>,

    /// Vocabulary size. Accepts shorthand: `"32K"` = 32768.
    #[serde(
        default,
        skip_serializing_if = "Option::is_none",
        deserialize_with = "deserialize_human_usize_opt"
    )]
    pub vocab_size: Option<usize>,

    /// Maximum sequence/position length. Accepts shorthand: `"2K"` = 2048, `"128K"` = 131072.
    #[serde(
        default,
        skip_serializing_if = "Option::is_none",
        alias = "max_position_embeddings",
        deserialize_with = "deserialize_human_usize_opt"
    )]
    pub max_seq_length: Option<usize>,

    /// RMS normalization epsilon
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub rms_norm_eps: Option<f32>,

    /// RoPE theta (rotary positional encoding base)
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub rope_theta: Option<f32>,

    /// Whether to use bias in linear layers
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub use_bias: Option<bool>,

    /// Per-head dimension override (for models where head_dim != hidden_size / num_heads)
    #[serde(
        default,
        skip_serializing_if = "Option::is_none",
        deserialize_with = "deserialize_human_usize_opt"
    )]
    pub head_dim: Option<usize>,

    /// Sequential layers (for sequential architecture type)
    #[serde(default, skip_serializing_if = "Option::is_none")]
    pub layers: Option<Vec<HashMap<String, serde_json::Value>>>,
}