use std::path::Path;
use serde::{Deserialize, Serialize};
use crate::error::{Error, Result};
fn default_oneharness_bin() -> String {
"oneharness".to_string()
}
fn default_judge_harness() -> String {
"claude-code".to_string()
}
fn default_timeout_secs() -> u64 {
120
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct OneharnessConfig {
#[serde(default = "default_oneharness_bin")]
pub bin: String,
#[serde(default = "default_judge_harness")]
pub judge_harness: String,
#[serde(default = "default_timeout_secs")]
pub timeout_secs: u64,
}
impl Default for OneharnessConfig {
fn default() -> Self {
Self {
bin: default_oneharness_bin(),
judge_harness: default_judge_harness(),
timeout_secs: default_timeout_secs(),
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct CommandConfig {
pub command: Vec<String>,
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
#[serde(tag = "kind", rename_all = "lowercase")]
pub enum ProviderConfig {
Oneharness(OneharnessConfig),
Command(CommandConfig),
}
impl Default for ProviderConfig {
fn default() -> Self {
ProviderConfig::Oneharness(OneharnessConfig::default())
}
}
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
#[serde(default, deny_unknown_fields)]
pub struct Config {
pub provider: ProviderConfig,
pub platforms: Vec<String>,
pub models: Vec<String>,
pub judge_model: String,
pub max_turns: u32,
}
impl Default for Config {
fn default() -> Self {
Self {
provider: ProviderConfig::default(),
platforms: vec!["claude-code".to_string()],
models: vec!["claude-opus-4-8".to_string()],
judge_model: String::new(),
max_turns: 8,
}
}
}
#[derive(Debug, Clone, Default)]
pub struct Overrides {
pub command_provider: Option<Vec<String>>,
pub oneharness_bin: Option<String>,
pub judge_harness: Option<String>,
pub timeout_secs: Option<u64>,
pub platforms: Vec<String>,
pub models: Vec<String>,
pub judge_model: Option<String>,
pub max_turns: Option<u32>,
}
impl Config {
pub fn load(path: &Path) -> Result<Self> {
let text = std::fs::read_to_string(path).map_err(|source| Error::Io {
path: path.to_path_buf(),
source,
})?;
let config: Config = serde_yaml::from_str(&text).map_err(|source| Error::Yaml {
path: path.to_path_buf(),
source,
})?;
config.validate()?;
Ok(config)
}
pub fn load_or_default(path: &Path) -> Result<Self> {
if path.is_file() {
Self::load(path)
} else {
Ok(Self::default())
}
}
pub fn apply_overrides(&mut self, overrides: Overrides) -> Result<()> {
if let Some(command) = overrides.command_provider {
self.provider = ProviderConfig::Command(CommandConfig { command });
} else if let ProviderConfig::Oneharness(oh) = &mut self.provider {
if let Some(bin) = overrides.oneharness_bin {
oh.bin = bin;
}
if let Some(judge_harness) = overrides.judge_harness {
oh.judge_harness = judge_harness;
}
if let Some(timeout) = overrides.timeout_secs {
oh.timeout_secs = timeout;
}
}
if !overrides.platforms.is_empty() {
self.platforms = overrides.platforms;
}
if !overrides.models.is_empty() {
self.models = overrides.models;
}
if let Some(judge) = overrides.judge_model {
self.judge_model = judge;
}
if let Some(max_turns) = overrides.max_turns {
self.max_turns = max_turns;
}
self.validate()
}
#[must_use]
pub fn effective_judge_model(&self) -> &str {
if self.judge_model.is_empty() {
self.models.first().map_or("", String::as_str)
} else {
&self.judge_model
}
}
pub fn validate(&self) -> Result<()> {
match &self.provider {
ProviderConfig::Oneharness(oh) => {
if oh.bin.trim().is_empty() {
return Err(Error::Invalid(
"config `provider.bin` must name the oneharness binary".into(),
));
}
if oh.judge_harness.trim().is_empty() {
return Err(Error::Invalid(
"config `provider.judge_harness` must name a harness".into(),
));
}
if oh.timeout_secs == 0 {
return Err(Error::Invalid(
"config `provider.timeout_secs` must be at least 1".into(),
));
}
}
ProviderConfig::Command(c) => {
if c.command.is_empty() {
return Err(Error::Invalid(
"config `provider.command` must name a command".into(),
));
}
}
}
if self.platforms.is_empty() {
return Err(Error::Invalid(
"config `platforms` must list at least one harness platform".into(),
));
}
if self.models.is_empty() {
return Err(Error::Invalid(
"config `models` must list at least one model".into(),
));
}
if self.max_turns == 0 {
return Err(Error::Invalid(
"config `max_turns` must be at least 1".into(),
));
}
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn defaults_are_valid_and_use_oneharness() {
let config = Config::default();
config.validate().unwrap();
assert!(matches!(config.provider, ProviderConfig::Oneharness(_)));
}
#[test]
fn command_override_switches_provider() {
let mut config = Config::default();
config
.apply_overrides(Overrides {
command_provider: Some(vec!["fake".into()]),
..Default::default()
})
.unwrap();
assert_eq!(
config.provider,
ProviderConfig::Command(CommandConfig {
command: vec!["fake".into()]
})
);
}
#[test]
fn oneharness_bin_override_applies() {
let mut config = Config::default();
config
.apply_overrides(Overrides {
oneharness_bin: Some("/tmp/oneharness".into()),
..Default::default()
})
.unwrap();
let ProviderConfig::Oneharness(oh) = &config.provider else {
panic!("expected oneharness provider");
};
assert_eq!(oh.bin, "/tmp/oneharness");
}
#[test]
fn parses_command_provider_yaml() {
let yaml = "provider:\n kind: command\n command: [\"prov\", \"--flag\"]\n";
let config: Config = serde_yaml::from_str(yaml).unwrap();
assert_eq!(
config.provider,
ProviderConfig::Command(CommandConfig {
command: vec!["prov".into(), "--flag".into()]
})
);
}
#[test]
fn parses_oneharness_provider_yaml() {
let yaml = "provider:\n kind: oneharness\n bin: oh\n judge_harness: codex\n";
let config: Config = serde_yaml::from_str(yaml).unwrap();
let ProviderConfig::Oneharness(oh) = &config.provider else {
panic!("expected oneharness provider");
};
assert_eq!(oh.bin, "oh");
assert_eq!(oh.judge_harness, "codex");
assert_eq!(oh.timeout_secs, 120);
}
#[test]
fn judge_model_falls_back_to_first_model() {
let config = Config::default();
assert_eq!(config.effective_judge_model(), "claude-opus-4-8");
}
#[test]
fn empty_models_is_invalid() {
let mut config = Config::default();
config.models.clear();
assert!(config.validate().is_err());
}
}