about summary refs log tree commit diff
path: root/stats_module/src/lib.rs
blob: a56d22d1ac04e480c9b61ca268ffd5dcc6dac9ff (plain)
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
use rusqlite::{Connection, params};
use smalldog::{ModuleRequest, ModuleResponse, Request, Response};
use time::{Duration, OffsetDateTime};

#[unsafe(no_mangle)]
extern "C" fn cgi_handle(req: *const ModuleRequest) -> *const ModuleResponse {
	let mut response = Response::new();
	let mut body = String::new();

	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, "could not open stats database");
	};

	let now = OffsetDateTime::now_utc();
	let fifteen_ago = now - Duration::minutes(15);

	let query = "SELECT count(requests.id) AS request_count, agents.agent FROM requests \
		INNER JOIN agents ON requests.agent_id = agents.id \
		WHERE requests.timestamp > ?1 \
		GROUP BY requests.agent_id;";

	let mut prepared = db.prepare(query).unwrap();
	let mut agents: Vec<(usize, String)> = prepared
		.query_map(params![fifteen_ago], |row| Ok((row.get(0)?, row.get(1)?)))
		.unwrap()
		.map(|r| r.unwrap())
		.collect();

	agents.sort_by(|a, b| a.0.cmp(&b.0).reverse());

	body.push_str("<p>In the last fifteen minutes:<br/><code><pre>");
	body.push_str("total | req/m | agent\n");
	for (count, agent) in &agents {
		body.push_str(&format!(
			"{count:<5} | {:<5.1} | {agent}\n",
			*count as f32 / 15.0
		));
	}
	body.push_str("</pre></code></p>");
	response.body(body.into_bytes());

	response.into_mod_response(200)
}

fn make_error<S: AsRef<str>>(code: u16, msg: S) -> *const ModuleResponse {
	let mut response = Response::new();
	response.header(c"Content-Length", c"text/html");
	response.body(msg.as_ref().as_bytes().to_vec());

	response.into_mod_response(code)
}

#[unsafe(no_mangle)]
extern "C" fn cgi_cleanup(response: *const ModuleResponse) {
	Response::cleanup(response);
}