systemprompt-cli 0.14.0

Unified CLI for systemprompt.io AI governance: agent orchestration, MCP governance, analytics, profiles, cloud deploy, and self-hosted operations.
Documentation
//! `cloud profile` subcommands for managing deployment profiles.
//!
//! Dispatches [`ProfileCommands`] to create, list, show, edit, and delete
//! profiles, and drives the interactive operation picker when no subcommand is
//! given.

mod api_keys;
mod args;
mod builders;
mod create;
mod create_setup;
mod create_tenant;
pub(super) mod delete;
mod edit;
mod edit_secrets;
mod edit_settings;
mod list;
mod profile_steps;
mod show;
mod show_display;
mod show_types;
pub mod templates;

pub use api_keys::collect_api_keys;
pub use args::{CreateArgs, DeleteArgs, EditArgs, ProfileCommands, ShowFilter, TenantTypeArg};
pub use create::{CreatedProfile, create_profile_for_tenant};
pub use create_setup::{get_cloud_user, handle_local_tenant_setup};

use crate::cli_settings::CliConfig;
use crate::shared::render_result;
use anyhow::Result;
use dialoguer::Select;
use dialoguer::theme::ColorfulTheme;
use systemprompt_cloud::{ProfilePath, ProjectContext};
use systemprompt_logging::CliService;

pub async fn execute(cmd: Option<ProfileCommands>, config: &CliConfig) -> Result<()> {
    if let Some(cmd) = cmd {
        execute_command(cmd, config).await.map(drop)
    } else {
        if !config.is_interactive() {
            return Err(anyhow::anyhow!(
                "Profile subcommand required in non-interactive mode"
            ));
        }
        while let Some(cmd) = select_operation()? {
            if execute_command(cmd, config).await? {
                break;
            }
        }
        Ok(())
    }
}

async fn execute_command(cmd: ProfileCommands, config: &CliConfig) -> Result<bool> {
    match cmd {
        ProfileCommands::Create(args) => create::execute(&args, config).await.map(|()| true),
        ProfileCommands::List => {
            let result = list::execute(config)?;
            render_result(&result);
            Ok(false)
        },
        ProfileCommands::Show {
            name,
            filter,
            json,
            yaml,
        } => show::execute(name.as_deref(), filter, json, yaml, config).map(|()| false),
        ProfileCommands::Delete(args) => {
            let result = delete::execute(&args, config)?;
            render_result(&result);
            Ok(false)
        },
        ProfileCommands::Edit(args) => edit::execute(&args, config).map(|()| false),
    }
}

fn select_operation() -> Result<Option<ProfileCommands>> {
    let ctx = ProjectContext::discover();
    let profiles_dir = ctx.profiles_dir();
    let has_profiles = profiles_dir.exists()
        && std::fs::read_dir(&profiles_dir).is_ok_and(|entries| {
            entries
                .filter_map(Result::ok)
                .any(|e| e.path().is_dir() && ProfilePath::Config.resolve(&e.path()).exists())
        });

    let edit_label = if has_profiles {
        "Edit".to_owned()
    } else {
        "Edit (unavailable - no profiles)".to_owned()
    };
    let delete_label = if has_profiles {
        "Delete".to_owned()
    } else {
        "Delete (unavailable - no profiles)".to_owned()
    };

    let operations = vec![
        "List".to_owned(),
        edit_label,
        delete_label,
        "Done".to_owned(),
    ];

    let selection = Select::with_theme(&ColorfulTheme::default())
        .with_prompt("Profile operation")
        .items(&operations)
        .default(0)
        .interact()?;

    let cmd = match selection {
        0 => Some(ProfileCommands::List),
        1 | 2 if !has_profiles => {
            CliService::warning("No profiles found");
            CliService::info(
                "Run 'systemprompt cloud tenant create' (or 'just tenant') to create a tenant \
                 with a profile.",
            );
            return Ok(Some(ProfileCommands::List));
        },
        1 => Some(ProfileCommands::Edit(EditArgs {
            name: None,
            set_anthropic_key: None,
            set_openai_key: None,
            set_gemini_key: None,
            set_github_token: None,
            set_database_url: None,
            set_external_url: None,
            set_host: None,
            set_port: None,
        })),
        2 => select_profile("Select profile to delete")?
            .map(|name| ProfileCommands::Delete(DeleteArgs { name, yes: false })),
        3 => None,
        _ => unreachable!(),
    };

    Ok(cmd)
}

fn select_profile(prompt: &str) -> Result<Option<String>> {
    let ctx = ProjectContext::discover();
    let profiles_dir = ctx.profiles_dir();

    if !profiles_dir.exists() {
        CliService::warning("No profiles directory found.");
        return Ok(None);
    }

    let profiles: Vec<String> = std::fs::read_dir(&profiles_dir)?
        .filter_map(Result::ok)
        .filter(|e| e.path().is_dir() && ProfilePath::Config.resolve(&e.path()).exists())
        .filter_map(|e| e.file_name().to_str().map(String::from))
        .collect();

    if profiles.is_empty() {
        CliService::warning("No profiles found.");
        return Ok(None);
    }

    let selection = Select::with_theme(&ColorfulTheme::default())
        .with_prompt(prompt)
        .items(&profiles)
        .default(0)
        .interact()?;

    Ok(Some(profiles[selection].clone()))
}