summaryrefslogtreecommitdiff
path: root/src/channel/routes.rs
blob: 4f83a8b3108040a0b6ac28084e0775c8b0ce4a3a (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
90
91
92
93
94
95
96
use axum::{
    extract::{Form, Path, State},
    http::StatusCode,
    response::{
        sse::{self, Sse},
        IntoResponse, Redirect,
    },
    routing::{get, post},
    Router,
};
use axum_extra::TypedHeader;
use chrono::{format::SecondsFormat, DateTime};
use futures::{future, stream::TryStreamExt as _};

use super::{
    header::LastEventId,
    repo::{channels::Id as ChannelId, messages::BroadcastMessage},
};
use crate::{
    app::App, clock::RequestedAt, error::BoxedError, error::InternalError,
    login::repo::logins::Login,
};

pub fn router() -> Router<App> {
    Router::new()
        .route("/create", post(on_create))
        .route("/:channel/send", post(on_send))
        .route("/:channel/events", get(on_events))
}

#[derive(serde::Deserialize)]
struct CreateRequest {
    name: String,
}

async fn on_create(
    State(app): State<App>,
    _: Login, // requires auth, but doesn't actually care who you are
    Form(form): Form<CreateRequest>,
) -> Result<impl IntoResponse, InternalError> {
    app.channels().create(&form.name).await?;

    Ok(Redirect::to("/"))
}

#[derive(serde::Deserialize)]
struct SendRequest {
    message: String,
}

async fn on_send(
    Path(channel): Path<ChannelId>,
    RequestedAt(sent_at): RequestedAt,
    State(app): State<App>,
    login: Login,
    Form(form): Form<SendRequest>,
) -> Result<impl IntoResponse, InternalError> {
    app.channels()
        .send(&login, &channel, &form.message, &sent_at)
        .await?;

    Ok(StatusCode::ACCEPTED)
}

async fn on_events(
    Path(channel): Path<ChannelId>,
    State(app): State<App>,
    _: Login, // requires auth, but doesn't actually care who you are
    last_event_id: Option<TypedHeader<LastEventId>>,
) -> Result<impl IntoResponse, InternalError> {
    let resume_at = last_event_id
        .map(|TypedHeader(header)| header)
        .map(|LastEventId(header)| header)
        .map(|header| DateTime::parse_from_rfc3339(&header))
        .transpose()?
        .map(|ts| ts.to_utc());

    let stream = app
        .channels()
        .events(&channel, resume_at.as_ref())
        .await?
        .and_then(|msg| future::ready(to_event(msg)));

    Ok(Sse::new(stream).keep_alive(sse::KeepAlive::default()))
}

fn to_event(msg: BroadcastMessage) -> Result<sse::Event, BoxedError> {
    let data = serde_json::to_string(&msg)?;
    let event = sse::Event::default()
        .id(msg
            .sent_at
            .to_rfc3339_opts(SecondsFormat::AutoSi, /* use_z */ true))
        .data(&data);

    Ok(event)
}