diff --git a/server/main-api/src/calendar/mod.rs b/server/main-api/src/calendar/mod.rs index 84a4b7063..2468d93b1 100644 --- a/server/main-api/src/calendar/mod.rs +++ b/server/main-api/src/calendar/mod.rs @@ -50,14 +50,14 @@ pub async fn calendar_handler( Ok(ids) => ids, Err(e) => return e, }; - let locations = match get_locations(&data.db, &ids).await { + let locations = match get_locations(&data.pool, &ids).await { Ok(l) => l.0, Err(e) => return e, }; if let Err(e) = validate_locations(&ids, &locations) { return e; } - match get_from_db(&data.db, &locations, &args.start_after, &args.end_before).await { + match get_from_db(&data.pool, &locations, &args.start_after, &args.end_before).await { Ok(events) => HttpResponse::Ok().json(events), Err(e) => { error!("could not get entries from the db for {ids:?} because {e:?}"); @@ -278,7 +278,7 @@ mod tests { let app = test::init_service( App::new() .app_data(web::Data::new(AppData { - db: pg.pool.clone(), + pool: pg.pool.clone(), })) .service(calendar_handler), ) diff --git a/server/main-api/src/details.rs b/server/main-api/src/details.rs index f1ebe63c1..cbb1d362a 100644 --- a/server/main-api/src/details.rs +++ b/server/main-api/src/details.rs @@ -15,16 +15,16 @@ pub async fn get_handler( let id = params .into_inner() .replace(|c: char| c.is_whitespace() || c.is_control(), ""); - let Some((probable_id, redirect_url)) = get_alias_and_redirect(&data.db, &id).await else { + let Some((probable_id, redirect_url)) = get_alias_and_redirect(&data.pool, &id).await else { return HttpResponse::NotFound().body("Not found"); }; let result = if args.should_use_english() { sqlx::query_scalar!("SELECT data FROM en WHERE key = $1", probable_id) - .fetch_optional(&data.db) + .fetch_optional(&data.pool) .await } else { sqlx::query_scalar!("SELECT data FROM de WHERE key = $1", probable_id) - .fetch_optional(&data.db) + .fetch_optional(&data.pool) .await }; match result { diff --git a/server/main-api/src/main.rs b/server/main-api/src/main.rs index 56bb84de7..960f5c139 100644 --- a/server/main-api/src/main.rs +++ b/server/main-api/src/main.rs @@ -1,6 +1,7 @@ use std::borrow::Cow; use std::collections::HashMap; use std::error::Error; +use std::sync::Arc; use actix_cors::Cors; use actix_web::{get, middleware, web, App, HttpResponse, HttpServer}; @@ -9,7 +10,8 @@ use meilisearch_sdk::client::Client; use sentry::SessionMode; use sqlx::postgres::PgPoolOptions; use sqlx::prelude::*; -use sqlx::PgPool; +use sqlx::{PgPool, Pool, Postgres}; +use tokio::sync::RwLock; use tracing::{debug, debug_span, error, info}; use tracing_actix_web::TracingLogger; @@ -27,9 +29,25 @@ type BoxedError = Box; const MAX_JSON_PAYLOAD: usize = 1024 * 1024; // 1 MB -#[derive(Debug)] +#[derive(Clone, Debug)] pub struct AppData { - db: PgPool, + /// shared [sqlx::PgPool] to connect to postgres + pool: PgPool, + /// necessary, as otherwise we could return empty results during initialisation + meilisearch_initialised: Arc>, +} + +impl AppData { + async fn new() -> Self { + let pool = PgPoolOptions::new() + .connect(&connection_string()) + .await + .expect("make sure that postgres is running in the background"); + AppData { + pool, + meilisearch_initialised: Arc::new(Default::default()), + } + } } #[get("/api/status")] @@ -38,7 +56,7 @@ async fn health_status_handler(data: web::Data) -> HttpResponse { Some(hash) => format!("https://github.com/TUM-Dev/navigatum/tree/{hash}"), None => "unknown commit hash, probably running in development".to_string(), }; - return match data.db.execute("SELECT 1").await { + return match data.pool.execute("SELECT 1").await { Ok(_) => HttpResponse::Ok() .content_type("text/plain") .body(format!("healthy\nsource_code: {github_link}")), @@ -103,13 +121,10 @@ fn main() -> Result<(), BoxedError> { actix_web::rt::System::new().block_on(async { run().await })?; Ok(()) } -async fn run_maintenance_work() { - let pool = PgPoolOptions::new() - .connect(&connection_string()) - .await - .expect("make sure that postgres is running in the background"); +async fn run_maintenance_work(pool: Pool, meilisearch_initalised: Arc>) { if std::env::var("SKIP_MS_SETUP") != Ok("true".to_string()) { let _ = debug_span!("updating meilisearch data").enter(); + let _ = meilisearch_initalised.write().await; let ms_url = std::env::var("MIELI_URL").unwrap_or_else(|_| "http://localhost:7700".to_string()); let client = Client::new(ms_url, std::env::var("MEILI_MASTER_KEY").ok()).unwrap(); @@ -130,7 +145,11 @@ async fn run_maintenance_work() { /// we split main and run because otherwise sentry could not be properly instrumented async fn run() -> Result<(), BoxedError> { - let maintenance_thread = tokio::spawn(run_maintenance_work()); + let data = AppData::new().await; + let maintenance_thread = tokio::spawn(run_maintenance_work( + data.pool.clone(), + data.meilisearch_initialised.clone(), + )); debug!("setting up metrics"); let labels = HashMap::from([( @@ -140,15 +159,11 @@ async fn run() -> Result<(), BoxedError> { .to_string(), )]); let prometheus = PrometheusMetricsBuilder::new("navigatum_mainapi") - .endpoint("/api/main/metrics") + .endpoint("/api/metrics") .const_labels(labels) .build() .unwrap(); - let pool = PgPoolOptions::new() - .connect(&connection_string()) - .await - .expect("make sure that postgres is running in the background"); - let shutdown_pool_clone = pool.clone(); + let shutdown_pool_clone = data.pool.clone(); info!("running the server"); HttpServer::new(move || { let cors = Cors::default() @@ -164,7 +179,7 @@ async fn run() -> Result<(), BoxedError> { .wrap(middleware::Compress::default()) .wrap(sentry_actix::Sentry::new()) .app_data(web::JsonConfig::default().limit(MAX_JSON_PAYLOAD)) - .app_data(web::Data::new(AppData { db: pool.clone() })) + .app_data(web::Data::new(data.clone())) .service(health_status_handler) .service(calendar::calendar_handler) .service(web::scope("/api/preview").configure(maps::configure)) diff --git a/server/main-api/src/maps/mod.rs b/server/main-api/src/maps/mod.rs index fdb198aa2..9cade14d7 100644 --- a/server/main-api/src/maps/mod.rs +++ b/server/main-api/src/maps/mod.rs @@ -201,12 +201,12 @@ pub async fn maps_handler( let id = params .into_inner() .replace(|c: char| c.is_whitespace() || c.is_control(), ""); - if let Some(redirect_url) = get_possible_redirect_url(&data.db, &id, &args).await { + if let Some(redirect_url) = get_possible_redirect_url(&data.pool, &id, &args).await { let mut res = HttpResponse::PermanentRedirect(); res.insert_header((LOCATION, redirect_url)); return res.finish(); } - let data = match get_localised_data(&data.db, &id, args.lang.should_use_english()).await { + let data = match get_localised_data(&data.pool, &id, args.lang.should_use_english()).await { Ok(data) => data, Err(e) => { return e; diff --git a/server/main-api/src/search/mod.rs b/server/main-api/src/search/mod.rs index 3502b4d75..d655ea570 100644 --- a/server/main-api/src/search/mod.rs +++ b/server/main-api/src/search/mod.rs @@ -1,5 +1,6 @@ use std::time::Instant; +use crate::AppData; use actix_web::{get, web, HttpResponse}; use cached::proc_macro::cached; use serde::{Deserialize, Serialize}; @@ -78,8 +79,12 @@ impl From<&SearchQueryArgs> for Highlighting { } } #[get("/api/search")] -pub async fn search_handler(web::Query(args): web::Query) -> HttpResponse { +pub async fn search_handler( + data: web::Data, + web::Query(args): web::Query, +) -> HttpResponse { let start_time = Instant::now(); + let _ = data.meilisearch_initialised.read().await; // otherwise we could return empty results during initialisation let limits = Limits::from(&args); let highlighting = Highlighting::from(&args);