summaryrefslogtreecommitdiffstats
path: root/atuin-server/src/handlers/history.rs
blob: 1aebdde1f8c9405e56c2cdd5c0a9dbb0babe6880 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
use std::convert::Infallible;

use warp::{http::StatusCode, reply::json};

use crate::database::Database;
use crate::models::{NewHistory, User};
use atuin_common::api::{
    AddHistoryRequest, CountResponse, ErrorResponse, SyncHistoryRequest, SyncHistoryResponse,
};

pub async fn count(
    user: User,
    db: impl Database + Clone + Send + Sync,
) -> Result<Box<dyn warp::Reply>, Infallible> {
    db.count_history(&user).await.map_or(
        Ok(Box::new(ErrorResponse::reply(
            "failed to query history count",
            StatusCode::INTERNAL_SERVER_ERROR,
        ))),
        |count| Ok(Box::new(json(&CountResponse { count }))),
    )
}

pub async fn list(
    req: SyncHistoryRequest,
    user: User,
    db: impl Database + Clone + Send + Sync,
) -> Result<Box<dyn warp::Reply>, Infallible> {
    let history = db
        .list_history(
            &user,
            req.sync_ts.naive_utc(),
            req.history_ts.naive_utc(),
            req.host,
        )
        .await;

    if let Err(e) = history {
        error!("failed to load history: {}", e);
        let resp =
            ErrorResponse::reply("failed to load history", StatusCode::INTERNAL_SERVER_ERROR);
        let resp = Box::new(resp);
        return Ok(resp);
    }

    let history: Vec<String> = history
        .unwrap()
        .iter()
        .map(|i| i.data.to_string())
        .collect();

    debug!(
        "loaded {} items of history for user {}",
        history.len(),
        user.id
    );

    Ok(Box::new(json(&SyncHistoryResponse { history })))
}

pub async fn add(
    req: Vec<AddHistoryRequest>,
    user: User,
    db: impl Database + Clone + Send + Sync,
) -> Result<Box<dyn warp::Reply>, Infallible> {
    debug!("request to add {} history items", req.len());

    let history: Vec<NewHistory> = req
        .iter()
        .map(|h| NewHistory {
            client_id: h.id.as_str(),
            user_id: user.id,
            hostname: h.hostname.as_str(),
            timestamp: h.timestamp.naive_utc(),
            data: h.data.as_str(),
        })
        .collect();

    if let Err(e) = db.add_history(&history).await {
        error!("failed to add history: {}", e);

        return Ok(Box::new(ErrorResponse::reply(
            "failed to add history",
            StatusCode::INTERNAL_SERVER_ERROR,
        )));
    };

    Ok(Box::new(warp::reply()))
}