diff --git a/server/Cargo.lock b/server/Cargo.lock index 47fcb14bf..1c9708b88 100644 --- a/server/Cargo.lock +++ b/server/Cargo.lock @@ -559,6 +559,56 @@ dependencies = [ "generic-array", ] +[[package]] +name = "bollard" +version = "0.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0aed08d3adb6ebe0eff737115056652670ae290f177759aac19c30456135f94c" +dependencies = [ + "base64 0.22.1", + "bollard-stubs", + "bytes", + "futures-core", + "futures-util", + "hex", + "home", + "http 1.1.0", + "http-body-util", + "hyper 1.3.1", + "hyper-named-pipe", + "hyper-rustls 0.26.0", + "hyper-util", + "hyperlocal-next", + "log", + "pin-project-lite", + "rustls 0.22.4", + "rustls-native-certs", + "rustls-pemfile 2.1.2", + "rustls-pki-types", + "serde", + "serde_derive", + "serde_json", + "serde_repr", + "serde_urlencoded", + "thiserror", + "tokio", + "tokio-util", + "tower-service", + "url", + "winapi", +] + +[[package]] +name = "bollard-stubs" +version = "1.44.0-rc.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "709d9aa1c37abb89d40f19f5d0ad6f0d88cb1581264e571c9350fc5bb89cf1c5" +dependencies = [ + "serde", + "serde_repr", + "serde_with", +] + [[package]] name = "brotli" version = "6.0.0" @@ -637,7 +687,7 @@ dependencies = [ "cached_proc_macro_types", "directories", "futures", - "hashbrown", + "hashbrown 0.14.5", "instant", "once_cell", "rmp-serde", @@ -892,7 +942,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "978747c1d849a7d2ee5e8adc0159961c48fb7e5db2f06af6723b80123bb53856" dependencies = [ "cfg-if", - "hashbrown", + "hashbrown 0.14.5", "lock_api", "once_cell", "parking_lot_core 0.9.10", @@ -965,6 +1015,15 @@ dependencies = [ "dirs-sys", ] +[[package]] +name = "dirs" +version = "5.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "44c45a9d03d6676652bcb5e724c7e988de1acad23a711b5217ab9cbecbec2225" +dependencies = [ + "dirs-sys", +] + [[package]] name = "dirs-sys" version = "0.4.1" @@ -977,6 +1036,17 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "docker_credential" +version = "1.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31951f49556e34d90ed28342e1df7e1cb7a229c4cab0aecc627b5d91edd41d07" +dependencies = [ + "base64 0.21.7", + "serde", + "serde_json", +] + [[package]] name = "dotenvy" version = "0.15.7" @@ -985,9 +1055,9 @@ checksum = "1aaf95b3e5c8f23aa320147307562d361db0ae0d51242340f558153b4eb2439b" [[package]] name = "either" -version = "1.11.0" +version = "1.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a47c1c47d2f5964e29c61246e81db715514cd532db6b5116a25ea3c03d6780a2" +checksum = "3dca9240753cf90908d7e4aac30f630662b02aebaa1b58a3cadabdb23385b58b" dependencies = [ "serde", ] @@ -1142,6 +1212,21 @@ version = "1.0.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" +[[package]] +name = "foreign-types" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6f339eb8adc052cd2ca78910fda869aefa38d22d5cb648e6485e4d3fc06f3b1" +dependencies = [ + "foreign-types-shared", +] + +[[package]] +name = "foreign-types-shared" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b" + [[package]] name = "form_urlencoded" version = "1.2.1" @@ -1347,7 +1432,7 @@ dependencies = [ "futures-sink", "futures-util", "http 0.2.12", - "indexmap", + "indexmap 2.2.6", "slab", "tokio", "tokio-util", @@ -1366,7 +1451,7 @@ dependencies = [ "futures-sink", "futures-util", "http 1.1.0", - "indexmap", + "indexmap 2.2.6", "slab", "tokio", "tokio-util", @@ -1383,6 +1468,12 @@ dependencies = [ "crunchy", ] +[[package]] +name = "hashbrown" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" + [[package]] name = "hashbrown" version = "0.14.5" @@ -1399,7 +1490,7 @@ version = "0.8.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e8094feaf31ff591f651a2664fb9cfd92bba7a60ce3197265e9482ebe753c8f7" dependencies = [ - "hashbrown", + "hashbrown 0.14.5", ] [[package]] @@ -1630,6 +1721,21 @@ dependencies = [ "want", ] +[[package]] +name = "hyper-named-pipe" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73b7d8abf35697b81a825e386fc151e0d503e8cb5fcb93cc8669c376dfd6f278" +dependencies = [ + "hex", + "hyper 1.3.1", + "hyper-util", + "pin-project-lite", + "tokio", + "tower-service", + "winapi", +] + [[package]] name = "hyper-rustls" version = "0.24.2" @@ -1682,6 +1788,22 @@ dependencies = [ "webpki-roots 0.26.1", ] +[[package]] +name = "hyper-tls" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70206fc6890eaca9fde8a0bf71caa2ddfc9fe045ac9e5c70df101a7dbde866e0" +dependencies = [ + "bytes", + "http-body-util", + "hyper 1.3.1", + "hyper-util", + "native-tls", + "tokio", + "tokio-native-tls", + "tower-service", +] + [[package]] name = "hyper-util" version = "0.1.3" @@ -1702,6 +1824,21 @@ dependencies = [ "tracing", ] +[[package]] +name = "hyperlocal-next" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "acf569d43fa9848e510358c07b80f4adf34084ddc28c6a4a651ee8474c070dcc" +dependencies = [ + "hex", + "http-body-util", + "hyper 1.3.1", + "hyper-util", + "pin-project-lite", + "tokio", + "tower-service", +] + [[package]] name = "iana-time-zone" version = "0.1.60" @@ -1808,6 +1945,17 @@ version = "1.10.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "44feda355f4159a7c757171a77de25daf6411e217b4cabd03bd6650690468126" +[[package]] +name = "indexmap" +version = "1.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd070e393353796e801d209ad339e89596eb4c8d430d18ede6a1cced8fafbd99" +dependencies = [ + "autocfg", + "hashbrown 0.12.3", + "serde", +] + [[package]] name = "indexmap" version = "2.2.6" @@ -1815,7 +1963,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "168fb715dda47215e360912c096649d23d58bf392ac62f73919e831745e40f26" dependencies = [ "equivalent", - "hashbrown", + "hashbrown 0.14.5", + "serde", ] [[package]] @@ -2221,6 +2370,23 @@ dependencies = [ "typenum", ] +[[package]] +name = "native-tls" +version = "0.2.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8614eb2c83d59d1c8cc974dd3f920198647674a0a035e1af1fa58707e317466" +dependencies = [ + "libc", + "log", + "openssl", + "openssl-probe", + "openssl-sys", + "schannel", + "security-framework", + "security-framework-sys", + "tempfile", +] + [[package]] name = "navigatum-main-api" version = "1.0.0" @@ -2255,6 +2421,8 @@ dependencies = [ "sqlx", "structured-logger", "tempfile", + "testcontainers", + "testcontainers-modules", "time", "tokio", "unicode-truncate", @@ -2483,12 +2651,50 @@ version = "1.19.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92" +[[package]] +name = "openssl" +version = "0.10.64" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "95a0481286a310808298130d22dd1fef0fa571e05a8f44ec801801e84b216b1f" +dependencies = [ + "bitflags 2.5.0", + "cfg-if", + "foreign-types", + "libc", + "once_cell", + "openssl-macros", + "openssl-sys", +] + +[[package]] +name = "openssl-macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.61", +] + [[package]] name = "openssl-probe" version = "0.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" +[[package]] +name = "openssl-sys" +version = "0.9.102" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c597637d56fbc83893a35eb0dd04b2b8e7a50c91e64e9493e398b5df4fb45fa2" +dependencies = [ + "cc", + "libc", + "pkg-config", + "vcpkg", +] + [[package]] name = "option-ext" version = "0.2.0" @@ -2552,6 +2758,31 @@ dependencies = [ "windows-targets 0.52.5", ] +[[package]] +name = "parse-display" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "06af5f9333eb47bd9ba8462d612e37a8328a5cb80b13f0af4de4c3b89f52dee5" +dependencies = [ + "parse-display-derive", + "regex", + "regex-syntax", +] + +[[package]] +name = "parse-display-derive" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc9252f259500ee570c75adcc4e317fa6f57a1e47747d622e0bf838002a7b790" +dependencies = [ + "proc-macro2", + "quote", + "regex", + "regex-syntax", + "structmeta", + "syn 2.0.61", +] + [[package]] name = "paste" version = "1.0.15" @@ -3075,11 +3306,13 @@ dependencies = [ "http-body-util", "hyper 1.3.1", "hyper-rustls 0.27.2", + "hyper-tls", "hyper-util", "ipnet", "js-sys", "log", "mime", + "native-tls", "once_cell", "percent-encoding", "pin-project-lite", @@ -3092,6 +3325,7 @@ dependencies = [ "serde_urlencoded", "sync_wrapper 1.0.1", "tokio", + "tokio-native-tls", "tokio-rustls 0.26.0", "tokio-util", "tower-service", @@ -3441,6 +3675,17 @@ dependencies = [ "serde", ] +[[package]] +name = "serde_repr" +version = "0.1.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c64451ba24fc7a6a2d60fc75dd9c83c90903b19028d4eff35e88fc1e86564e9" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.61", +] + [[package]] name = "serde_spanned" version = "0.6.5" @@ -3462,13 +3707,43 @@ dependencies = [ "serde", ] +[[package]] +name = "serde_with" +version = "3.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ad483d2ab0149d5a5ebcd9972a3852711e0153d863bf5a5d0391d28883c4a20" +dependencies = [ + "base64 0.22.1", + "chrono", + "hex", + "indexmap 1.9.3", + "indexmap 2.2.6", + "serde", + "serde_derive", + "serde_json", + "serde_with_macros", + "time", +] + +[[package]] +name = "serde_with_macros" +version = "3.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "65569b702f41443e8bc8bbb1c5779bd0450bbe723b56198980e80ec45780bce2" +dependencies = [ + "darling", + "proc-macro2", + "quote", + "syn 2.0.61", +] + [[package]] name = "serde_yaml" version = "0.9.34+deprecated" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6a8b1a1a2ebf674015cc02edccce75287f1a0130d394307b36743c2f5d504b47" dependencies = [ - "indexmap", + "indexmap 2.2.6", "itoa", "ryu", "serde", @@ -3692,7 +3967,7 @@ dependencies = [ "futures-util", "hashlink", "hex", - "indexmap", + "indexmap 2.2.6", "log", "memchr", "once_cell", @@ -4087,6 +4362,43 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "testcontainers" +version = "0.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2c9b71635ab25d4f789a86678d114a4f390467c8b93fd7feeaf7c443732a511" +dependencies = [ + "async-trait", + "bollard", + "bollard-stubs", + "bytes", + "dirs", + "docker_credential", + "either", + "futures", + "log", + "memchr", + "parse-display", + "reqwest 0.12.5", + "serde", + "serde_json", + "serde_with", + "thiserror", + "tokio", + "tokio-stream", + "tokio-util", + "url", +] + +[[package]] +name = "testcontainers-modules" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "322fe829905e734f38a73be07e200a25713fe4baa65074277af7b30dbc4ea487" +dependencies = [ + "testcontainers", +] + [[package]] name = "thiserror" version = "1.0.60" @@ -4194,6 +4506,16 @@ dependencies = [ "syn 2.0.61", ] +[[package]] +name = "tokio-native-tls" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbae76ab933c85776efabc971569dd6119c580d8f5d448769dec1764bf796ef2" +dependencies = [ + "native-tls", + "tokio", +] + [[package]] name = "tokio-rustls" version = "0.24.1" @@ -4277,7 +4599,7 @@ version = "0.22.12" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d3328d4f68a705b2a4498da1d580585d39a6510f98318a2cec3018a7ec61ddef" dependencies = [ - "indexmap", + "indexmap 2.2.6", "serde", "serde_spanned", "toml_datetime", diff --git a/server/main-api/Cargo.toml b/server/main-api/Cargo.toml index 375fda053..ed8d9b2b3 100644 --- a/server/main-api/Cargo.toml +++ b/server/main-api/Cargo.toml @@ -69,6 +69,8 @@ time = "0.3.36" [dev-dependencies] pretty_assertions = "1.4.0" +testcontainers = "0.18.0" +testcontainers-modules = {version = "0.6.1",features = ["postgres"] } [features] skip_db_setup = [] diff --git a/server/main-api/migrations/20240505224430_fix-calendar-id-not-extracted.up.sql b/server/main-api/migrations/20240505224430_fix-calendar-id-not-extracted.up.sql index 9d37224d0..826ee3122 100644 --- a/server/main-api/migrations/20240505224430_fix-calendar-id-not-extracted.up.sql +++ b/server/main-api/migrations/20240505224430_fix-calendar-id-not-extracted.up.sql @@ -1,5 +1,4 @@ -- Add up migration script here --- was never used DROP TABLE rooms; -- migrating to using the json type instead of having elaborate insertion logic diff --git a/server/main-api/src/main.rs b/server/main-api/src/main.rs index f8db7c867..bafc6abf1 100644 --- a/server/main-api/src/main.rs +++ b/server/main-api/src/main.rs @@ -5,6 +5,7 @@ use actix_cors::Cors; use actix_web::{get, middleware, web, App, HttpResponse, HttpServer}; use actix_web_prom::PrometheusMetricsBuilder; use log::{debug, error, info}; +use meilisearch_sdk::client::Client; use sqlx::postgres::PgPoolOptions; use sqlx::prelude::*; use sqlx::PgPool; @@ -54,23 +55,25 @@ fn connection_string() -> String { format!("postgres://{username}:{password}@{url}/{db}") } -fn setup_logging() { - #[cfg(debug_assertions)] +pub fn setup_logging() { + #[cfg(any(debug_assertions, test))] { let log_level = std::env::var("LOG_LEVEL").unwrap_or_else(|_| "debug".to_string()); let filter = format!("{log_level},hyper=info,rustls=info,h2=info,sqlx=info,hickory_resolver=info,hickory_proto=info"); let env = env_logger::Env::default().default_filter_or(&filter); - env_logger::Builder::from_env(env).init(); + let _ = env_logger::Builder::from_env(env) + .is_test(cfg!(test)) + .try_init(); } - #[cfg(not(debug_assertions))] + #[cfg(not(any(debug_assertions, test)))] { let log_level = std::env::var("LOG_LEVEL").unwrap_or_else(|_| "info".to_string()); - structured_logger::Builder::with_level(&log_level) + let _ = structured_logger::Builder::with_level(&log_level) .with_target_writer( "*", structured_logger::async_json::new_writer(tokio::io::stdout()), ) - .init(); + .try_init(); } } @@ -83,9 +86,18 @@ async fn main() -> Result<(), BoxedError> { .await .unwrap(); #[cfg(not(feature = "skip_db_setup"))] - setup::database::setup(&pool).await.unwrap(); + { + setup::database::setup(&pool).await.unwrap(); + setup::database::load_data(&pool).await.unwrap(); + } #[cfg(not(feature = "skip_ms_setup"))] - setup::meilisearch::setup().await.unwrap(); + { + 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(); + setup::meilisearch::setup(&client).await.unwrap(); + setup::meilisearch::load_data(&client).await.unwrap(); + } calendar::refresh::all_entries(&pool).await; }); @@ -102,6 +114,7 @@ async fn main() -> Result<(), BoxedError> { info!("running the server"); let pool = PgPoolOptions::new().connect(&connection_string()).await?; + let shutdown_pool_clone = pool.clone(); HttpServer::new(move || { let cors = Cors::default() .allow_any_origin() @@ -127,5 +140,6 @@ async fn main() -> Result<(), BoxedError> { .run() .await?; maintenance_thread.abort(); + shutdown_pool_clone.close().await; Ok(()) } diff --git a/server/main-api/src/setup/database/mod.rs b/server/main-api/src/setup/database/mod.rs index 83f411322..f1c1681cc 100644 --- a/server/main-api/src/setup/database/mod.rs +++ b/server/main-api/src/setup/database/mod.rs @@ -7,25 +7,26 @@ pub async fn setup(pool: &sqlx::PgPool) -> Result<(), crate::BoxedError> { info!("setting up the database"); sqlx::migrate!("./migrations").run(pool).await?; info!("migrations complete"); - + Ok(()) +} +pub async fn load_data(pool: &sqlx::PgPool) -> Result<(), crate::BoxedError> { let mut tx = pool.begin().await?; - load_data(&mut tx).await?; + + info!("deleting old data"); + cleanup(&mut tx).await?; + info!("loading new data"); + data::load_all_to_db(&mut tx).await?; + info!("loading new aliases"); + alias::load_all_to_db(&mut tx).await?; tx.commit().await?; Ok(()) } -async fn load_data( - tx: &mut sqlx::Transaction<'_, sqlx::Postgres>, -) -> Result<(), crate::BoxedError> { - info!("deleting old data"); + +async fn cleanup(tx: &mut sqlx::Transaction<'_, sqlx::Postgres>) -> Result<(), crate::BoxedError> { sqlx::query!("DELETE FROM aliases") .execute(&mut **tx) .await?; sqlx::query!("DELETE FROM en").execute(&mut **tx).await?; sqlx::query!("DELETE FROM de").execute(&mut **tx).await?; - - info!("loading new data"); - data::load_all_to_db(tx).await?; - info!("loading new aliases"); - alias::load_all_to_db(tx).await?; Ok(()) } diff --git a/server/main-api/src/setup/meilisearch.rs b/server/main-api/src/setup/meilisearch.rs index f1ba4e114..a55a6a90b 100644 --- a/server/main-api/src/setup/meilisearch.rs +++ b/server/main-api/src/setup/meilisearch.rs @@ -1,7 +1,7 @@ use std::collections::HashMap; use std::time::Duration; -use log::{error, info}; +use log::{debug, error, info}; use meilisearch_sdk::client::Client; use meilisearch_sdk::settings::Settings; use meilisearch_sdk::tasks::Task; @@ -44,20 +44,15 @@ async fn wait_for_healthy(client: &Client) { } } -pub async fn setup() -> Result<(), crate::BoxedError> { - info!("setting up meilisearch"); - let start = std::time::Instant::now(); - let ms_url = std::env::var("MIELI_URL").unwrap_or_else(|_| "http://localhost:7700".to_string()); - info!("connecting to Meilisearch at {ms_url}", ms_url = ms_url); - let client = Client::new(ms_url, std::env::var("MEILI_MASTER_KEY").ok())?; - info!("waiting for Meilisearch to be healthy"); - wait_for_healthy(&client).await; +pub async fn setup(client: &Client) -> Result<(), crate::BoxedError> { + debug!("waiting for Meilisearch to be healthy"); + wait_for_healthy(client).await; info!("Meilisearch is healthy"); client .create_index("entries", Some("ms_id")) .await? - .wait_for_completion(&client, POLLING_RATE, TIMEOUT) + .wait_for_completion(client, POLLING_RATE, TIMEOUT) .await?; let entries = client.index("entries"); @@ -97,12 +92,17 @@ pub async fn setup() -> Result<(), crate::BoxedError> { let res = entries .set_settings(&settings) .await? - .wait_for_completion(&client, POLLING_RATE, TIMEOUT) + .wait_for_completion(client, POLLING_RATE, TIMEOUT) .await?; if let Task::Failed { content } = res { - panic!("Failed to add documents to Meilisearch: {content:#?}"); + panic!("Failed to add settings to Meilisearch: {content:#?}"); } + Ok(()) +} +pub async fn load_data(client: &Client) -> Result<(), crate::BoxedError> { + let start = std::time::Instant::now(); + let entries = client.index("entries"); let cdn_url = std::env::var("CDN_URL").unwrap_or_else(|_| "https://nav.tum.de/cdn".to_string()); let documents = reqwest::get(format!("{cdn_url}/search_data.json")) .await? @@ -111,7 +111,7 @@ pub async fn setup() -> Result<(), crate::BoxedError> { let res = entries .add_documents(&documents, Some("ms_id")) .await? - .wait_for_completion(&client, POLLING_RATE, TIMEOUT) + .wait_for_completion(client, POLLING_RATE, TIMEOUT) .await?; if let Task::Failed { content } = res { panic!("Failed to add documents to Meilisearch: {content:#?}"); diff --git a/server/main-api/src/setup/mod.rs b/server/main-api/src/setup/mod.rs index 80c6dc003..e258e3ba2 100644 --- a/server/main-api/src/setup/mod.rs +++ b/server/main-api/src/setup/mod.rs @@ -3,3 +3,5 @@ pub mod database; #[cfg(not(feature = "skip_ms_setup"))] pub mod meilisearch; +#[cfg(test)] +pub mod tests; diff --git a/server/main-api/src/setup/tests.rs b/server/main-api/src/setup/tests.rs new file mode 100644 index 000000000..48eef0374 --- /dev/null +++ b/server/main-api/src/setup/tests.rs @@ -0,0 +1,89 @@ +use meilisearch_sdk::client::Client; +use sqlx::postgres::PgPoolOptions; +use sqlx::{Pool, Postgres}; +use testcontainers::core::{ContainerPort, WaitFor}; +use testcontainers::{ContainerAsync, GenericImage, ImageExt}; +use testcontainers_modules::{postgres, testcontainers::runners::AsyncRunner}; + +#[cfg(not(feature = "skip_db_setup"))] +pub struct PostgresTestContainer { + _container: ContainerAsync, + pub pool: Pool, +} + +#[cfg(not(feature = "skip_db_setup"))] +impl PostgresTestContainer { + /// Create a postgres instance for testing against + pub async fn new() -> Self { + let container = postgres::Postgres::default() + .with_tag("16") + .start() + .await + .unwrap(); + let connection_string = format!( + "postgres://postgres:postgres@{host}:{port}/postgres", + host = container.get_host().await.unwrap(), + port = container.get_host_port_ipv4(5432).await.unwrap(), + ); + let pool = PgPoolOptions::new() + .connect(&connection_string) + .await + .unwrap(); + crate::setup::database::setup(&pool).await.unwrap(); + Self { + _container: container, + pool, + } + } +} + +#[cfg(not(feature = "skip_ms_setup"))] +pub struct MeiliSearchTestContainer { + _container: ContainerAsync, + pub client: Client, +} + +#[cfg(not(feature = "skip_ms_setup"))] +impl MeiliSearchTestContainer { + /// Create a meilisearch instance for testing against + pub async fn new() -> Self { + let container = GenericImage::new("getmeili/meilisearch", "v1.8.0") + .with_exposed_port(ContainerPort::Tcp(7700)) + .with_wait_for(WaitFor::message_on_stderr( + "Actix runtime found; starting in Actix runtime", + )) + .start() + .await + .unwrap(); + let meili_url = format!( + "http://{host}:{port}", + host = container.get_host().await.unwrap(), + port = container.get_host_port_ipv4(7700).await.unwrap(), + ); + + let client = Client::new(meili_url.clone(), None::).unwrap(); + super::meilisearch::setup(&client).await.unwrap(); + Self { + _container: container, + client, + } + } +} + +#[tokio::test] +#[cfg(not(feature = "skip_db_setup"))] +async fn test_db_setup() { + crate::setup_logging(); + let pg = PostgresTestContainer::new().await; + crate::setup::database::load_data(&pg.pool).await.unwrap(); +} + +#[tokio::test] +#[cfg(not(feature = "skip_ms_setup"))] +async fn test_meilisearch_setup() { + crate::setup_logging(); + let ms = MeiliSearchTestContainer::new().await; + crate::setup::meilisearch::load_data(&ms.client) + .await + .unwrap(); +}