kimun_notes/cli/commands/
journal.rs1use 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 #[arg(long)]
15 pub date: Option<String>,
16 pub content: Option<String>,
18 #[command(subcommand)]
19 pub subcommand: Option<JournalSubcommand>,
20}
21
22#[derive(Subcommand, Debug)]
23pub enum JournalSubcommand {
24 Show {
26 #[arg(long)]
28 date: Option<String>,
29 #[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
44fn 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
57fn 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 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 = ¬e_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}