lash-core 0.1.0-alpha.71

Sans-IO turn machine and runtime kernel for the lash agent runtime.
Documentation
use std::collections::{BTreeMap, BTreeSet};
use std::sync::Arc;

use super::*;

impl PluginSession {
    pub fn resolved_tool_catalog(
        &self,
        session_id: &str,
    ) -> Result<Arc<crate::ToolCatalog>, PluginError> {
        let tools = self.tools.tool_manifests();
        let contract_provider = Arc::clone(&self.tools);
        let resolve_contract: lash_sansio::ToolContractResolver =
            Arc::new(move |name: &str| contract_provider.resolve_contract(name));
        Ok(Arc::new(self.resolve_tool_catalog(ToolCatalogContext {
            session_id: session_id.to_string(),
            tools,
            resolve_contract: Some(Arc::clone(&resolve_contract)),
            tool_access: self.tool_access.clone(),
            subagent: self.subagent.clone(),
            extensions: self.extensions.clone(),
        })?))
    }

    /// Project every Tool Catalog member to a JSON record for host-owned
    /// discovery (e.g. the reference `search_tools` example in `lash-cli`).
    pub fn tool_catalog(&self, session_id: &str) -> Result<Vec<serde_json::Value>, PluginError> {
        let catalog = self.resolved_tool_catalog(session_id)?;
        Ok(crate::tool_registry::project_tool_catalog(
            catalog.tools.iter().cloned(),
        ))
    }

    pub fn resolve_tool_catalog(
        &self,
        ctx: ToolCatalogContext,
    ) -> Result<crate::ToolCatalog, PluginError> {
        let mut contributions = collect_owned_sync(
            &self.contributions.tool_catalog_contributors,
            ToolCatalogContext {
                session_id: ctx.session_id.clone(),
                tools: ctx.tools.clone(),
                resolve_contract: ctx.resolve_contract.clone(),
                tool_access: ctx.tool_access.clone(),
                subagent: ctx.subagent.clone(),
                extensions: ctx.extensions.clone(),
            },
            |hook, ctx| hook(ctx),
        )?
        .into_iter()
        .map(|owned| owned.value)
        .collect::<Vec<_>>();
        contributions.push(self.tool_catalog_overlay.clone());
        let (tools, resolve_contract) = if ctx.tool_access.tools.is_empty() {
            (ctx.tools, ctx.resolve_contract)
        } else {
            let contracts = ctx
                .tool_access
                .tools
                .iter()
                .map(|tool| (tool.name().to_string(), Arc::new(tool.contract())))
                .collect::<BTreeMap<_, _>>();
            (
                ctx.tool_access
                    .tools
                    .iter()
                    .map(|tool| tool.manifest())
                    .collect(),
                Some(Arc::new(move |name: &str| contracts.get(name).cloned())
                    as lash_sansio::ToolContractResolver),
            )
        };
        let authority_hidden_tools = tools
            .iter()
            .filter(|tool| ctx.tool_access.hides(&tool.name))
            .map(|tool| tool.name.clone())
            .collect::<BTreeSet<_>>();
        if !authority_hidden_tools.is_empty() {
            contributions.push(ToolCatalogContribution {
                remove: authority_hidden_tools.into_iter().collect(),
            });
        }
        Ok(crate::build_tool_catalog(crate::ToolCatalogBuildInput {
            tools,
            resolve_contract,
            contributions,
        }))
    }
}