about summary refs log tree commit diff
path: root/stats_module
diff options
context:
space:
mode:
authorgennyble <gen@nyble.dev>2025-03-27 05:11:46 -0500
committergennyble <gen@nyble.dev>2025-03-27 05:11:46 -0500
commitb79b84ce06ee34b5957d7f19aa19ebeff2af1df9 (patch)
tree5e725e3b8da40493fbb681e56cf14fdb54bac57d /stats_module
parent362bc937cee716ae0af77eb467756e420f9324a5 (diff)
downloadcorgi-b79b84ce06ee34b5957d7f19aa19ebeff2af1df9.tar.gz
corgi-b79b84ce06ee34b5957d7f19aa19ebeff2af1df9.zip
add smalldog as the module ffi crate
Diffstat (limited to 'stats_module')
-rw-r--r--stats_module/Cargo.toml1
-rw-r--r--stats_module/src/lib.rs99
2 files changed, 23 insertions, 77 deletions
diff --git a/stats_module/Cargo.toml b/stats_module/Cargo.toml
index ea0f965..95b6d41 100644
--- a/stats_module/Cargo.toml
+++ b/stats_module/Cargo.toml
@@ -10,3 +10,4 @@ crate-type = ["cdylib"]
 [dependencies]
 rusqlite = { version = "0.34.0", features = ["bundled", "time"] }
 time = "0.3.40"
+smalldog = { path = "../smalldog" }
diff --git a/stats_module/src/lib.rs b/stats_module/src/lib.rs
index 18c0e63..96cbaa9 100644
--- a/stats_module/src/lib.rs
+++ b/stats_module/src/lib.rs
@@ -1,55 +1,20 @@
-use std::{
-	ffi::{self, CStr, CString},
-	sync::Mutex,
-};
+use std::ffi::CStr;
 
 use rusqlite::{Connection, params};
+use smalldog::{ModuleRequest, ModuleResponse, Request, Response};
 use time::{Duration, OffsetDateTime};
 
-#[repr(C)]
-struct ModuleRequest<'req> {
-	headers_len: ffi::c_ulong,
-	headers: &'req [[*const ffi::c_char; 2]],
-	body_len: ffi::c_ulong,
-	body: *const u8,
-}
-
-#[repr(C)]
-struct ModuleResponse {
-	status: ffi::c_ushort,
-	headers_len: ffi::c_ulong,
-	headers: &'static [[*const ffi::c_char; 2]],
-	body_len: ffi::c_ulong,
-	body: *const u8,
-}
-
-const HEADERS: &'static [[*const ffi::c_char; 2]] =
-	&[[c"Content-Type".as_ptr(), c"text/html".as_ptr()]];
-
 #[unsafe(no_mangle)]
 extern "C" fn cgi_handle(req: *const ModuleRequest) -> *const ModuleResponse {
-	let mut ret = String::new();
-
-	// unwrap is bad here
-	let reqref = unsafe { req.as_ref() }.unwrap();
-
-	let mut stats_path = None;
-	for idx in 0..reqref.headers_len {
-		let kvarr = reqref.headers[idx as usize];
-		let k = unsafe { CStr::from_ptr(kvarr[0]) }.to_string_lossy();
-		let v = unsafe { CStr::from_ptr(kvarr[1]) }.to_string_lossy();
-
-		match k.as_ref() {
-			"CORGI_STATS_DB" => stats_path = Some(v.into_owned()),
-			_ => (),
-		}
-	}
+	let mut response = Response::new();
 
-	let db = if let Some(path) = stats_path {
+	let request = Request::from_mod_request(req);
+	let db = if let Some(path) = request.header("CORGI_STATS_DB") {
 		Connection::open(path).unwrap()
 	} else {
-		return make_error(500, c"could not open stats database");
+		return make_error(500, "could not open stats database");
 	};
+
 	let now = OffsetDateTime::now_utc();
 	let fifteen_ago = now - Duration::minutes(15);
 
@@ -67,52 +32,32 @@ extern "C" fn cgi_handle(req: *const ModuleRequest) -> *const ModuleResponse {
 
 	agents.sort_by(|a, b| a.0.cmp(&b.0).reverse());
 
-	ret.push_str("<p>In the last fifteen minutes:<br/><code><pre>");
-	ret.push_str("total | req/m | agent\n");
+	response.push_str("<p>In the last fifteen minutes:<br/><code><pre>");
+	response.push_str("total | req/m | agent\n");
 	for (count, agent) in &agents {
-		ret.push_str(&format!(
+		response.push_str(&format!(
 			"{count:<5} | {:<5.1} | {agent}\n",
 			*count as f32 / 15.0
 		));
 	}
-	ret.push_str("</pre></code></p>");
-
-	let body = CString::new(ret).unwrap();
-
-	let resp = ModuleResponse {
-		status: 200,
-		headers_len: 1,
-		headers: HEADERS,
-		body_len: body.as_bytes().len() as u64,
-		body: body.into_raw() as *const u8,
-	};
+	response.push_str("</pre></code></p>");
 
-	let boxed = Box::new(resp);
-	Box::<ModuleResponse>::into_raw(boxed)
+	response.into_mod_response(200)
 }
 
-fn make_error<S: Into<CString>>(code: u16, msg: S) -> *const ModuleResponse {
-	let body = msg.into();
+fn make_error<S: AsRef<str>>(code: u16, msg: S) -> *const ModuleResponse {
+	unsafe {
+		smalldog::HEADERS[0][0] = c"Content-Length".as_ptr();
+		smalldog::HEADERS[0][1] = c"text/html".as_ptr();
+	}
 
-	let resp = ModuleResponse {
-		status: code,
-		headers_len: 1,
-		headers: HEADERS,
-		body_len: body.as_bytes().len() as u64,
-		body: body.into_raw() as *const u8,
-	};
+	let mut response = Response::new();
+	response.push_str(msg.as_ref());
 
-	let boxed = Box::new(resp);
-	Box::<ModuleResponse>::into_raw(boxed)
+	response.into_mod_response(code)
 }
 
 #[unsafe(no_mangle)]
-extern "C" fn cgi_cleanup(req: *const ModuleResponse) {
-	// from_raw what we need to here so that these get dropped
-	let boxed = unsafe { Box::from_raw(req as *mut ModuleResponse) };
-	let body = unsafe { CString::from_raw(boxed.body as *mut i8) };
-
-	// Explicitly calling drop here to feel good about myself
-	drop(body);
-	drop(boxed);
+extern "C" fn cgi_cleanup(response: *const ModuleResponse) {
+	Response::cleanup(response);
 }