1
0

Compare commits

5 Commits

Author SHA1 Message Date
0c468b4953 Export booted system and kernel hash as well 2023-03-23 20:55:02 +01:00
7b674a8262 Beautify error handling 2023-03-23 18:53:57 +01:00
19120d4d0a Rename prometheus valid metric 2023-02-04 00:32:26 +01:00
b89910f649 Fix prometheus query 2023-02-04 00:03:15 +01:00
c0cb8db954 Fix validation for exporter mode 2023-01-11 21:32:23 +01:00
4 changed files with 58 additions and 94 deletions

2
Cargo.lock generated
View File

@@ -450,7 +450,7 @@ dependencies = [
[[package]] [[package]]
name = "nixos-exporter" name = "nixos-exporter"
version = "0.1.0" version = "0.3.0"
dependencies = [ dependencies = [
"axum", "axum",
"reqwest", "reqwest",

View File

@@ -1,6 +1,6 @@
[package] [package]
name = "nixos-exporter" name = "nixos-exporter"
version = "0.1.0" version = "0.3.0"
edition = "2021" edition = "2021"
[dependencies] [dependencies]

View File

@@ -10,7 +10,7 @@
in { in {
nixos-exporter = pkgs.rustPlatform.buildRustPackage rec { nixos-exporter = pkgs.rustPlatform.buildRustPackage rec {
pname = "nixos-exporter"; pname = "nixos-exporter";
version = "0.1.0"; version = "0.3.0";
src = ./.; src = ./.;

View File

@@ -1,7 +1,7 @@
use axum::{ use axum::{
extract::{State, Query}, extract::{State, Query},
http::StatusCode, http::StatusCode,
response::{IntoResponse, Response}, response::IntoResponse,
routing::get, routing::get,
Router, Router,
}; };
@@ -45,11 +45,11 @@ impl AppState{
println!("operationmode is not set"); println!("operationmode is not set");
valid = false; valid = false;
} }
if self.prometheus_url == String::from("") { if self.prometheus_url == String::from("") && self.operationmode == OperationMode::Validator {
println!("Prometheus url is not specified"); println!("Prometheus url is not specified");
valid = false; valid = false;
} }
if self.hydra_url == String::from("") { if self.hydra_url == String::from("") && self.operationmode == OperationMode::Validator {
println!("Hydra url is not specified"); println!("Hydra url is not specified");
valid = false; valid = false;
} }
@@ -59,12 +59,12 @@ impl AppState{
} }
fn parse_nix_store_path(path: std::path::PathBuf) -> Result<(String, String), String> { fn parse_nix_store_path(path: std::path::PathBuf) -> Result<(String, String), String> {
let (hash, name) = path.file_name() let (hash, name) = path.iter().nth(3)
.ok_or_else(String::default)? .ok_or_else(|| String::from("Can't read store path name"))?
.to_str() .to_str()
.ok_or_else(String::default)? .ok_or_else(|| String::from("Failed converting store path name to string"))?
.split_once("-") .split_once("-")
.ok_or_else(String::default)?; .ok_or_else(|| String::from("Failed splitting store path name for hash and name"))?;
return Ok((hash.to_string(), name.to_string())); return Ok((hash.to_string(), name.to_string()));
} }
@@ -143,114 +143,78 @@ async fn main() {
.unwrap(); .unwrap();
} }
fn get_current_system() -> Result<(String, String), String> { fn parse_symlink(path: String) -> Result<(String, String), String> {
let symlink = match std::fs::read_link("/run/current-system") { let symlink = std::fs::read_link(path).map_err(|err| err.to_string())?;
Ok(symlink) => symlink,
Err(err) => return Err(err.to_string()),
};
let (hash, name) = parse_nix_store_path(symlink)?; let (hash, name) = parse_nix_store_path(symlink)?;
Ok((String::from(hash), String::from(name))) Ok((String::from(hash), String::from(name)))
} }
async fn metrics() -> Response { fn gen_prometheus_metric(path: String, infix: String) -> Result<String, String> {
let current_system = get_current_system(); let (hash, name) = parse_symlink(path).map_err(|err| err)?;
let (hash, name) = match current_system {
Ok((hash, name)) => (hash, name),
Err(err) => {
println!("failed: {}", err);
return (StatusCode::INTERNAL_SERVER_ERROR, "").into_response();
}
};
( return Ok(format!("nixos_{}_hash{{hash=\"{}\"}} 1\nnixos_{}_name{{name=\"{}\"}} 1\n", infix, hash, infix, name));
StatusCode::OK,
format!("nixos_current_system_hash{{hash=\"{}\"}} 1\nnixos_current_system_name{{name=\"{}\"}} 1\n", hash, name)
).into_response()
} }
async fn check(State(app_state): State<AppState>, Query(params): Query<HashMap<String, String>>) -> Response { async fn metrics() -> Result<(StatusCode, impl IntoResponse), (StatusCode, impl IntoResponse)> {
let target = match params.get("target") { let mut out = String::new();
Some(target) => target, out.push_str(&gen_prometheus_metric(String::from("/run/current-system"), String::from("current_system"))
None => { .map_err(|err| (StatusCode::INTERNAL_SERVER_ERROR, err))?);
return (StatusCode::NOT_FOUND, "specify target").into_response(); out.push_str(&gen_prometheus_metric(String::from("/run/current-system/kernel"), String::from("current_system_kernel"))
}, .map_err(|err| (StatusCode::INTERNAL_SERVER_ERROR, err))?);
}; out.push_str(&gen_prometheus_metric(String::from("/run/booted-system"), String::from("booted_system"))
.map_err(|err| (StatusCode::INTERNAL_SERVER_ERROR, err))?);
out.push_str(&gen_prometheus_metric(String::from("/run/booted-system/kernel"), String::from("booted_system_kernel"))
.map_err(|err| (StatusCode::INTERNAL_SERVER_ERROR, err))?);
return Ok((
StatusCode::OK,
out,
));
}
async fn check(State(app_state): State<AppState>, Query(params): Query<HashMap<String, String>>) -> Result<(StatusCode, impl IntoResponse), (StatusCode, impl IntoResponse)> {
let target = params.get("target")
.ok_or_else(|| (StatusCode::NOT_FOUND, "specify target"))?;
if target.contains("\"") { if target.contains("\"") {
return (StatusCode::INTERNAL_SERVER_ERROR, "Invalid target name").into_response(); return Err((StatusCode::INTERNAL_SERVER_ERROR, "Invalid target name"));
} }
let client = reqwest::Client::new(); let client = reqwest::Client::new();
let prometheus_req = match client.get(format!("{}/api/v1/query?query=nixos_nixos_current_system_hash{{{}}}", app_state.prometheus_url, app_state.prometheus_query_tag_template.clone().replace("{}", target))) let prometheus_req = client.get(format!("{}/api/v1/query?query=nixos_current_system_hash{{{}}}", app_state.prometheus_url, app_state.prometheus_query_tag_template.clone().replace("{}", target)))
.header("Accept", "application/json") .header("Accept", "application/json")
.send().await { .send().await
Ok(req) => req, .map_err(|_err| (StatusCode::INTERNAL_SERVER_ERROR, "Promehteus can't get reached"))?;
Err(_) => {
return (StatusCode::INTERNAL_SERVER_ERROR, "Promehteus can't get reached").into_response();
},
};
if prometheus_req.status() != reqwest::StatusCode::OK {
match prometheus_req.status() { return Err((StatusCode::NOT_FOUND, "Target does not exist in Hydra"));
reqwest::StatusCode::OK => (),
_ => {
return (StatusCode::NOT_FOUND, "Target does not exist in Hydra").into_response();
},
} }
let prometheus_body = match prometheus_req.json::<serde_json::Value>().await { let prometheus_body = prometheus_req.json::<serde_json::Value>().await
Ok(body) => body, .map_err(|_err| (StatusCode::INTERNAL_SERVER_ERROR, "Invalid response from Hydra"))?;
Err(_) => {
return (StatusCode::INTERNAL_SERVER_ERROR, "Invalid response from Hydra").into_response();
},
};
let current_system_hash = match prometheus_body["data"]["result"][0]["metric"]["hash"].as_str() { let current_system_hash = prometheus_body["data"]["result"][0]["metric"]["hash"].as_str()
Some(nix_store_path) => nix_store_path, .ok_or_else(|| (StatusCode::INTERNAL_SERVER_ERROR, "No buildoutput found in Hydra"))?;
_ => {
return (StatusCode::INTERNAL_SERVER_ERROR, "No buildoutput found in Hydra").into_response();
},
};
let hydra_req = match client.get(format!("{}/job/{}/latest", app_state.hydra_url, app_state.hydra_job_template.clone().replace("{}", target))) let hydra_req = client.get(format!("{}/job/{}/latest", app_state.hydra_url, app_state.hydra_job_template.clone().replace("{}", target)))
.header("Accept", "application/json") .header("Accept", "application/json")
.send().await { .send().await
Ok(req) => req, .map_err(|_err| (StatusCode::INTERNAL_SERVER_ERROR, "Hydra can't get reached"))?;
Err(_) => {
return (StatusCode::INTERNAL_SERVER_ERROR, "Hydra can't get reached").into_response();
},
};
match hydra_req.status() { if hydra_req.status() != reqwest::StatusCode::OK {
reqwest::StatusCode::OK => (), return Err((StatusCode::NOT_FOUND, "Target does not exist in Hydra"));
_ => {
return (StatusCode::NOT_FOUND, "Target does not exist in Hydra").into_response();
},
} }
let hydra_body = match hydra_req.json::<serde_json::Value>().await { let hydra_body = hydra_req.json::<serde_json::Value>().await
Ok(body) => body, .map_err(|_err| (StatusCode::INTERNAL_SERVER_ERROR, "Invalid response from Hydra"))?;
Err(_) => {
return (StatusCode::INTERNAL_SERVER_ERROR, "Invalid response from Hydra").into_response();
},
};
let nix_store_path = match hydra_body["buildoutputs"]["out"]["path"].as_str() { let nix_store_path = hydra_body["buildoutputs"]["out"]["path"].as_str()
Some(nix_store_path) => nix_store_path, .ok_or_else(|| (StatusCode::INTERNAL_SERVER_ERROR, "No buildoutput found in Hydra"))?;
_ => {
return (StatusCode::INTERNAL_SERVER_ERROR, "No buildoutput found in Hydra").into_response();
},
};
let (hydra_system_hash, _) = match parse_nix_store_path(std::path::PathBuf::from(nix_store_path)) { let (hydra_system_hash, _) = parse_nix_store_path(std::path::PathBuf::from(nix_store_path))
Ok((hash, name)) => (hash, name), .map_err(|_err| (StatusCode::INTERNAL_SERVER_ERROR, "Invalid store path returned by Hydra"))?;
Err(_) => {
return (StatusCode::INTERNAL_SERVER_ERROR, "Invalid store path returned by Hydra").into_response();
},
};
let mut status = "0"; let mut status = "0";
@@ -258,8 +222,8 @@ async fn check(State(app_state): State<AppState>, Query(params): Query<HashMap<S
status = "1"; status = "1";
} }
return ( return Ok((
StatusCode::OK, StatusCode::OK,
format!("nixos_current_system_valid{{{}}} {}\n", app_state.prometheus_query_tag_template.clone().replace("{}", target), status) format!("nixos_current_system_is_sync{{}} {}\n", status)
).into_response(); ));
} }