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
// db_logger
// Copyright 2022 Julio Merino
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License.  You may obtain a copy
// of the License at:
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the
// License for the specific language governing permissions and limitations
// under the License.

//! A database-backed logger for use with the [`log` crate][log-crate-url].

// Keep these in sync with other top-level files.
#![allow(clippy::await_holding_refcell_ref)]
#![warn(anonymous_parameters, bad_style, missing_docs)]
#![warn(unused, unused_extern_crates, unused_import_braces, unused_qualifications)]
#![warn(unsafe_code)]

use std::sync::Arc;

mod clocks;
pub(crate) mod logger;
use crate::logger::LogEntry;
pub use logger::{init, Handle};
#[cfg(test)]
mod testutils;

#[cfg(not(any(feature = "postgres", feature = "sqlite")))]
compile_error!("one of the features ['postgres', 'sqlite'] must be enabled");
#[cfg(feature = "postgres")]
pub mod postgres;
#[cfg(feature = "sqlite")]
pub mod sqlite;

/// Opaque type representing a connection to the logging database.
#[derive(Clone)]
pub struct Connection(Arc<dyn Db + Send + Sync + 'static>);

impl Connection {
    /// Initializes the database schema.
    pub async fn create_schema(&self) -> Result<()> {
        self.0.create_schema().await
    }
}

/// Result type for this library.
pub(crate) type Result<T> = std::result::Result<T, String>;

/// Abstraction over the database connection.
#[async_trait::async_trait]
pub(crate) trait Db {
    /// Initializes the database schema.
    async fn create_schema(&self) -> Result<()>;

    /// Returns the sorted list of all log entries in the database.
    ///
    /// Given that this is exposed for testing purposes only, this just returns a flat textual
    /// representation of the log entry and does not try to deserialize it as a `LogEntry`.  This
    /// is for simplicity given that a `LogEntry` keeps references to static strings and we cannot
    /// obtain those from the database.
    async fn get_log_entries(&self) -> Result<Vec<String>>;

    /// Appends a series of `entries` to the log.
    ///
    /// All entries are inserted at once into the database to avoid unnecessary round trips for each
    /// log entry, which happen to be very expensive for something as frequent as log messages.
    ///
    /// This takes a `Vec` instead of a slice for efficiency, as the writes may have to truncate the
    /// entries.
    async fn put_log_entries(&self, entries: Vec<LogEntry<'_, '_>>) -> Result<()>;
}

/// Fits the string in `input` within the specified `max_len`.
fn truncate_option_str(input: Option<&str>, max_len: usize) -> Option<String> {
    match input {
        Some(s) => {
            let mut s = s.to_owned();
            s.truncate(max_len);
            Some(s)
        }
        None => None,
    }
}