summaryrefslogtreecommitdiff
path: root/src/error.rs
blob: 2a6555f7ee4ee8829a633e7a91a22d5796e77f60 (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
use std::{error, fmt};

use axum::{
    http::StatusCode,
    response::{IntoResponse, Response},
};

use crate::id::Id as BaseId;

// I'm making an effort to avoid `anyhow` here, as that crate is _enormously_
// complex (though very usable). We don't need to be overly careful about
// allocations on errors in this app, so this is fine for most "general
// failure" cases.
type BoxedError = Box<dyn error::Error + Send + Sync>;

// Returns a 500 Internal Server Error to the client. Meant to be used via the
// `?` operator; _does not_ return the originating error to the client.
pub struct InternalError(Id, BoxedError);

impl<E> From<E> for InternalError
where
    E: Into<BoxedError>,
{
    fn from(error: E) -> Self {
        let id = Id::generate();
        Self(id, error.into())
    }
}

impl IntoResponse for InternalError {
    fn into_response(self) -> Response {
        let Self(id, error) = self;
        eprintln!("hi: [{id}] {error}");
        (
            StatusCode::INTERNAL_SERVER_ERROR,
            format!("internal server error\nerror id: {}", id),
        )
            .into_response()
    }
}

/// Transient identifier for an InternalError. Prefixed with `E`.
pub struct Id(BaseId);

impl From<BaseId> for Id {
    fn from(id: BaseId) -> Self {
        Self(id)
    }
}

impl Id {
    pub fn generate() -> Self {
        BaseId::generate("E")
    }
}

impl fmt::Display for Id {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        self.0.fmt(f)
    }
}