Line data Source code
1 : use std::{error::Error as StdError, fmt, io};
2 :
3 : /// Upcast (almost) any error into an opaque [`io::Error`].
4 0 : pub fn io_error(e: impl Into<Box<dyn StdError + Send + Sync>>) -> io::Error {
5 0 : io::Error::new(io::ErrorKind::Other, e)
6 0 : }
7 :
8 : /// A small combinator for pluggable error logging.
9 0 : pub fn log_error<E: fmt::Display>(e: E) -> E {
10 0 : tracing::error!("{e}");
11 0 : e
12 0 : }
13 :
14 : /// Marks errors that may be safely shown to a client.
15 : /// This trait can be seen as a specialized version of [`ToString`].
16 : ///
17 : /// NOTE: This trait should not be implemented for [`anyhow::Error`], since it
18 : /// is way too convenient and tends to proliferate all across the codebase,
19 : /// ultimately leading to accidental leaks of sensitive data.
20 : pub trait UserFacingError: ReportableError {
21 : /// Format the error for client, stripping all sensitive info.
22 : ///
23 : /// Although this might be a no-op for many types, it's highly
24 : /// recommended to override the default impl in case error type
25 : /// contains anything sensitive: various IDs, IP addresses etc.
26 : #[inline(always)]
27 0 : fn to_string_client(&self) -> String {
28 0 : self.to_string()
29 0 : }
30 : }
31 :
32 12 : #[derive(Copy, Clone, Debug, Eq, PartialEq)]
33 : pub enum ErrorKind {
34 : /// Wrong password, unknown endpoint, protocol violation, etc...
35 : User,
36 :
37 : /// Network error between user and proxy. Not necessarily user error
38 : ClientDisconnect,
39 :
40 : /// Proxy self-imposed user rate limits
41 : RateLimit,
42 :
43 : /// Proxy self-imposed service-wise rate limits
44 : ServiceRateLimit,
45 :
46 : /// internal errors
47 : Service,
48 :
49 : /// Error communicating with control plane
50 : ControlPlane,
51 :
52 : /// Postgres error
53 : Postgres,
54 :
55 : /// Error communicating with compute
56 : Compute,
57 : }
58 :
59 : impl ErrorKind {
60 0 : pub fn to_metric_label(&self) -> &'static str {
61 0 : match self {
62 0 : ErrorKind::User => "user",
63 0 : ErrorKind::ClientDisconnect => "clientdisconnect",
64 0 : ErrorKind::RateLimit => "ratelimit",
65 0 : ErrorKind::ServiceRateLimit => "serviceratelimit",
66 0 : ErrorKind::Service => "service",
67 0 : ErrorKind::ControlPlane => "controlplane",
68 0 : ErrorKind::Postgres => "postgres",
69 0 : ErrorKind::Compute => "compute",
70 : }
71 0 : }
72 : }
73 :
74 : pub trait ReportableError: fmt::Display + Send + 'static {
75 : fn get_error_kind(&self) -> ErrorKind;
76 : }
77 :
78 : impl ReportableError for tokio_postgres::error::Error {
79 0 : fn get_error_kind(&self) -> ErrorKind {
80 0 : if self.as_db_error().is_some() {
81 0 : ErrorKind::Postgres
82 : } else {
83 0 : ErrorKind::Compute
84 : }
85 0 : }
86 : }
|