rust-analyzer/xtask/src/metrics.rs

243 lines
6.2 KiB
Rust
Raw Normal View History

2020-07-24 14:28:07 +00:00
use std::{
collections::BTreeMap,
env,
fmt::{self, Write as _},
io::Write as _,
time::{Instant, SystemTime, UNIX_EPOCH},
};
use anyhow::{bail, format_err, Result};
use crate::not_bash::{fs2, pushd, rm_rf, run};
type Unit = &'static str;
2020-07-24 22:16:21 +00:00
pub struct MetricsCmd {
pub dry_run: bool,
}
impl MetricsCmd {
pub fn run(self) -> Result<()> {
let mut metrics = Metrics::new()?;
if !self.dry_run {
rm_rf("./target/release")?;
}
metrics.measure_build()?;
metrics.measure_analysis_stats_self()?;
2020-07-24 14:28:07 +00:00
2020-07-24 22:16:21 +00:00
if !self.dry_run {
let _d = pushd("target");
let metrics_token = env::var("METRICS_TOKEN").unwrap();
let repo = format!("https://{}@github.com/rust-analyzer/metrics.git", metrics_token);
run!("git clone --depth 1 {}", repo)?;
let _d = pushd("metrics");
2020-07-24 18:53:08 +00:00
2020-07-24 22:16:21 +00:00
let mut file = std::fs::OpenOptions::new().append(true).open("metrics.json")?;
writeln!(file, "{}", metrics.json())?;
run!("git add .")?;
run!("git -c user.name=Bot -c user.email=dummy@example.com commit --message 📈")?;
run!("git push origin master")?;
}
eprintln!("{:#?}", metrics);
Ok(())
2020-07-24 14:28:07 +00:00
}
}
impl Metrics {
fn measure_build(&mut self) -> Result<()> {
run!("cargo fetch")?;
2020-07-24 22:16:21 +00:00
let time = Instant::now();
2020-07-24 19:07:58 +00:00
run!("cargo build --release --package rust-analyzer --bin rust-analyzer")?;
2020-07-24 22:16:21 +00:00
let time = time.elapsed();
self.report("build", time.as_millis() as u64, "ms");
Ok(())
}
fn measure_analysis_stats_self(&mut self) -> Result<()> {
let time = Instant::now();
run!("./target/release/rust-analyzer analysis-stats .")?;
let time = time.elapsed();
self.report("analysis-stats/self", time.as_millis() as u64, "ms");
2020-07-24 14:28:07 +00:00
Ok(())
}
}
#[derive(Debug)]
struct Metrics {
host: Host,
timestamp: SystemTime,
revision: String,
metrics: BTreeMap<String, (u64, Unit)>,
}
#[derive(Debug)]
struct Host {
os: String,
cpu: String,
mem: String,
}
impl Metrics {
fn new() -> Result<Metrics> {
let host = Host::new()?;
let timestamp = SystemTime::now();
let revision = run!("git rev-parse HEAD")?;
Ok(Metrics { host, timestamp, revision, metrics: BTreeMap::new() })
}
fn report(&mut self, name: &str, value: u64, unit: Unit) {
self.metrics.insert(name.into(), (value, unit));
}
fn json(&self) -> Json {
let mut json = Json::default();
self.to_json(&mut json);
json
}
fn to_json(&self, json: &mut Json) {
json.begin_object();
{
json.field("host");
self.host.to_json(json);
json.field("timestamp");
let timestamp = self.timestamp.duration_since(UNIX_EPOCH).unwrap();
json.number(timestamp.as_secs() as f64);
json.field("revision");
json.string(&self.revision);
json.field("metrics");
json.begin_object();
{
for (k, &(value, unit)) in &self.metrics {
json.field(k);
json.begin_array();
{
json.number(value as f64);
json.string(unit);
}
json.end_array();
}
}
json.end_object()
}
json.end_object();
}
}
impl Host {
fn new() -> Result<Host> {
if cfg!(not(target_os = "linux")) {
bail!("can only collect metrics on Linux ");
}
let os = read_field("/etc/os-release", "PRETTY_NAME=")?.trim_matches('"').to_string();
let cpu =
read_field("/proc/cpuinfo", "model name")?.trim_start_matches(':').trim().to_string();
let mem = read_field("/proc/meminfo", "MemTotal:")?;
return Ok(Host { os, cpu, mem });
fn read_field<'a>(path: &str, field: &str) -> Result<String> {
let text = fs2::read_to_string(path)?;
let line = text
.lines()
.find(|it| it.starts_with(field))
.ok_or_else(|| format_err!("can't parse {}", path))?;
Ok(line[field.len()..].trim().to_string())
}
}
fn to_json(&self, json: &mut Json) {
json.begin_object();
{
json.field("os");
json.string(&self.os);
json.field("cpu");
json.string(&self.cpu);
json.field("mem");
json.string(&self.mem);
}
json.end_object();
}
}
2020-07-24 19:03:49 +00:00
struct State {
obj: bool,
first: bool,
}
2020-07-24 14:28:07 +00:00
#[derive(Default)]
struct Json {
2020-07-24 19:03:49 +00:00
stack: Vec<State>,
2020-07-24 14:28:07 +00:00
buf: String,
}
impl Json {
fn begin_object(&mut self) {
2020-07-24 19:03:49 +00:00
self.stack.push(State { obj: true, first: true });
2020-07-24 14:28:07 +00:00
self.buf.push('{');
}
fn end_object(&mut self) {
2020-07-24 19:03:49 +00:00
self.stack.pop();
2020-07-24 14:28:07 +00:00
self.buf.push('}')
}
fn begin_array(&mut self) {
2020-07-24 19:03:49 +00:00
self.stack.push(State { obj: false, first: true });
2020-07-24 14:28:07 +00:00
self.buf.push('[');
}
fn end_array(&mut self) {
2020-07-24 19:03:49 +00:00
self.stack.pop();
2020-07-24 14:28:07 +00:00
self.buf.push(']')
}
fn field(&mut self, name: &str) {
self.object_comma();
self.string_token(name);
self.buf.push(':');
}
fn string(&mut self, value: &str) {
self.array_comma();
self.string_token(value);
}
fn string_token(&mut self, value: &str) {
self.buf.push('"');
self.buf.extend(value.escape_default());
self.buf.push('"');
}
fn number(&mut self, value: f64) {
self.array_comma();
write!(self.buf, "{}", value).unwrap();
}
fn array_comma(&mut self) {
2020-07-24 19:03:49 +00:00
let state = self.stack.last_mut().unwrap();
if state.obj {
return;
}
if !state.first {
2020-07-24 14:28:07 +00:00
self.buf.push(',');
}
2020-07-24 19:03:49 +00:00
state.first = false;
2020-07-24 14:28:07 +00:00
}
fn object_comma(&mut self) {
2020-07-24 19:03:49 +00:00
let state = self.stack.last_mut().unwrap();
if !state.first {
2020-07-24 14:28:07 +00:00
self.buf.push(',');
}
2020-07-24 19:03:49 +00:00
state.first = false;
2020-07-24 14:28:07 +00:00
}
}
impl fmt::Display for Json {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.buf)
}
}