systemprompt_cli/commands/core/content/analytics/
clicks.rs1use crate::cli_settings::CliConfig;
2use crate::commands::core::content::types::{ClickRow, ClicksOutput};
3use crate::shared::CommandOutput;
4use anyhow::Result;
5use clap::Args;
6use systemprompt_content::LinkAnalyticsService;
7use systemprompt_database::DbPool;
8use systemprompt_identifiers::LinkId;
9use systemprompt_runtime::AppContext;
10
11#[derive(Debug, Args)]
12pub struct ClicksArgs {
13 #[arg(help = "Link ID")]
14 pub link_id: String,
15
16 #[arg(long, default_value = "20")]
17 pub limit: i64,
18
19 #[arg(long, default_value = "0")]
20 pub offset: i64,
21}
22
23pub async fn execute(args: ClicksArgs, config: &CliConfig) -> Result<CommandOutput> {
24 let ctx = AppContext::new().await?;
25 execute_with_pool(args, ctx.db_pool(), config).await
26}
27
28pub async fn execute_with_pool(
29 args: ClicksArgs,
30 pool: &DbPool,
31 _config: &CliConfig,
32) -> Result<CommandOutput> {
33 let service = LinkAnalyticsService::new(pool)?;
34
35 let link_id = LinkId::new(args.link_id.clone());
36 let clicks = service
37 .get_link_clicks(&link_id, Some(args.limit), Some(args.offset))
38 .await?;
39
40 let total = clicks.len() as i64;
41 let click_rows: Vec<ClickRow> = clicks
42 .into_iter()
43 .filter_map(|click| {
44 Some(ClickRow {
45 click_id: click.id.to_string(),
46 session_id: click.session_id,
47 user_id: click.user_id,
48 clicked_at: click.clicked_at?,
49 referrer_page: click.referrer_page,
50 device_type: click.device_type,
51 country: click.country,
52 is_conversion: click.is_conversion.unwrap_or(false),
53 })
54 })
55 .collect();
56
57 let output = ClicksOutput {
58 link_id,
59 clicks: click_rows,
60 total,
61 };
62
63 Ok(CommandOutput::table_of(
64 vec![
65 "click_id",
66 "session_id",
67 "clicked_at",
68 "device_type",
69 "country",
70 ],
71 &output.clicks,
72 )
73 .with_title("Link Clicks"))
74}