metrics-lib 1.0.0

High-performance Rust metrics library: sub-2ns counters, sub-1ns gauges, nanosecond timers, tumbling-window rate meters, async timing, adaptive sampling, and system health. Cross-platform with minimal dependencies.
Documentation
#!/usr/bin/env node
/*
  Bench Regression Check
  - Fetches the GitHub Pages benchmark feed and compares the last two runs
  - Fails (exit 1) if per-benchmark or geomean thresholds are exceeded
*/
const https = require('https');

const FEED_URL = process.env.BENCH_FEED_URL || 'https://jamesgober.github.io/metrics-lib/benchmark-data/data.js';

// Thresholds: same as docs/benchmarks/main.js
const THRESHOLDS = {
  geomean: 0.08, // 8%
  perBenchmark: {
    'counter.inc': 0.08,
    'counter.add': 0.08,
    'timer.record': 0.08,
    'timer.record_ns': 0.08,
    'timer.start_stop': 0.08,
    'rate.tick': 0.08,
    'rate.tick_n': 0.08,
    'rate.tick_n.concurrent(4)': 0.10,
    // noisy sub-ns
    'gauge.set': 0.18,
    'gauge.set_max': 0.18,
    'gauge.set_min': 0.18,
    'gauge.access': 0.18,
  },
};
// Core benches considered most stable on CI
const CORE = ['counter.inc','timer.record','timer.record_ns','rate.tick','rate.tick_n'];

function get(url){
  return new Promise((resolve, reject) => {
    https.get(url, (res) => {
      if(res.statusCode !== 200){ reject(new Error('HTTP '+res.statusCode)); return; }
      let data='';
      res.on('data', (c)=> data += c);
      res.on('end', ()=> resolve(data));
    }).on('error', reject);
  });
}

function extractJSON(text){
  // Try JSON directly
  try { return JSON.parse(text) } catch(_){ }
  // Try object assignment window.BENCHMARK_DATA = { ... };
  let i = text.indexOf('{'); let j = text.lastIndexOf('}');
  if(i>=0 && j>i){
    const objStr = text.slice(i, j+1);
    try { return JSON.parse(objStr) } catch(_){ }
  }
  // Try array entries
  i = text.indexOf('['); j = text.lastIndexOf(']');
  if(i>=0 && j>i){
    const arrStr = text.slice(i, j+1);
    try { return { entries: { Criterion: JSON.parse(arrStr) } } } catch(_){ }
  }
  return null;
}

function friendlyName(name){
  if(name === 'tick_n') return 'rate.tick_n';
  if(name === 'tick') return 'rate.tick';
  if(name === 'tick_n_concurrent_4_threads') return 'rate.tick_n.concurrent(4)';
  if(name === 'increment') return 'counter.inc';
  if(name === 'add') return 'counter.add';
  if(name === 'get') return 'get';
  if(name === 'set') return 'gauge.set';
  if(name === 'set_max') return 'gauge.set_max';
  if(name === 'set_min') return 'gauge.set_min';
  if(name === 'counter_access') return 'counter.access';
  if(name === 'gauge_access') return 'gauge.access';
  if(name === 'record') return 'timer.record';
  if(name === 'record_ns') return 'timer.record_ns';
  if(name === 'start_stop') return 'timer.start_stop';
  if(name === 'stats') return 'timer.stats';
  return name;
}

function buildHistory(obj){
  if(!obj || !obj.entries || typeof obj.entries!=='object') return null;
  const suite = Array.isArray(obj.entries.Criterion) ? obj.entries.Criterion : null;
  if(!suite){
    const ks = Object.keys(obj.entries);
    for(const k of ks){ if(Array.isArray(obj.entries[k])) return obj.entries[k]; }
    return null;
  }
  return suite;
}

function median(arr){
  const xs = arr.filter((v)=> typeof v==='number' && isFinite(v));
  if(!xs.length) return null;
  const s = xs.slice().sort((a,b)=>a-b);
  const mid = Math.floor(s.length/2);
  return s.length % 2 ? s[mid] : (s[mid-1]+s[mid])/2;
}
function baselinePrevMedian(series, k=5){
  if(!series || series.length<2) return null;
  const prev = series.slice(0, -1); // exclude latest
  const window = prev.slice(-k);
  if(!window.length) return null;
  return median(window);
}

function geomean(arr){
  const xs = arr.filter((v)=> typeof v==='number' && isFinite(v) && v>0);
  if(!xs.length) return null;
  const sum = xs.reduce((acc,v)=> acc + Math.log(v), 0);
  return Math.exp(sum/xs.length);
}

(async function main(){
  try{
    const txt = await get(FEED_URL);
    const parsed = extractJSON(txt);
    if(!parsed){
      console.error('Could not parse benchmark feed');
      process.exit(0); // do not hard-fail if feed unavailable
    }
    const suite = buildHistory(parsed);
    if(!suite){
      console.error('No suite history available');
      process.exit(0);
    }
    // Build series map: name -> array of values
    const series = {};
    suite.forEach((run)=>{
      (run.benches||[]).forEach((b)=>{
        if(b.name==='base'||b.name==='change') return;
        const n = friendlyName(b.name);
        if(!series[n]) series[n]=[];
        series[n].push(b.value);
      });
    });

    // Determine which series to enforce
    const inCiMode = process.env.CI_MODE === '1' || process.env.CI_MODE === 'true';
    const perBenchKeys = inCiMode ? CORE : Object.keys(THRESHOLDS.perBenchmark);

    // Per-benchmark checks (compare last value vs median of previous up to 5 runs)
    const breaches = [];
    for(const name of perBenchKeys){
      const thr = THRESHOLDS.perBenchmark[name];
      const s = series[name];
      if(!s || s.length<2) continue;
      const b = s[s.length-1];
      const a = baselinePrevMedian(s, 5);
      if(!(a>0 && b>0)) continue;
      const ratio = (b-a)/a; // positive means slower
      if(ratio > thr){ breaches.push(`${name}: +${(ratio*100).toFixed(2)}% (> ${(thr*100).toFixed(0)}%)`); }
    }

    // Geomean across core benches (use per-series ratio of last vs median of prev)
    const coreKeys = CORE;
    const geos = [];
    coreKeys.forEach((k)=>{
      const s = series[k]; if(!s||s.length<2) return; const b=s[s.length-1]; const a=baselinePrevMedian(s,5); if(a>0&&b>0){ geos.push(b/a); }
    });
    const gm = geomean(geos);
    if(gm!==null){
      const gmr = gm-1; if(gmr > THRESHOLDS.geomean){ breaches.unshift(`geomean: +${(gmr*100).toFixed(2)}% (> ${(THRESHOLDS.geomean*100).toFixed(0)}%)`); }
    }

    if(breaches.length){
      console.error('Benchmark regression detected:\n'+breaches.join('\n'));
      process.exit(1);
    }
    console.log('Benchmark regression check: OK');
  }catch(err){
    // Do not fail the job if the feed is unreachable; log and continue
    console.error('bench_regression_check error:', err.message||err);
    process.exit(0);
  }
})();