Skip to main content

kimun_notes/cli/commands/
journal.rs

1// tui/src/cli/commands/journal.rs
2//
3// Top-level `kimun journal` command: append to and show journal entries.
4
5use clap::Subcommand;
6use color_eyre::eyre::Result;
7use kimun_core::{NoteVault, nfs::VaultPath};
8
9use crate::cli::output::OutputFormat;
10
11#[derive(clap::Args, Debug)]
12pub struct JournalArgs {
13    /// Date in YYYY-MM-DD format (defaults to today)
14    #[arg(long)]
15    pub date: Option<String>,
16    /// Text to append (reads from stdin if omitted and stdin is not a TTY)
17    pub content: Option<String>,
18    #[command(subcommand)]
19    pub subcommand: Option<JournalSubcommand>,
20}
21
22#[derive(Subcommand, Debug)]
23pub enum JournalSubcommand {
24    /// Show a journal entry
25    Show {
26        /// Date in YYYY-MM-DD format (defaults to today)
27        #[arg(long)]
28        date: Option<String>,
29        /// Output format
30        #[arg(long, value_enum, default_value = "text")]
31        format: OutputFormat,
32    },
33}
34
35pub async fn run(args: JournalArgs, vault: &NoteVault, workspace_name: &str) -> Result<()> {
36    match args.subcommand {
37        Some(JournalSubcommand::Show { date, format }) => {
38            run_show(vault, date.as_deref(), format, workspace_name).await
39        }
40        None => run_append(vault, args.date.as_deref(), args.content).await,
41    }
42}
43
44/// Validate and return a `YYYY-MM-DD` date string. Defaults to today when `None`.
45fn resolve_date(date: Option<&str>) -> Result<String> {
46    match date {
47        None => Ok(chrono::Utc::now().format("%Y-%m-%d").to_string()),
48        Some(d) => {
49            chrono::NaiveDate::parse_from_str(d, "%Y-%m-%d").map_err(|_| {
50                color_eyre::eyre::eyre!("Invalid date '{}' — expected format YYYY-MM-DD", d)
51            })?;
52            Ok(d.to_string())
53        }
54    }
55}
56
57/// Build the vault path for a journal entry using the vault's configured journal path.
58fn journal_entry_path(vault: &NoteVault, date_str: &str) -> VaultPath {
59    vault
60        .journal_path()
61        .append(&VaultPath::note_path_from(date_str))
62        .absolute()
63}
64
65async fn run_append(vault: &NoteVault, date: Option<&str>, content: Option<String>) -> Result<()> {
66    use crate::cli::helpers::resolve_content;
67
68    let text = resolve_content(content)?;
69    if text.is_empty() {
70        return Ok(());
71    }
72
73    // Today and a specific date both resolve to journal/<date>; append under the
74    // per-note lock so concurrent journal writes can't lose an entry.
75    let date_str = resolve_date(date)?;
76    let vault_path = journal_entry_path(vault, &date_str);
77    vault
78        .append_to_note(&vault_path, &text, Some(format!("# {}\n\n", date_str)))
79        .await
80        .map_err(|e| color_eyre::eyre::eyre!("{}", e))?;
81
82    println!("Note saved: {}", vault_path);
83    Ok(())
84}
85
86async fn run_show(
87    vault: &NoteVault,
88    date: Option<&str>,
89    format: OutputFormat,
90    workspace_name: &str,
91) -> Result<()> {
92    use crate::cli::commands::note_ops::format_note_show_text;
93    use crate::cli::json_output::{
94        JsonNoteEntry, JsonNoteMetadata, JsonOutput, JsonOutputMetadata,
95    };
96    use crate::cli::metadata_extractor::{extract_headers, extract_links, extract_tags};
97    use chrono::Utc;
98    use kimun_core::error::{FSError, VaultError};
99    use std::time::UNIX_EPOCH;
100
101    if matches!(format, OutputFormat::Paths) {
102        return Err(color_eyre::eyre::eyre!(
103            "--format paths is not valid for journal show; use 'text' or 'json'"
104        ));
105    }
106
107    let date_str = resolve_date(date)?;
108    let vault_path = journal_entry_path(vault, &date_str);
109
110    let note_details = vault.load_note(&vault_path).await.map_err(|e| match e {
111        VaultError::FSError(FSError::VaultPathNotFound { .. }) => {
112            color_eyre::eyre::eyre!("No journal entry found for {}", date_str)
113        }
114        _ => color_eyre::eyre::eyre!("{}", e),
115    })?;
116
117    let content = &note_details.raw_text;
118    let content_data = note_details.get_content_data();
119
120    let backlinks = vault
121        .get_backlinks(&vault_path)
122        .await
123        .map_err(|e| color_eyre::eyre::eyre!("{}", e))?;
124    let backlink_paths: Vec<String> = backlinks.iter().map(|(e, _)| e.path.to_string()).collect();
125
126    match format {
127        OutputFormat::Text => {
128            let tags = extract_tags(content);
129            let links = extract_links(content);
130            print!(
131                "{}",
132                format_note_show_text(
133                    &vault_path,
134                    content,
135                    &content_data.title,
136                    &tags,
137                    &links,
138                    &backlink_paths,
139                )
140            );
141        }
142        OutputFormat::Json => {
143            let meta = tokio::fs::metadata(vault.path_to_pathbuf(&vault_path))
144                .await
145                .map_err(|e| color_eyre::eyre::eyre!("{}", e))?;
146            let modified_secs = meta
147                .modified()
148                .map(|t| t.duration_since(UNIX_EPOCH).unwrap_or_default().as_secs())
149                .unwrap_or(0);
150            let tags = extract_tags(content);
151            let links = extract_links(content);
152            let headers = extract_headers(content);
153            let journal_date = vault
154                .journal_date(&vault_path)
155                .map(|d| d.format("%Y-%m-%d").to_string());
156            let entry = JsonNoteEntry {
157                path: vault_path.to_string_with_ext(),
158                title: content_data.title.clone(),
159                content: content.clone(),
160                size: meta.len(),
161                modified: modified_secs,
162                created: modified_secs,
163                hash: format!("{:x}", content_data.hash),
164                journal_date,
165                metadata: JsonNoteMetadata {
166                    tags,
167                    links,
168                    headers,
169                },
170                backlinks: if backlink_paths.is_empty() {
171                    None
172                } else {
173                    Some(backlink_paths)
174                },
175            };
176            let output = JsonOutput {
177                metadata: JsonOutputMetadata {
178                    workspace: workspace_name.to_string(),
179                    workspace_path: vault.workspace_path().to_string_lossy().to_string(),
180                    total_results: 1,
181                    query: None,
182                    is_listing: false,
183                    generated_at: Utc::now().to_rfc3339(),
184                },
185                notes: vec![entry],
186            };
187            print!(
188                "{}",
189                serde_json::to_string(&output).map_err(|e| color_eyre::eyre::eyre!("{}", e))?
190            );
191        }
192        OutputFormat::Paths => unreachable!("guarded above"),
193    }
194
195    Ok(())
196}