use crate::definition::{MobDefinition, SkillSource};
use crate::error::MobError;
use crate::ids::{MeerkatId, MobId, ProfileName};
use crate::profile::Profile;
use meerkat::AgentBuildConfig;
use meerkat_core::PeerMeta;
use meerkat_core::Session;
use meerkat_core::service::{CreateSessionRequest, DeferredPromptPolicy, MobToolAuthorityContext};
use meerkat_core::session::SessionMetadata;
use meerkat_core::types::SessionId;
use std::sync::Arc;
#[derive(Clone, Debug, Default, PartialEq, Eq)]
pub enum MobToolAccessContext {
#[default]
None,
InjectedAuthority(MobToolAuthorityContext),
}
impl MobToolAccessContext {
pub fn authority(&self) -> Option<MobToolAuthorityContext> {
match self {
Self::None => None,
Self::InjectedAuthority(authority) => Some(authority.clone()),
}
}
}
pub struct BuildAgentConfigParams<'a> {
pub mob_id: &'a MobId,
pub profile_name: &'a ProfileName,
pub meerkat_id: &'a MeerkatId,
pub profile: &'a Profile,
pub definition: &'a MobDefinition,
pub external_tools: Option<Arc<dyn meerkat_core::AgentToolDispatcher>>,
pub context: Option<serde_json::Value>,
pub labels: Option<std::collections::BTreeMap<String, String>>,
pub additional_instructions: Option<Vec<String>>,
pub shell_env: Option<std::collections::HashMap<String, String>>,
pub mob_tool_access_context: MobToolAccessContext,
}
pub struct BuildResumedAgentConfigParams<'a> {
pub base: BuildAgentConfigParams<'a>,
pub expected_session_id: &'a SessionId,
pub resumed_session: Session,
}
pub async fn build_agent_config(
params: BuildAgentConfigParams<'_>,
) -> Result<AgentBuildConfig, MobError> {
let BuildAgentConfigParams {
mob_id,
profile_name,
meerkat_id,
profile,
definition,
external_tools,
context,
labels,
additional_instructions,
shell_env,
mob_tool_access_context,
} = params;
if !profile.tools.comms {
return Err(MobError::WiringError(format!(
"profile '{profile_name}' has tools.comms=false; mob meerkats require comms=true"
)));
}
let comms_name = format!("{mob_id}/{profile_name}/{meerkat_id}");
let mut peer_meta = PeerMeta::default().with_description(&profile.peer_description);
if let Some(app_labels) = labels {
for (k, v) in app_labels {
peer_meta = peer_meta.with_label(&k, &v);
}
}
peer_meta = peer_meta
.with_label("mob_id", mob_id.as_str())
.with_label("role", profile_name.as_str())
.with_label("meerkat_id", meerkat_id.as_str());
let realm_id = format!("mob:{mob_id}");
let system_prompt = assemble_system_prompt(profile, definition).await?;
let mut config = AgentBuildConfig::new(profile.model.clone());
config.keep_alive = false;
config.comms_name = Some(comms_name);
config.peer_meta = Some(peer_meta);
config.realm_id = Some(realm_id);
if !system_prompt.is_empty() {
config.system_prompt = Some(system_prompt);
}
config.preload_skills = Some(vec![meerkat_core::skills::SkillId::from(
"mob-communication",
)]);
config.silent_comms_intents = vec!["mob.peer_added".into(), "mob.peer_retired".into()];
config.max_inline_peer_notifications = profile.max_inline_peer_notifications;
config.override_builtins =
meerkat_core::ToolCategoryOverride::from_effective(profile.tools.builtins);
config.override_shell = meerkat_core::ToolCategoryOverride::from_effective(profile.tools.shell);
config.override_memory =
meerkat_core::ToolCategoryOverride::from_effective(profile.tools.memory);
config.override_mob = meerkat_core::ToolCategoryOverride::from_effective(
mob_tool_access_context.authority().is_some(),
);
config.mob_tool_authority_context = mob_tool_access_context.authority();
config.external_tools = external_tools;
config.app_context = context;
config.additional_instructions = additional_instructions;
config.shell_env = shell_env;
config.provider_params = profile.provider_params.clone();
if let Some(schema_value) = &profile.output_schema {
let schema = meerkat_core::MeerkatSchema::new(schema_value.clone()).map_err(|e| {
MobError::WiringError(format!(
"invalid output_schema for profile '{profile_name}': {e}"
))
})?;
config.output_schema = Some(meerkat_core::OutputSchema {
schema,
name: Some(format!("{mob_id}_{profile_name}")),
strict: true,
compat: Default::default(),
format: Default::default(),
});
}
Ok(config)
}
pub async fn build_resumed_agent_config(
params: BuildResumedAgentConfigParams<'_>,
) -> Result<AgentBuildConfig, MobError> {
let BuildResumedAgentConfigParams {
base,
expected_session_id,
resumed_session,
} = params;
if resumed_session.id() != expected_session_id {
return Err(MobError::Internal(format!(
"resume session id mismatch: expected '{}', got '{}'",
expected_session_id,
resumed_session.id()
)));
}
let mut config = build_agent_config(base).await?;
let metadata = resumed_session
.session_metadata()
.ok_or_else(|| MobError::Internal("missing durable session metadata".to_string()))?;
apply_resumed_session_metadata(&mut config, &metadata)?;
config.resume_session = Some(resumed_session);
config.system_prompt = None;
config.additional_instructions = None;
config.app_context = None;
config.shell_env = None;
Ok(config)
}
fn apply_resumed_session_metadata(
config: &mut AgentBuildConfig,
metadata: &SessionMetadata,
) -> Result<(), MobError> {
let current_comms_name = config.comms_name.clone();
let Some(stored_comms_name) = metadata.comms_name.clone() else {
return Err(MobError::Internal(
"missing durable comms_name for resumed mob member".to_string(),
));
};
if current_comms_name.as_deref() != Some(stored_comms_name.as_str()) {
return Err(MobError::Internal(format!(
"persisted comms_name '{}' does not match current mob identity '{}'",
stored_comms_name,
current_comms_name.unwrap_or_else(|| "<none>".to_string())
)));
}
config.model = metadata.model.clone();
config.max_tokens = Some(metadata.max_tokens);
config.provider = Some(metadata.provider);
config.provider_params = metadata.provider_params.clone();
config.override_builtins = metadata.tooling.builtins;
config.override_shell = metadata.tooling.shell;
config.override_memory = metadata.tooling.memory;
if matches!(
config.override_mob,
meerkat_core::ToolCategoryOverride::Inherit
) {
config.override_mob = metadata.tooling.mob;
}
config.preload_skills = metadata.tooling.active_skills.clone();
config.comms_name = Some(stored_comms_name);
config.peer_meta = metadata.peer_meta.clone();
Ok(())
}
pub fn to_create_session_request(
config: &AgentBuildConfig,
prompt: meerkat_core::types::ContentInput,
) -> CreateSessionRequest {
let build_options = config.to_session_build_options();
CreateSessionRequest {
model: config.model.clone(),
prompt,
render_metadata: None,
system_prompt: config.system_prompt.clone(),
max_tokens: config.max_tokens,
event_tx: None,
skill_references: None,
initial_turn: meerkat_core::service::InitialTurnPolicy::Defer,
deferred_prompt_policy: DeferredPromptPolicy::Discard,
build: Some(build_options),
labels: None,
}
}
async fn assemble_system_prompt(
profile: &Profile,
definition: &MobDefinition,
) -> Result<String, MobError> {
let mut sections = Vec::new();
for skill_ref in &profile.skills {
if let Some(source) = definition.skills.get(skill_ref) {
match source {
SkillSource::Inline { content } => {
sections.push(content.clone());
}
SkillSource::Path { path } => {
#[cfg(not(target_arch = "wasm32"))]
{
let content = tokio::fs::read_to_string(path).await.map_err(|error| {
MobError::Internal(format!(
"failed to read skill file '{path}' while building system prompt: {error}"
))
})?;
sections.push(content);
}
#[cfg(target_arch = "wasm32")]
return Err(MobError::Internal(format!(
"file-based skill path '{path}' is not supported on wasm32"
)));
}
}
}
}
Ok(sections.join("\n\n"))
}
#[cfg(test)]
#[allow(clippy::unwrap_used, clippy::expect_used, clippy::panic)]
mod tests {
use super::*;
use crate::definition::{BackendConfig, MobDefinition, OrchestratorConfig, WiringRules};
use crate::profile::ToolConfig;
use std::collections::BTreeMap;
use std::fs;
fn sample_definition() -> MobDefinition {
let mut profiles = BTreeMap::new();
profiles.insert(
ProfileName::from("lead"),
Profile {
model: "claude-opus-4-6".into(),
skills: vec!["leader-skill".into()],
tools: ToolConfig {
builtins: true,
shell: true,
comms: true,
memory: false,
mob: true,
mob_tasks: true,
mcp: vec![],
rust_bundles: vec![],
},
peer_description: "Orchestrates the mob".into(),
external_addressable: true,
backend: None,
runtime_mode: crate::MobRuntimeMode::AutonomousHost,
max_inline_peer_notifications: None,
output_schema: None,
provider_params: None,
},
);
profiles.insert(
ProfileName::from("worker"),
Profile {
model: "claude-sonnet-4-5".into(),
skills: vec![],
tools: ToolConfig {
builtins: true,
shell: false,
comms: true,
memory: false,
mob: false,
mob_tasks: false,
mcp: vec![],
rust_bundles: vec![],
},
peer_description: "Does work".into(),
external_addressable: false,
backend: None,
runtime_mode: crate::MobRuntimeMode::AutonomousHost,
max_inline_peer_notifications: None,
output_schema: None,
provider_params: None,
},
);
let mut skills = BTreeMap::new();
skills.insert(
"leader-skill".into(),
SkillSource::Inline {
content: "You are the team lead.".into(),
},
);
MobDefinition {
id: MobId::from("test-mob"),
orchestrator: Some(OrchestratorConfig {
profile: ProfileName::from("lead"),
}),
profiles,
mcp_servers: BTreeMap::new(),
wiring: WiringRules::default(),
skills,
backend: BackendConfig::default(),
flows: BTreeMap::new(),
topology: None,
supervisor: None,
limits: None,
spawn_policy: None,
event_router: None,
owner_session_id: None,
session_cleanup_policy: crate::definition::SessionCleanupPolicy::Manual,
is_implicit: false,
}
}
fn injected_authority() -> MobToolAccessContext {
MobToolAccessContext::InjectedAuthority(
meerkat_core::service::MobToolAuthorityContext::new(
meerkat_core::service::OpaquePrincipalToken::new("test-principal"),
true,
)
.with_managed_mob_scope(["test-mob"]),
)
}
#[tokio::test]
async fn test_build_agent_config_non_keep_alive() {
let def = sample_definition();
let profile = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert!(!config.keep_alive, "keep_alive must be false for mob spawn");
}
#[tokio::test]
async fn test_build_agent_config_comms_name() {
let def = sample_definition();
let profile = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(
config.comms_name.as_deref(),
Some("test-mob/lead/lead-1"),
"comms_name should be mob_id/profile/meerkat_id"
);
}
#[tokio::test]
async fn test_build_agent_config_peer_meta_labels() {
let def = sample_definition();
let profile = &def.profiles[&ProfileName::from("worker")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("worker"),
meerkat_id: &MeerkatId::from("w-1"),
profile,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
let meta = config.peer_meta.as_ref().expect("peer_meta should be set");
assert_eq!(
meta.labels.get("mob_id").map(String::as_str),
Some("test-mob")
);
assert_eq!(meta.labels.get("role").map(String::as_str), Some("worker"));
assert_eq!(
meta.labels.get("meerkat_id").map(String::as_str),
Some("w-1")
);
assert_eq!(meta.description.as_deref(), Some("Does work"));
}
#[tokio::test]
async fn test_build_agent_config_realm_id() {
let def = sample_definition();
let profile = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(
config.realm_id.as_deref(),
Some("mob:test-mob"),
"realm_id should be 'mob:test-mob'"
);
}
#[tokio::test]
async fn test_build_agent_config_tool_overrides() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(
config.override_builtins,
meerkat_core::ToolCategoryOverride::Enable
);
assert_eq!(
config.override_shell,
meerkat_core::ToolCategoryOverride::Enable
);
assert_eq!(
config.override_memory,
meerkat_core::ToolCategoryOverride::Disable
);
assert_eq!(
config.override_mob,
meerkat_core::ToolCategoryOverride::Disable
);
let worker = &def.profiles[&ProfileName::from("worker")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("worker"),
meerkat_id: &MeerkatId::from("w-1"),
profile: worker,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(
config.override_builtins,
meerkat_core::ToolCategoryOverride::Enable
);
assert_eq!(
config.override_shell,
meerkat_core::ToolCategoryOverride::Disable
);
assert_eq!(
config.override_memory,
meerkat_core::ToolCategoryOverride::Disable
);
assert_eq!(
config.override_mob,
meerkat_core::ToolCategoryOverride::Disable
);
}
#[tokio::test]
async fn test_build_agent_config_operator_context_enables_mob_override() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-operator"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: injected_authority(),
})
.await
.expect("build_agent_config");
assert_eq!(
config.override_mob,
meerkat_core::ToolCategoryOverride::Enable
);
assert!(
config.mob_tool_authority_context.is_some(),
"typed injected authority should flow into the build config"
);
}
#[tokio::test]
async fn test_build_resumed_agent_config_does_not_restore_mob_override_without_operator_context()
{
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let session_id = SessionId::new();
let mut resumed_session = Session::with_id(session_id.clone());
resumed_session
.set_session_metadata(SessionMetadata {
model: "claude-opus-4-6".to_string(),
max_tokens: 2048,
structured_output_retries: 2,
provider: meerkat_core::Provider::Anthropic,
provider_params: None,
tooling: meerkat_core::session::SessionTooling {
builtins: meerkat_core::session::ToolCategoryOverride::Enable,
shell: meerkat_core::session::ToolCategoryOverride::Enable,
comms: meerkat_core::session::ToolCategoryOverride::Enable,
mob: meerkat_core::session::ToolCategoryOverride::Enable,
memory: meerkat_core::session::ToolCategoryOverride::Disable,
active_skills: None,
},
keep_alive: false,
comms_name: Some("test-mob/lead/lead-1".to_string()),
peer_meta: None,
realm_id: None,
instance_id: None,
backend: None,
config_generation: None,
})
.expect("session metadata");
let config = build_resumed_agent_config(BuildResumedAgentConfigParams {
base: BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
},
expected_session_id: &session_id,
resumed_session,
})
.await
.expect("build_resumed_agent_config");
assert_eq!(
config.override_mob,
meerkat_core::ToolCategoryOverride::Disable,
"runtime-injected absence of operator capabilities must beat resumed mob metadata"
);
}
#[tokio::test]
async fn test_build_agent_config_fails_when_comms_disabled() {
let mut def = sample_definition();
let worker = def
.profiles
.get_mut(&ProfileName::from("worker"))
.expect("worker profile");
worker.tools.comms = false;
let worker = def
.profiles
.get(&ProfileName::from("worker"))
.expect("worker profile");
let result = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("worker"),
meerkat_id: &MeerkatId::from("w-1"),
profile: worker,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await;
assert!(
matches!(result, Err(MobError::WiringError(_))),
"tools.comms=false must be rejected at build_agent_config"
);
}
#[tokio::test]
async fn test_build_agent_config_system_prompt_includes_skills() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
let prompt = config.system_prompt.as_deref().expect("system_prompt set");
assert!(
prompt.contains("You are the team lead."),
"prompt should contain resolved inline skill"
);
}
#[tokio::test]
async fn test_build_agent_config_preloads_mob_communication_skill() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
let preload = config
.preload_skills
.as_ref()
.expect("preload_skills should be set");
assert!(
preload.iter().any(|id| id.0 == "mob-communication"),
"preload_skills should include mob-communication"
);
}
#[tokio::test]
async fn test_build_agent_config_sets_silent_comms_intents() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert!(
config
.silent_comms_intents
.contains(&"mob.peer_added".to_string()),
"silent_comms_intents should include mob.peer_added"
);
assert!(
config
.silent_comms_intents
.contains(&"mob.peer_retired".to_string()),
"silent_comms_intents should include mob.peer_retired"
);
assert_eq!(config.max_inline_peer_notifications, None);
}
#[tokio::test]
async fn test_build_agent_config_propagates_max_inline_peer_notifications() {
let mut def = sample_definition();
let lead_key = ProfileName::from("lead");
def.profiles
.get_mut(&lead_key)
.expect("lead profile")
.max_inline_peer_notifications = Some(15);
let lead = &def.profiles[&lead_key];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &lead_key,
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(config.max_inline_peer_notifications, Some(15));
}
#[tokio::test]
async fn test_build_agent_config_model() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(config.model, "claude-opus-4-6");
}
#[tokio::test]
async fn test_to_create_session_request() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
let req = to_create_session_request(&config, "Hello mob".to_string().into());
assert_eq!(req.model, "claude-opus-4-6");
assert_eq!(req.prompt.text_content(), "Hello mob");
assert!(req.system_prompt.is_some());
assert_eq!(
req.initial_turn,
meerkat_core::service::InitialTurnPolicy::Defer
);
assert_eq!(req.deferred_prompt_policy, DeferredPromptPolicy::Discard);
let build = req.build.expect("build options should be set");
assert_eq!(build.comms_name.as_deref(), Some("test-mob/lead/lead-1"));
assert!(build.peer_meta.is_some());
assert_eq!(build.realm_id.as_deref(), Some("mob:test-mob"));
assert_eq!(
build.override_builtins,
meerkat_core::ToolCategoryOverride::Enable
);
assert_eq!(
build.override_shell,
meerkat_core::ToolCategoryOverride::Enable
);
}
#[tokio::test]
async fn test_to_create_session_request_worker() {
let def = sample_definition();
let worker = &def.profiles[&ProfileName::from("worker")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("worker"),
meerkat_id: &MeerkatId::from("w-1"),
profile: worker,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
let req = to_create_session_request(&config, "Start working".to_string().into());
assert_eq!(req.model, "claude-sonnet-4-5");
assert_eq!(req.deferred_prompt_policy, DeferredPromptPolicy::Discard);
let build = req.build.expect("build options");
assert_eq!(
build.override_shell,
meerkat_core::ToolCategoryOverride::Disable
);
}
#[tokio::test]
async fn test_build_agent_config_resolves_path_skills() {
let mut def = sample_definition();
let tempdir = tempfile::tempdir().expect("tempdir");
let skill_path = tempdir.path().join("leader.md");
fs::write(&skill_path, "Path skill content for leader.").expect("write path skill");
def.skills.insert(
"path-skill".into(),
SkillSource::Path {
path: skill_path.display().to_string(),
},
);
let lead = def
.profiles
.get_mut(&ProfileName::from("lead"))
.expect("lead profile");
lead.skills.push("path-skill".into());
let lead = def
.profiles
.get(&ProfileName::from("lead"))
.expect("lead profile");
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config should resolve path skill");
let prompt = config.system_prompt.expect("system prompt");
assert!(prompt.contains("Path skill content for leader."));
assert!(!prompt.contains("[skill from:"));
}
#[tokio::test]
async fn test_build_agent_config_fails_for_missing_path_skill() {
let mut def = sample_definition();
let missing_path = std::env::temp_dir()
.join("meerkat-mob-missing-skill.md")
.display()
.to_string();
def.skills.insert(
"missing-path-skill".into(),
SkillSource::Path {
path: missing_path.clone(),
},
);
let lead = def
.profiles
.get_mut(&ProfileName::from("lead"))
.expect("lead profile");
lead.skills = vec!["missing-path-skill".into()];
let lead = def
.profiles
.get(&ProfileName::from("lead"))
.expect("lead profile");
let err = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect_err("missing path skill should fail");
match err {
MobError::Internal(message) => {
assert!(message.contains("failed to read skill file"));
assert!(message.contains(&missing_path));
}
other => panic!("expected MobError::Internal, got: {other:?}"),
}
}
#[tokio::test]
async fn test_build_agent_config_passes_app_context() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let ctx = serde_json::json!({"key": "val"});
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: Some(ctx.clone()),
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(
config.app_context,
Some(ctx),
"app_context should be passed through to AgentBuildConfig"
);
}
#[tokio::test]
async fn test_build_agent_config_none_context() {
let def = sample_definition();
let lead = &def.profiles[&ProfileName::from("lead")];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("lead"),
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(
config.app_context, None,
"app_context should be None when no context is provided"
);
}
#[tokio::test]
async fn test_build_agent_config_passes_provider_params() {
let mut def = sample_definition();
let lead_key = ProfileName::from("lead");
def.profiles
.get_mut(&lead_key)
.expect("lead profile")
.provider_params = Some(serde_json::json!({
"thinking_budget": 4096,
"top_k": 40
}));
let lead = &def.profiles[&lead_key];
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &lead_key,
meerkat_id: &MeerkatId::from("lead-1"),
profile: lead,
definition: &def,
external_tools: None,
context: None,
labels: None,
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
assert_eq!(
config.provider_params,
Some(serde_json::json!({
"thinking_budget": 4096,
"top_k": 40
})),
"provider_params should be passed through to AgentBuildConfig"
);
let req = to_create_session_request(&config, "hello".to_string().into());
let build = req.build.expect("build options should be set");
assert_eq!(
build.provider_params,
Some(serde_json::json!({
"thinking_budget": 4096,
"top_k": 40
})),
"provider_params should survive into SessionBuildOptions"
);
}
#[tokio::test]
async fn test_build_agent_config_app_labels_overwritten_by_mob_labels() {
let def = sample_definition();
let worker = &def.profiles[&ProfileName::from("worker")];
let mut app_labels = std::collections::BTreeMap::new();
app_labels.insert("faction".to_string(), "north".to_string());
app_labels.insert("mob_id".to_string(), "sneaky-override".to_string());
let config = build_agent_config(BuildAgentConfigParams {
mob_id: &def.id,
profile_name: &ProfileName::from("worker"),
meerkat_id: &MeerkatId::from("w-1"),
profile: worker,
definition: &def,
external_tools: None,
context: None,
labels: Some(app_labels),
additional_instructions: None,
shell_env: None,
mob_tool_access_context: MobToolAccessContext::None,
})
.await
.expect("build_agent_config");
let meta = config.peer_meta.as_ref().expect("peer_meta should be set");
assert_eq!(
meta.labels.get("faction").map(String::as_str),
Some("north"),
"app labels should be present in peer_meta"
);
assert_eq!(
meta.labels.get("mob_id").map(String::as_str),
Some("test-mob"),
"mob standard labels must overwrite app labels on conflict"
);
assert_eq!(meta.labels.get("role").map(String::as_str), Some("worker"));
assert_eq!(
meta.labels.get("meerkat_id").map(String::as_str),
Some("w-1")
);
}
}