kimun-notes 0.11.0

A terminal-based notes application
Documentation
// tui/src/cli/mod.rs
pub mod commands;
pub mod helpers;
pub mod json_output;
pub mod metadata_extractor;
pub mod output;

use clap::Subcommand;
use color_eyre::eyre::{Result, eyre};
use commands::JournalArgs;
use commands::note_ops::NoteSubcommand;
use commands::workspace::WorkspaceSubcommand;
use helpers::{
    create_and_init_vault, load_and_resolve_workspace, load_settings, resolve_inbox_path,
    resolve_quick_note_path,
};
use kimun_core::{NoteVault, VaultConfig};
use output::OutputFormat;

#[derive(Subcommand)]
pub enum CliCommand {
    /// Search notes by query
    Search {
        query: String,
        #[arg(long, value_enum, default_value = "text")]
        format: OutputFormat,
    },
    /// List all notes
    Notes {
        #[arg(long, help = "Filter notes by path prefix")]
        path: Option<String>,
        #[arg(long, value_enum, default_value = "text")]
        format: OutputFormat,
    },
    /// Manage workspaces
    Workspace {
        #[command(subcommand)]
        subcommand: WorkspaceSubcommand,
    },
    /// Note operations (create, append, show)
    Note {
        #[command(subcommand)]
        subcommand: NoteSubcommand,
    },
    /// Append to or show journal entries
    Journal(JournalArgs),
    /// Start the MCP server (stdio transport)
    Mcp,
    /// List all hashtag labels in the vault with note counts
    Labels {
        #[arg(long, value_enum, default_value = "text")]
        format: OutputFormat,
    },
}

pub async fn run_cli(command: CliCommand, config_path: Option<std::path::PathBuf>) -> Result<()> {
    match command {
        CliCommand::Workspace { subcommand } => {
            let mut settings = load_settings(config_path)?;
            commands::workspace::run(subcommand, &mut settings).await
        }
        CliCommand::Note { subcommand } => {
            let (settings, workspace_path, workspace_name) =
                load_and_resolve_workspace(config_path)?;
            let quick_note_path = resolve_quick_note_path(&settings);
            let inbox_path = resolve_inbox_path(&settings);
            let cache_path = settings.cache_path_for(&workspace_name);
            let mut vault =
                NoteVault::new(VaultConfig::new(&workspace_path).with_db_path(cache_path)).await?;
            vault.set_inbox_path(kimun_core::nfs::VaultPath::new(&inbox_path));
            match vault.validate().await? {
                kimun_core::DBStatus::Ready => {
                    commands::note_ops::run(subcommand, &vault, &quick_note_path, &workspace_name)
                        .await
                }
                status => Err(eyre!(
                    "Workspace index is not ready ({status}).\nRun `kimun workspace reindex` to initialise it."
                )),
            }
        }
        CliCommand::Search { query, format } => {
            let (vault, workspace_name) = create_and_init_vault(config_path).await?;
            commands::search::run(&vault, &query, format, &workspace_name, false).await
        }
        CliCommand::Notes { path, format } => {
            let (vault, workspace_name) = create_and_init_vault(config_path).await?;
            commands::notes::run(&vault, path.as_deref(), format, &workspace_name, false).await
        }
        CliCommand::Journal(args) => {
            let (vault, workspace_name) = create_and_init_vault(config_path).await?;
            commands::journal::run(args, &vault, &workspace_name).await
        }
        CliCommand::Mcp => commands::mcp::run(config_path).await,
        CliCommand::Labels { format } => {
            let (vault, workspace_name) = create_and_init_vault(config_path).await?;
            commands::labels::run(&vault, format, &workspace_name).await
        }
    }
}