You've already forked AstralRinth
forked from didirus/AstralRinth
* Update Java dependencies * Baselint lint fixes * Update Rust version * Update actix-files 0.6.6 -> 0.6.8 * Update actix-http 3.11.0 -> 3.11.2 * Update actix-rt 2.10.0 -> 2.11.0 * Update async_zip 0.0.17 -> 0.0.18 * Update async-compression 0.4.27 -> 0.4.32 * Update async-trait 0.1.88 -> 0.1.89 * Update async-tungstenite 0.30.0 -> 0.31.0 * Update const_format 0.2.34 -> 0.2.35 * Update bitflags 2.9.1 -> 2.9.4 * Update bytemuck 1.23.1 -> 1.24.0 * Update typed-path 0.11.0 -> 0.12.0 * Update chrono 0.4.41 -> 0.4.42 * Update cidre 0.11.2 -> 0.11.3 * Update clap 4.5.43 -> 4.5.48 * Update data-url 0.3.1 -> 0.3.2 * Update discord-rich-presence 0.2.5 -> 1.0.0 * Update enumset 1.1.7 -> 1.1.10 * Update flate2 1.1.2 -> 1.1.4 * Update hyper 1.6.0 -> 1.7.0 * Update hyper-util 0.1.16 -> 0.1.17 * Update iana-time-zone 0.1.63 -> 0.1.64 * Update image 0.25.6 -> 0.25.8 * Update indexmap 2.10.0 -> 2.11.4 * Update json-patch 4.0.0 -> 4.1.0 * Update meilisearch-sdk 0.29.1 -> 0.30.0 * Update clickhouse 0.13.3 -> 0.14.0 * Fix some prettier things * Update lettre 0.11.18 -> 0.11.19 * Update phf 0.12.1 -> 0.13.1 * Update png 0.17.16 -> 0.18.0 * Update quick-xml 0.38.1 -> 0.38.3 * Update redis 0.32.4 -> 0.32.7 * Update regex 1.11.1 -> 1.11.3 * Update reqwest 0.12.22 -> 0.12.23 * Update rust_decimal 1.37.2 -> 1.38.0 * Update rust-s3 0.35.1 -> 0.37.0 * Update serde 1.0.219 -> 1.0.228 * Update serde_bytes 0.11.17 -> 0.11.19 * Update serde_json 1.0.142 -> 1.0.145 * Update serde_with 3.14.0 -> 3.15.0 * Update sentry 0.42.0 -> 0.45.0 and sentry-actix 0.42.0 -> 0.45.0 * Update spdx 0.10.9 -> 0.12.0 * Update sysinfo 0.36.1 -> 0.37.2 * Update tauri 2.7.0 -> 2.8.5 * Update tauri-build 2.3.1 -> 2.4.1 * Update tauri-plugin-deep-link 2.4.1 -> 2.4.3 * Update tauri-plugin-dialog 2.3.2 -> 2.4.0 * Update tauri-plugin-http 2.5.1 -> 2.5.2 * Update tauri-plugin-opener 2.4.0 -> 2.5.0 * Update tauri-plugin-os 2.3.0 -> 2.3.1 * Update tauri-plugin-single-instance 2.3.2 -> 2.3.4 * Update tempfile 3.20.0 -> 3.23.0 * Update thiserror 2.0.12 -> 2.0.17 * Update tracing-subscriber 0.3.19 -> 0.3.20 * Update url 2.5.4 -> 2.5.7 * Update uuid 1.17.0 -> 1.18.1 * Update webp 0.3.0 -> 0.3.1 * Update whoami 1.6.0 -> 1.6.1 * Note that windows and windows-core can't be updated yet * Update zbus 5.9.0 -> 5.11.0 * Update zip 4.3.0 -> 6.0.0 * Fix build * Enforce rustls crypto provider * Refresh Cargo.lock * Update transitive dependencies * Bump Gradle usage to Java 17 * Use ubuntu-latest consistently across workflows * Fix lint * Fix lint in Rust * Update native-dialog 0.9.0 -> 0.9.2 * Update regex 1.11.3 -> 1.12.2 * Update reqwest 0.12.23 -> 0.12.24 * Update rust_decimal 1.38.0 -> 1.39.0 * Remaining lock-only updates * chore: move TLS impl of some other dependencies to aws-lc-rs The AWS bloatware "virus" expands by sheer force of widespread adoption by the ecosystem... 🫣 * chore(fmt): run Tombi --------- Co-authored-by: Alejandro González <me@alegon.dev>
310 lines
9.8 KiB
Rust
310 lines
9.8 KiB
Rust
use actix_web::dev::Service;
|
|
use actix_web::middleware::from_fn;
|
|
use actix_web::{App, HttpServer};
|
|
use actix_web_prom::PrometheusMetricsBuilder;
|
|
use clap::Parser;
|
|
|
|
use labrinth::app_config;
|
|
use labrinth::background_task::BackgroundTask;
|
|
use labrinth::database::redis::RedisPool;
|
|
use labrinth::file_hosting::{S3BucketConfig, S3Host};
|
|
use labrinth::queue::email::EmailQueue;
|
|
use labrinth::search;
|
|
use labrinth::util::anrok;
|
|
use labrinth::util::env::parse_var;
|
|
use labrinth::util::ratelimit::rate_limit_middleware;
|
|
use labrinth::{check_env_vars, clickhouse, database, file_hosting, queue};
|
|
use std::ffi::CStr;
|
|
use std::str::FromStr;
|
|
use std::sync::Arc;
|
|
use tracing::level_filters::LevelFilter;
|
|
use tracing::{Instrument, error, info, info_span};
|
|
use tracing_actix_web::TracingLogger;
|
|
use tracing_ecs::ECSLayerBuilder;
|
|
use tracing_subscriber::EnvFilter;
|
|
use tracing_subscriber::layer::SubscriberExt;
|
|
use tracing_subscriber::util::SubscriberInitExt;
|
|
|
|
#[cfg(target_os = "linux")]
|
|
#[global_allocator]
|
|
static ALLOC: tikv_jemallocator::Jemalloc = tikv_jemallocator::Jemalloc;
|
|
|
|
#[unsafe(export_name = "malloc_conf")]
|
|
pub static MALLOC_CONF: &CStr = c"prof:true,prof_active:true,lg_prof_sample:19";
|
|
|
|
#[derive(Clone)]
|
|
pub struct Pepper {
|
|
pub pepper: String,
|
|
}
|
|
|
|
#[derive(Parser)]
|
|
#[command(version)]
|
|
struct Args {
|
|
/// Don't run regularly scheduled background tasks. This means the tasks should be run
|
|
/// manually with --run-background-task.
|
|
#[arg(long)]
|
|
no_background_tasks: bool,
|
|
|
|
/// Don't automatically run migrations. This means the migrations should be run via --run-background-task.
|
|
#[arg(long)]
|
|
no_migrations: bool,
|
|
|
|
/// Run a single background task and then exit. Perfect for cron jobs.
|
|
#[arg(long, value_enum, id = "task")]
|
|
run_background_task: Option<BackgroundTask>,
|
|
}
|
|
|
|
#[derive(Debug, Clone, Default, PartialEq, Eq)]
|
|
enum OutputFormat {
|
|
#[default]
|
|
Human,
|
|
Json,
|
|
}
|
|
|
|
impl FromStr for OutputFormat {
|
|
type Err = ();
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
match s {
|
|
"human" => Ok(Self::Human),
|
|
"json" => Ok(Self::Json),
|
|
_ => Err(()),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[actix_rt::main]
|
|
async fn main() -> std::io::Result<()> {
|
|
let args = Args::parse();
|
|
|
|
color_eyre::install().expect("failed to install `color-eyre`");
|
|
dotenvy::dotenv().ok();
|
|
let console_layer = console_subscriber::spawn();
|
|
let env_filter = EnvFilter::builder()
|
|
.with_default_directive(LevelFilter::INFO.into())
|
|
.from_env_lossy();
|
|
|
|
let output_format =
|
|
dotenvy::var("LABRINTH_FORMAT").map_or(OutputFormat::Human, |format| {
|
|
format
|
|
.parse::<OutputFormat>()
|
|
.unwrap_or_else(|_| panic!("invalid output format '{format}'"))
|
|
});
|
|
|
|
match output_format {
|
|
OutputFormat::Human => {
|
|
tracing_subscriber::registry()
|
|
.with(console_layer)
|
|
.with(env_filter)
|
|
.with(tracing_subscriber::fmt::layer())
|
|
.init();
|
|
}
|
|
OutputFormat::Json => {
|
|
tracing_subscriber::registry()
|
|
.with(console_layer)
|
|
.with(env_filter)
|
|
.with(ECSLayerBuilder::default().stdout())
|
|
.init();
|
|
}
|
|
}
|
|
|
|
if check_env_vars() {
|
|
error!("Some environment variables are missing!");
|
|
std::process::exit(1);
|
|
}
|
|
|
|
rustls::crypto::aws_lc_rs::default_provider()
|
|
.install_default()
|
|
.unwrap();
|
|
|
|
// DSN is from SENTRY_DSN env variable.
|
|
// Has no effect if not set.
|
|
let sentry = sentry::init(sentry::ClientOptions {
|
|
release: sentry::release_name!(),
|
|
traces_sample_rate: 0.1,
|
|
..Default::default()
|
|
});
|
|
if sentry.is_enabled() {
|
|
info!("Enabled Sentry integration");
|
|
unsafe {
|
|
std::env::set_var("RUST_BACKTRACE", "1");
|
|
}
|
|
}
|
|
|
|
if args.run_background_task.is_none() {
|
|
info!(
|
|
"Starting labrinth on {}",
|
|
dotenvy::var("BIND_ADDR").unwrap()
|
|
);
|
|
|
|
if !args.no_migrations {
|
|
database::check_for_migrations()
|
|
.await
|
|
.expect("An error occurred while running migrations.");
|
|
}
|
|
}
|
|
|
|
// Database Connector
|
|
let (pool, ro_pool) = database::connect_all()
|
|
.await
|
|
.expect("Database connection failed");
|
|
|
|
// Redis connector
|
|
let redis_pool = RedisPool::new(None);
|
|
|
|
let storage_backend =
|
|
dotenvy::var("STORAGE_BACKEND").unwrap_or_else(|_| "local".to_string());
|
|
|
|
let file_host: Arc<dyn file_hosting::FileHost + Send + Sync> =
|
|
match storage_backend.as_str() {
|
|
"s3" => {
|
|
let config_from_env = |bucket_type| S3BucketConfig {
|
|
name: parse_var(&format!("S3_{bucket_type}_BUCKET_NAME"))
|
|
.unwrap(),
|
|
uses_path_style: parse_var(&format!(
|
|
"S3_{bucket_type}_USES_PATH_STYLE_BUCKET"
|
|
))
|
|
.unwrap(),
|
|
region: parse_var(&format!("S3_{bucket_type}_REGION"))
|
|
.unwrap(),
|
|
url: parse_var(&format!("S3_{bucket_type}_URL")).unwrap(),
|
|
access_token: parse_var(&format!(
|
|
"S3_{bucket_type}_ACCESS_TOKEN"
|
|
))
|
|
.unwrap(),
|
|
secret: parse_var(&format!("S3_{bucket_type}_SECRET"))
|
|
.unwrap(),
|
|
};
|
|
|
|
Arc::new(
|
|
S3Host::new(
|
|
config_from_env("PUBLIC"),
|
|
config_from_env("PRIVATE"),
|
|
)
|
|
.unwrap(),
|
|
)
|
|
}
|
|
"local" => Arc::new(file_hosting::MockHost::new()),
|
|
_ => panic!("Invalid storage backend specified. Aborting startup!"),
|
|
};
|
|
|
|
info!("Initializing clickhouse connection");
|
|
let mut clickhouse = clickhouse::init_client().await.unwrap();
|
|
|
|
let search_config = search::SearchConfig::new(None);
|
|
|
|
let stripe_client =
|
|
stripe::Client::new(dotenvy::var("STRIPE_API_KEY").unwrap());
|
|
|
|
let anrok_client = anrok::Client::from_env().unwrap();
|
|
let email_queue =
|
|
EmailQueue::init(pool.clone(), redis_pool.clone()).unwrap();
|
|
|
|
if let Some(task) = args.run_background_task {
|
|
info!("Running task {task:?} and exiting");
|
|
task.run(
|
|
pool,
|
|
redis_pool,
|
|
search_config,
|
|
clickhouse,
|
|
stripe_client,
|
|
anrok_client.clone(),
|
|
email_queue,
|
|
)
|
|
.await;
|
|
return Ok(());
|
|
}
|
|
|
|
let maxmind_reader =
|
|
Arc::new(queue::maxmind::MaxMindIndexer::new().await.unwrap());
|
|
|
|
let prometheus = PrometheusMetricsBuilder::new("labrinth")
|
|
.endpoint("/metrics")
|
|
.exclude_regex(r"^/api/v1/.*$")
|
|
.exclude_regex(r"^/maven/.*$")
|
|
.exclude("/_internal/launcher_socket")
|
|
.mask_unmatched_patterns("UNKNOWN")
|
|
.build()
|
|
.expect("Failed to create prometheus metrics middleware");
|
|
|
|
database::register_and_set_metrics(&pool, &prometheus.registry)
|
|
.await
|
|
.expect("Failed to register database metrics");
|
|
redis_pool
|
|
.register_and_set_metrics(&prometheus.registry)
|
|
.await
|
|
.expect("Failed to register redis metrics");
|
|
|
|
#[cfg(target_os = "linux")]
|
|
labrinth::routes::debug::jemalloc_memory_stats(&prometheus.registry)
|
|
.expect("Failed to register jemalloc metrics");
|
|
|
|
let labrinth_config = labrinth::app_setup(
|
|
pool.clone(),
|
|
ro_pool.clone(),
|
|
redis_pool.clone(),
|
|
search_config.clone(),
|
|
&mut clickhouse,
|
|
file_host.clone(),
|
|
maxmind_reader.clone(),
|
|
stripe_client,
|
|
anrok_client.clone(),
|
|
email_queue,
|
|
!args.no_background_tasks,
|
|
);
|
|
|
|
info!("Starting Actix HTTP server!");
|
|
|
|
HttpServer::new(move || {
|
|
App::new()
|
|
.wrap(TracingLogger::default())
|
|
.wrap_fn(|req, srv| {
|
|
// We capture the same fields as `tracing-actix-web`'s `RootSpanBuilder`.
|
|
// See `root_span!` macro.
|
|
let span = info_span!(
|
|
"HTTP request",
|
|
http.method = %req.method(),
|
|
http.client_ip = %req.connection_info().realip_remote_addr().unwrap_or(""),
|
|
http.user_agent = %req.headers().get("User-Agent").map_or("", |h| h.to_str().unwrap_or("")),
|
|
http.target = %req.uri().path_and_query().map_or("", |p| p.as_str()),
|
|
http.authenticated = %req.headers().get("Authorization").is_some()
|
|
);
|
|
|
|
let fut = srv.call(req);
|
|
async move {
|
|
fut.await.inspect(|resp| {
|
|
let _span = info_span!(
|
|
"HTTP response",
|
|
http.status = %resp.response().status().as_u16(),
|
|
).entered();
|
|
|
|
resp.response()
|
|
.error()
|
|
.inspect(|err| log_error(err));
|
|
})
|
|
}
|
|
.instrument(span)
|
|
})
|
|
.wrap(prometheus.clone())
|
|
.wrap(from_fn(rate_limit_middleware))
|
|
.wrap(actix_web::middleware::Compress::default())
|
|
.wrap(sentry_actix::Sentry::new())
|
|
.configure(|cfg| app_config(cfg, labrinth_config.clone()))
|
|
})
|
|
.bind(dotenvy::var("BIND_ADDR").unwrap())?
|
|
.run()
|
|
.await
|
|
}
|
|
|
|
fn log_error(err: &actix_web::Error) {
|
|
if err.as_response_error().status_code().is_client_error() {
|
|
tracing::debug!(
|
|
"Error encountered while processing the incoming HTTP request: {err}"
|
|
);
|
|
} else {
|
|
tracing::error!(
|
|
"Error encountered while processing the incoming HTTP request: {err}"
|
|
);
|
|
}
|
|
}
|