--[[ Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. The ASF licenses this file to you under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ]] --[[ mod_lua implementation of the server-status page ]] local ssversion = "0.11" -- version of this script local redact_ips = true -- whether to replace the last two bits of every IP with 'x.x' local warning_banner = [[

Don't be alarmed - this page is here for a reason!

This is an example server status page for the Apache HTTP Server. Nothing on this server is secret, no URL tokens, no sensitive passwords. Everything served from here is static data.

]] local show_warning = true -- whether to display the above warning/notice on the page local show_modules = false -- Whether to list loaded modules or not local show_threads = true -- whether to list thread information or not -- pre-declare some variables defined at the bottom of this script: local status_js, status_css, quokka_js -- quick and dirty JSON conversion local function quickJSON(input) if type(input) == "table" then local t = 'array' for k, v in pairs(input) do if type(k) ~= "number" then t = 'hash' break end end if t == 'hash' then local out = "" local tbl = {} for k, v in pairs(input) do local kv = ([["%s": %s]]):format(k, quickJSON(v)) table.insert(tbl, kv) end return "{" .. table.concat(tbl, ", ") .. "}" else local tbl = {} for k, v in pairs(input) do table.insert(tbl, quickJSON(v)) end return "[" .. table.concat(tbl, ", ") .. "]" end elseif type(input) == "string" then return ([["%s"]]):format(input:gsub('"', '\\"'):gsub("[\r\n\t]", " ")) elseif type(input) == "number" then return tostring(input) elseif type(input) == "boolean" then return (input and "true" or "false") else return "null" end end -- Module information callback local function modInfo(r, modname) if modname then r:puts [[ Module information ]] r:puts( ("

Details for module %s

\n"):format(r:escape_html(modname)) ) -- Queries the server for information about a module local mod = r.module_info(modname) if mod then for k, v in pairs(mod.commands) do -- print out all directives accepted by this module r:puts( ("%s: %s
\n"):format(r:escape_html(k), v)) end end -- HTML tail r:puts[[ ]] end end -- Function for generating server stats function getServerState(r, verbose) local state = {} state.mpm = { type = "prefork", -- default to prefork until told otherwise threadsPerChild = 1, threaded = false, maxServers = r.mpm_query(12), activeServers = 0 } if r.mpm_query(14) == 1 then state.mpm.type = "event" -- this is event mpm elseif r.mpm_query(3) >= 1 then state.mpm.type = "worker" -- it's not event, but it's threaded, we'll assume worker mpm (could be motorz??) elseif r.mpm_query(2) == 1 then state.mpm.type = "winnt" -- it's threaded, but not worker nor event, so it's probably winnt end if state.mpm.type ~= "prefork" then state.mpm.threaded = true -- it's threaded state.mpm.threadsPerChild = r.mpm_query(6) -- get threads per child proc end state.processes = {} -- list of child procs state.connections = { -- overall connection info idle = 0, active = 0 } -- overall server stats state.server = { connections = 0, bytes = 0, built = r.server_built, localtime = os.time(), uptime = os.time() - r.started, version = r.banner, host = r.server_name, modules = nil, extended = show_threads, -- whether extended status is available or not } -- if show_modules is true, add list of modules to the JSON if show_modules then state.server.modules = {} for k, module in pairs(r:loaded_modules()) do table.insert(state.server.modules, module) end end -- Fetch process/thread data for i=0,state.mpm.maxServers-1,1 do local server = r.scoreboard_process(r, i); if server then local s = { active = false, pid = nil, bytes = 0, stime = 0, utime = 0, connections = 0, } local tstates = {} if server.pid then state.connections.idle = state.connections.idle + (server.keepalive or 0) s.connections = 0 if server.pid > 0 then state.mpm.activeServers = state.mpm.activeServers + 1 s.active = true s.pid = server.pid end for j = 0, state.mpm.threadsPerChild-1, 1 do local worker = r.scoreboard_worker(r, i, j) if worker then s.stime = s.stime + (worker.stimes or 0); s.utime = s.utime + (worker.utimes or 0); if verbose and show_threads then s.threads = s.threads or {} table.insert(s.threads, { bytes = worker.bytes_served, thread = ("0x%x"):format(worker.tid), client = redact_ips and (worker.client or "???"):gsub("[a-f0-9]+[.:]+[a-f0-9]+$", "x.x") or worker.client or "???", cost = ((worker.utimes or 0) + (worker.stimes or 0)), count = worker.access_count, vhost = worker.vhost:gsub(":%d+", ""), request = worker.request, last_used = math.floor(worker.last_used/1000000) }) end state.server.connections = state.server.connections + worker.access_count s.bytes = s.bytes + worker.bytes_served s.connections = s.connections + worker.access_count if server.pid > 0 then tstates[worker.status] = (tstates[worker.status] or 0) + 1 end end end end s.workerStates = { keepalive = (server.keepalive > 0) and server.keepalive or tstates[5] or 0, closing = tstates[8] or 0, idle = tstates[2] or 0, writing = tstates[4] or 0, reading = tstates[3] or 0, graceful = tstates[9] or 0 } table.insert(state.processes, s) state.server.bytes = state.server.bytes + s.bytes state.connections.active = state.connections.active + (tstates[8] or 0) + (tstates[4] or 0) + (tstates[3] or 0) end end return state end -- Handler function function handle(r) -- Parse GET data, if any, and set content type local GET = r:parseargs() if GET['module'] then modInfo(r, GET['module']) return apache2.OK end -- If we only need the stats feed, compact it and hand it over if GET['view'] and GET['view'] == "json" then local state = getServerState(r, GET['extended'] == 'true') r.content_type = "application/json" r:puts(quickJSON(state)) return apache2.OK end if not GET['resource'] then local state = getServerState(r, show_threads) -- Print out the HTML for the front page r.content_type = "text/html" r:puts ( ([=[ Server status for %s
%s
Quick Stats
Charts
]=]):format( r.server_name, r.banner, r.server_name, show_warning and warning_banner or "" ) ); -- HTML tail r:puts[[ ]] else -- Resource documents (CSS, JS, PNG) if GET['resource'] == 'js' then r.content_type = "application/javascript" r:puts(quokka_js) r:puts(status_js) elseif GET['resource'] == 'css' then r.content_type = "text/css" r:puts(status_css) elseif GET['resource'] == 'feather' then r.content_type = "image/png" r:write(r:base64_decode('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')) end end return apache2.OK; end ------------------------------------ -- JavaScript and CSS definitions -- ------------------------------------ -- Set up some JavaScripts: status_js = [==[ Number.prototype.pad = function(size) { var str = String(this); while (str.length < size) { str = "0" + str; } return str; } function getAsync(theUrl, xstate, callback) { var xmlHttp = null; if (window.XMLHttpRequest) { xmlHttp = new XMLHttpRequest(); } else { xmlhttp = new ActiveXObject("Microsoft.XMLHTTP"); } xmlHttp.open("GET", theUrl, true); xmlHttp.send(null); xmlHttp.onreadystatechange = function(state) { if (xmlHttp.readyState == 4 && xmlHttp.status == 200) { if (callback) { callback(JSON.parse(xmlHttp.responseText)); } } } } var actionCache = []; var connectionCache = []; var trafficCache = []; var processes = {}; var lastBytes = 0; var lastConnections = 0; var negativeBytes = 0; // cache for proc reloads, which skews traffic var updateSpeed = 5; // How fast do charts update? var maxRecords = 24; // How many records to show per chart var cpumax = 1000000; // random cpu max(?) function refreshCharts(json, state) { if (json && json.processes) { // general server info box var gs = document.getElementById('server_breakdown'); gs.innerHTML = ""; gs.innerHTML += "Server version: " + json.server.version + "
"; gs.innerHTML += "Server built: " + json.server.built + "
"; gs.innerHTML += "Server MPM: " + json.mpm.type + "
"; // Get a timestamp var now = new Date(); var ts = now.getHours().pad(2) + ":" + now.getMinutes().pad(2) + ":" + now.getSeconds().pad(2); var utime = 0; var stime = 0; // Construct state based on proc details var state = { timestamp: ts, closing: 0, idle: 0, writing: 0, reading: 0, keepalive: 0, graceful: 0 } for (var i in json.processes) { var proc = json.processes[i]; if (proc.pid) { state.closing += proc.workerStates.closing||0; state.idle += proc.workerStates.idle||0; state.writing += proc.workerStates.writing||0; state.reading += proc.workerStates.reading||0; state.keepalive += proc.workerStates.keepalive||0; state.graceful += proc.workerStates.graceful||0; utime += proc.utime; stime += proc.stime; } } // Push action state entry into action cache with timestamp // Shift if more than 10 entries in cache actionCache.push(state); if (actionCache.length > maxRecords) { actionCache.shift(); } // construct array for QuokkaLines var arr = []; for (var i in actionCache) { var el = actionCache[i]; if (json.mpm.type == 'event') { arr.push([el.timestamp, el.closing, el.idle, el.writing, el.reading, el.graceful]); } else { arr.push([el.timestamp, el.keepalive, el.closing, el.idle, el.writing, el.reading, el.graceful]); } } var states = ['Keepalive', 'Closing', 'Idle', 'Writing', 'Reading', 'Graceful'] if (json.mpm.type == 'event') { states.shift(); if (document.getElementById('mpminfo')) { document.getElementById('mpminfo').innerHTML = "(" + fn(parseInt(json.connections.idle)) + " connections in idle keepalive)"; } } // Draw action chart quokkaLines("actions_div", states, arr, { lastsum: true, hires: true, nosum: true, stack: true, curve: true, title: "Thread states" } ); // Get traffic, figure out how much it was this time (0 if just started!) var bytesThisTurn = 0; var connectionsThisTurn = 0; for (var i in json.processes) { var proc = json.processes[i]; var pid = proc.pid // if we haven't seen this proc before, ignore its bytes first time if (!processes[pid]) { processes[pid] = { bytes: proc.bytes, connections: proc.connections, } } else { bytesThisTurn += proc.bytes - processes[pid].bytes; if (pid) { x = proc.connections - processes[pid].connections; connectionsThisTurn += (x > 0) ? x : 0; } processes[pid].bytes = proc.bytes; processes[pid].connections = proc.connections; } } if (lastBytes == 0 ) { bytesThisTurn = 0; } lastBytes = 1; // Push a new element into cache, prune cache var el = { timestamp: ts, bytes: bytesThisTurn/updateSpeed }; trafficCache.push(el); if (trafficCache.length > maxRecords) { trafficCache.shift(); } // construct array for QuokkaLines arr = []; for (var i in trafficCache) { var el = trafficCache[i]; arr.push([el.timestamp, el.bytes]); } // Draw action chart quokkaLines("traffic_div", ['Traffic'], arr, { traffic: true, hires: true, nosum: true, stack: true, curve: true, title: "Traffic per second" } ); // Get connections per second // Push a new element into cache, prune cache var el = { timestamp: ts, connections: (connectionsThisTurn+1)/updateSpeed }; connectionCache.push(el); if (connectionCache.length > maxRecords) { connectionCache.shift(); } // construct array for QuokkaLines arr = []; for (var i in connectionCache) { var el = connectionCache[i]; arr.push([el.timestamp, el.connections]); } // Draw connection chart quokkaLines("connection_div", ['Connections/sec'], arr, { traffic: false, hires: true, nosum: true, stack: true, curve: true, title: "Connections per second" } ); // Thread info quokkaCircle("status_div", [ { title: 'Active', value: (json.mpm.threadsPerChild*json.mpm.activeServers)}, { title: 'Reserve', value: (json.mpm.threadsPerChild*(json.mpm.activeServers-json.mpm.maxServers))} ], { title: "Worker pool", hires: true}); // Idle vs active connections var idlecons = json.connections.idle; var activecons = json.connections.active; quokkaCircle("idle_div", [ { title: 'Idle', value: idlecons}, { title: 'Active', value: activecons}, ], { hires: true, title: "Idle vs active connections"}); // CPU info while ( (stime+utime) > cpumax ) { cpumax = cpumax * 2; } quokkaCircle("cpu_div", [ { title: 'Idle', value: (cpumax - stime - utime) / (cpumax/100)}, { title: 'System', value: stime/(cpumax/100)}, { title: 'User', value: utime/(cpumax/100)} ], { hires: true, title: "CPU usage", pct: true}); // General stats infobox var gstats = document.getElementById('general_stats'); gstats.innerHTML = ''; // wipe the box // Days since restart var u_f = Math.floor(json.server.uptime/8640.0) / 10; var u_d = Math.floor(json.server.uptime/86400); var u_h = Math.floor((json.server.uptime%86400)/3600); var u_m = Math.floor((json.server.uptime%3600)/60); var u_s = Math.floor(json.server.uptime %60); var str = u_d + " day" + (u_d != 1 ? "s, " : ", ") + u_h + " hour" + (u_h != 1 ? "s, " : ", ") + u_m + " minute" + (u_m != 1 ? "s" : ""); var ubox = document.createElement('div'); ubox.setAttribute("class", "statsbox"); ubox.innerHTML = "" + u_f + " days
since last (re)start.
" + str; gstats.appendChild(ubox); // Bytes transferred in total var MB = fnmb(json.server.bytes); var KB = (json.server.bytes > 0) ? fnmb(json.server.bytes/json.server.connections) : 0; var KBs = fnmb(json.server.bytes/json.server.uptime); var mbbox = document.createElement('div'); mbbox.setAttribute("class", "statsbox"); mbbox.innerHTML = "" + MB + "
transferred in total.
" + KBs + "/sec, " + KB + "/request"; gstats.appendChild(mbbox); // connections in total var cons = fn(json.server.connections); var cps = Math.floor(json.server.connections/json.server.uptime*100)/100; var conbox = document.createElement('div'); conbox.setAttribute("class", "statsbox"); conbox.innerHTML = "" + cons + " conns
since server started.
" + cps + " requests per second"; gstats.appendChild(conbox); // threads working var tpc = json.mpm.threadsPerChild; var activeThreads = fn(json.mpm.activeServers * json.mpm.threadsPerChild); var maxThreads = json.mpm.maxServers * json.mpm.threadsPerChild; var tbox = document.createElement('div'); tbox.setAttribute("class", "statsbox"); tbox.innerHTML = "" + activeThreads + " threads
currently at work (" + json.mpm.activeServers + "x" + tpc+" threads).
" + maxThreads + " (" + json.mpm.maxServers + "x"+tpc+") threads allowed."; gstats.appendChild(tbox); window.setTimeout(waitTwo, updateSpeed*1000); // resize pane document.getElementById('leftpane').style.height = document.getElementById('wrapper').getBoundingClientRect().height + "px"; // Do we have extended info and module lists?? if (json.server.extended) document.getElementById('threads_button').style.display = 'block'; if (json.server.modules && json.server.modules.length > 0) { var panel = document.getElementById('modules_breakdown'); var list = "
    "; for (var i in json.server.modules) { var mod = json.server.modules[i]; list += "
  • " + mod + "
  • "; } list += "
"; panel.innerHTML = list; document.getElementById('modules_button').style.display = 'block'; } } else if (json === false) { waitTwo(); } } function refreshThreads(json, state) { var box = document.getElementById('threads_breakdown'); box.innerHTML = ""; for (var i in json.processes) { var proc = json.processes[i]; var phtml = '
'; if (!proc.active) phtml = '
'; phtml += "

Process " + i + ":

"; phtml += "PID: " + (proc.pid||"None (not active)") + "
"; if (proc.threads && proc.active) { phtml += ""; for (var j in proc.threads) { var thread = proc.threads[j]; thread.request = (thread.request||"(Unknown)").replace(/[<>]+/g, ""); phtml += ""; } phtml += "
Thread IDAccess countBytes servedLast UsedLast clientLast request
"+thread.thread+""+thread.count+""+thread.bytes+""+thread.last_used+""+thread.client+""+thread.request+"
"; } else { phtml += "

No thread information available

"; } phtml += "
"; box.innerHTML += phtml; } } function waitTwo() { getAsync(location.href + "?view=json&rnd=" + Math.random(), null, refreshCharts) } function showPanel(what) { var items = ['dashboard','misc','threads','modules']; for (var i in items) { var item = items[i]; var btn = document.getElementById(item+'_button'); var panel = document.getElementById(item+'_panel'); if (item == what) { btn.setAttribute("class", "btn active"); panel.style.display = 'block'; } else { btn.setAttribute("class", "btn"); panel.style.display = 'none'; } } // special constructors if (what == 'threads') { getAsync(location.href + "?view=json&extended=true&rnd=" + Math.random(), null, refreshThreads) } } function fn(num) { num = num + ""; num = num.replace(/(\d)(\d{9})$/, '$1,$2'); num = num.replace(/(\d)(\d{6})$/, '$1,$2'); num = num.replace(/(\d)(\d{3})$/, '$1,$2'); return num; } function fnmb(num) { var add = "bytes"; var dec = ""; var mul = 1; if (num > 1024) { add = "KB"; mul= 1024; } if (num > (1024*1024)) { add = "MB"; mul= 1024*1024; } if (num > (1024*1024*1024)) { add = "GB"; mul= 1024*1024*1024; } if (num > (1024*1024*1024*1024)) { add = "TB"; mul= 1024*1024*1024*1024; } num = num / mul; if (add != "bytes") { dec = "." + Math.floor( (num - Math.floor(num)) * 100 ); } return ( fn(Math.floor(num)) + dec + " " + add ); } function sort(a,b){ last_col = -1; var sort_reverse = false; var sortWay = a.getAttribute("sort_" + b); if (sortWay && sortWay == "forward") { a.setAttribute("sort_" + b, "reverse"); sort_reverse = true; } else { a.setAttribute("sort_" + b, "forward"); } var c,d,e,f,g,h,i; c=a.rows.length; if(c<1){ return; } d=a.rows[1].cells.length; e=1; var j=new Array(c); f=0; for(h=e;hn[b]) ? true : false; var lt = (m[b] parseInt(n[b], 10) ? true : false; lt = parseInt(m[b], 10) < parseInt(n[b], 10) ? true : false; } if (sort_reverse) {gt = (!gt); lt = (!lt);} if(gt){ j[i+1]=m; j[i]=n; l=true; } } else{ if(lt){ j[i+1]=m; j[i]=n; l=true; } } } if(l===false){ break; } } f=e; for(h=0;h 1024) { add = "KB"; mul= 1024; } if (num > (1024*1024)) { add = "MB"; mul= 1024*1024; } if (num > (1024*1024*1024)) { add = "GB"; mul= 1024*1024*1024; } if (num > (1024*1024*1024*1024)) { add = "TB"; mul= 1024*1024*1024*1024; } num = num / mul; if (add != "b" && num < 10) { dec = "." + Math.floor( (num - Math.floor(num)) * 100 ); } return ( Math.floor(num) + dec + " " + add ); } // Hue, Saturation and Lightness to Red, Green and Blue: function quokka_internal_hsl2rgb (h,s,l) { var min, sv, switcher, fract, vsf; h = h % 1; if (s > 1) s = 1; if (l > 1) l = 1; var v = (l <= 0.5) ? (l * (1 + s)) : (l + s - l * s); if (v === 0) return { r: 0, g: 0, b: 0 }; min = 2 * l - v; sv = (v - min) / v; var sh = (6 * h) % 6; switcher = Math.floor(sh); fract = sh - switcher; vsf = v * sv * fract; switch (switcher) { case 0: return { r: v, g: min + vsf, b: min }; case 1: return { r: v - vsf, g: v, b: min }; case 2: return { r: min, g: v, b: min + vsf }; case 3: return { r: min, g: v - vsf, b: v }; case 4: return { r: min + vsf, g: min, b: v }; case 5: return { r: v, g: min, b: v - vsf }; } return {r:0, g:0, b: 0}; } // RGB to Hex conversion function quokka_internal_rgb2hex(r, g, b) { return "#" + ((1 << 24) + (Math.floor(r) << 16) + (Math.floor(g) << 8) + Math.floor(b)).toString(16).slice(1); } // Generate color list used for charts var colors = []; var rgbs = [] var numColorRows = 6; var numColorColumns = 20; for (var x=0;x 8 ? (Math.random()) : (rnd[0]/255), y > 8 ? (0.75+(y*0.05)) : (rnd[1]/255), y > 8 ? (0.42 + (y*0.05*(x/numColorRows))) : (0.1 + rnd[2]/512)); // Light (primary) color: var hex = quokka_internal_rgb2hex(color.r*255, color.g*255, color.b*255); // Darker variant for gradients: var dhex = quokka_internal_rgb2hex(color.r*131, color.g*131, color.b*131); // Medium variant for legends: var mhex = quokka_internal_rgb2hex(color.r*200, color.g*200, color.b*200); colors.push([hex, dhex, color, mhex]); } } /* Function for drawing pie diagrams * Example usage: * quokkaCircle("canvasName", [ { title: 'ups', value: 30}, { title: 'downs', value: 70} ] ); */ function quokkaCircle(id, tags, opts) { // Get Canvas object and context var canvas = document.getElementById(id); var ctx=canvas.getContext("2d"); // Calculate the total value of the pie var total = 0; var k; for (k in tags) { tags[k].value = Math.abs(tags[k].value); total += tags[k].value; } // Draw the empty pie var begin = 0; var stop = 0; var radius = (canvas.height*0.75)/2; ctx.clearRect(0, 0, canvas.width, canvas.height); ctx.beginPath(); ctx.shadowBlur = 6; ctx.shadowOffsetX = 6; ctx.shadowOffsetY = 6; ctx.shadowColor = "#555"; ctx.lineWidth = (opts && opts.hires) ? 6 : 2; ctx.strokeStyle = "#222"; ctx.arc((canvas.width-140)/2,canvas.height/2,radius, 0, Math.PI * 2); ctx.closePath(); ctx.stroke(); ctx.fill(); ctx.shadowBlur = 0; ctx.shadowOffsetY = 0; ctx.shadowOffsetX = 0; // Draw a title if set: if (opts && opts.title) { ctx.font= (opts && opts.hires) ? "28px Sans-Serif" : "15px Sans-Serif"; ctx.fillStyle = "#000000"; ctx.textAlign = "center"; ctx.fillText(opts.title,(canvas.width-140)/2, (opts && opts.hires) ? 30:15); ctx.textAlign = "left"; } ctx.beginPath(); var posY = 50; var left = 120 + ((canvas.width-140)/2) + ((opts && opts.hires) ? 40 : 25) for (k in tags) { var val = tags[k].value; stop = stop + (2 * Math.PI * (val / total)); // Make a pizza slice ctx.beginPath(); ctx.lineCap = 'round'; ctx.arc((canvas.width-140)/2,canvas.height/2,radius,begin,stop); ctx.lineTo((canvas.width-140)/2,canvas.height/2); ctx.closePath(); ctx.lineWidth = 0; ctx.stroke(); // Add color gradient var grd=ctx.createLinearGradient(0,canvas.height*0.2,0,canvas.height); grd.addColorStop(0,colors[k % colors.length][1]); grd.addColorStop(1,colors[k % colors.length][0]); ctx.fillStyle = grd; ctx.fill(); begin = stop; // Make color legend ctx.fillRect(left, posY-((opts && opts.hires) ? 15 : 10), (opts && opts.hires) ? 14 : 7, (opts && opts.hires) ? 14 : 7); // Add legend text ctx.shadowColor = "rgba(0,0,0,0)" ctx.font= (opts && opts.hires) ? "22px Sans-Serif" : "12px Sans-Serif"; ctx.fillStyle = "#000"; ctx.fillText(tags[k].title + " (" + Math.floor(val) + (opts && opts.pct ? "%" : "") + ")",left+20,posY); posY += (opts && opts.hires) ? 28 : 14; } } /* Function for drawing line charts * Example usage: * quokkaLines("myCanvas", ['Line a', 'Line b', 'Line c'], [ [x1,a1,b1,c1], [x2,a2,b2,c2], [x3,a3,b3,c3] ], { stacked: true, curve: false, title: "Some title" } ); */ function quokkaLines(id, titles, values, options, sums) { var canvas = document.getElementById(id); var ctx=canvas.getContext("2d"); // clear the canvas first ctx.clearRect(0, 0, canvas.width, canvas.height); ctx.lineWidth = 0.25; ctx.strokeStyle = "#000000"; var lwidth = 300; var lheight = 75; wspace = (options && options.hires) ? 110 : 55; var rectwidth = canvas.width - lwidth - wspace; var stack = options ? options.stack : false; var curve = options ? options.curve : false; var title = options ? options.title : null; var spots = options ? options.points : false; var noX = options ? options.nox : false; var verts = options ? options.verts : true; if (noX) { lheight = 0; } // calc rectwidth if titles are large var nlwidth = 0 for (var k in titles) { ctx.font= (options && options.hires) ? "24px Sans-Serif" : "12px Sans-Serif"; ctx.fillStyle = "#00000"; var x = parseInt(k) if (!noX) { x = x + 1; } var sum = 0 for (var y in values) { sum += values[y][x] } var t = titles[k] + (!options.nosum ? " (" + ((sums && sums[k]) ? sums[k] : sum.toFixed(0)) + ")" : ""); var w = ctx.measureText(t).width + 48; if (w > lwidth && w > nlwidth) { nlwidth = w } if (nlwidth > 0) { rectwidth -= nlwidth - lwidth lwidth = nlwidth } } // Draw a border ctx.lineWidth = 0.5; ctx.strokeRect((wspace*0.75), 30, rectwidth, canvas.height - lheight - 40); // Draw a title if set: if (title != null) { ctx.font= (options && options.hires) ? "24px Sans-Serif" : "15px Sans-Serif"; ctx.fillStyle = "#00000"; ctx.textAlign = "center"; ctx.fillText(title,rectwidth/2, 20); } // Draw legend ctx.textAlign = "left"; var posY = 50; for (var k in titles) { var x = parseInt(k) if (!noX) { x = x + 1; } var sum = 0 for (var y in values) { sum += values[y][x] } var title = titles[k] + (!options.nosum ? (" (" + ((sums && sums[k]) ? sums[k] : sum.toFixed(0)) + ")") : ""); if (options && options.lastsum) { title = titles[k] + " (" + values[values.length-1][x].toFixed(0) + ")"; } ctx.fillStyle = colors[k % colors.length][3]; ctx.fillRect(wspace + rectwidth + 75 , posY-((options && options.hires) ? 18:9), (options && options.hires) ? 20:10, (options && options.hires) ?20:10); // Add legend text ctx.font= (options && options.hires) ? "24px Sans-Serif" : "14px Sans-Serif"; ctx.fillStyle = "#00000"; ctx.fillText(title,canvas.width - lwidth + ((options && options.hires) ? 100:60), posY); posY += (options && options.hires) ? 30:15; } // Find max and min var max = null; var min = 0; var stacked = null; for (x in values) { var s = 0; for (y in values[x]) { if (y > 0 || noX) { s += values[x][y]; if (max === null || max < values[x][y]) { max = values[x][y]; } if (min === null || min > values[x][y]) { min = values[x][y]; } } } if (stacked === null || stacked < s) { stacked = s; } } if (min == max) max++; if (stack) { min = 0; max = stacked; } // Set number of lines to draw and each step var numLines = 5; var step = (max-min) / (numLines+1); // Prettify the max value so steps aren't ugly numbers if (step %1 != 0) { step = (Math.round(step+0.5)); max = step * (numLines+1); } // Draw horizontal lines for (x = -1; x <= numLines; x++) { ctx.beginPath(); var y = 30 + (((canvas.height-40-lheight) / (numLines+1)) * (x+1)); ctx.moveTo(wspace*0.75, y); ctx.lineTo(wspace*0.75 + rectwidth, y); ctx.lineWidth = 0.25; ctx.stroke(); // Add values ctx.font= (options && options.hires) ? "20px Sans-Serif" : "12px Sans-Serif"; ctx.fillStyle = "#000000"; var val = Math.round( ((max-min) - (step*(x+1))) ); if (options && options.traffic) { val = quokka_fnmb(val); } ctx.textAlign = "left"; ctx.fillText( val,canvas.width - lwidth - 20, y+8); ctx.textAlign = "right"; ctx.fillText( val,wspace-32, y+8); ctx.closePath(); } // Draw vertical lines var sx = 1 var numLines = values.length-1; var step = (canvas.width - lwidth - wspace*0.75) / values.length; while (step < 24) { step *= 2 sx *= 2 } if (verts) { ctx.beginPath(); for (var x = 1; x < values.length; x++) { if (x % sx == 0) { var y = (wspace*0.75) + (step * (x/sx)); ctx.moveTo(y, 30); ctx.lineTo(y, canvas.height - 10 - lheight); ctx.lineWidth = 0.25; ctx.stroke(); } } ctx.closePath(); } // Some pre-calculations of steps var step = (rectwidth) / (values.length > 1 ? values.length-1:1); // Draw X values if noX isn't set: if (noX != true) { ctx.beginPath(); for (var i = 0; i < values.length; i++) { zz = 1 var x = (wspace*0.75) + ((step) * i); var y = canvas.height - lheight + 5; if (i % sx == 0) { ctx.translate(x, y); ctx.moveTo(0,0); ctx.lineTo(0,-15); ctx.stroke(); ctx.rotate(45*Math.PI/180); ctx.textAlign = "left"; var val = values[i][0]; if (val.constructor.toString().match("Date()")) { val = val.toDateString(); } ctx.fillText(val.toString(), 0, 0); ctx.rotate(-45*Math.PI/180); ctx.translate(-x,-y); } } ctx.closePath(); } // Draw each line var stacks = []; var pstacks = []; for (k in values) { if (k > 0) { stacks[k] = 0; pstacks[k] = canvas.height - 40 - lheight; }} for (k in titles) { var maxY = 0, minY = 99999; ctx.beginPath(); var color = colors[k % colors.length][0]; var f = parseInt(k) + 1; if (noX) { f = parseInt(k); } var value = values[0][f]; var step = rectwidth / numLines; var x = (wspace*0.75); var y = (canvas.height - 10 - lheight) - (((value-min) / (max-min)) * (canvas.height - 40 - lheight)); var py = y; if (stack) { stacks[0] = stacks[0] ? stacks[0] : 0 y -= stacks[0]; pstacks[0] = stacks[0]; stacks[0] += (((value-min) / (max-min)) * (canvas.height - 40 - lheight)); } // Draw line ctx.moveTo(x, y); var pvalY = y; var pvalX = x; for (var i in values) { if (i > 0) { x = (wspace*0.75) + (step*i); var f = parseInt(k) + 1; if (noX == true) { f = parseInt(k); } value = values[i][f]; y = (canvas.height - 10 - lheight) - (((value-min) / (max-min)) * (canvas.height - 40 - lheight)); if (stack) { y -= stacks[i]; pstacks[i] = stacks[i]; stacks[i] += (((value-min) / (max-min)) * (canvas.height - 40- lheight)); } if (y > maxY) maxY = y; if (y < minY) minY = y; // Draw curved lines?? /* We'll do: (x1,y1)-----(x1.5,y1) * | * (x1.5,y2)-----(x2,y2) * with a quadratic beizer thingy */ if (curve) { ctx.bezierCurveTo((pvalX + x) / 2, pvalY, (pvalX + x) / 2, y, x, y); pvalX = x; pvalY = y; } // Nope, just draw straight lines else { ctx.lineTo(x, y); } if (spots) { ctx.fillStyle = color; ctx.translate(x-2, y-2); ctx.rotate(-45*Math.PI/180); ctx.fillRect(-2,1,4,4); ctx.rotate(45*Math.PI/180); ctx.translate(-x+2, -y+2); } } } ctx.lineWidth = 4; ctx.strokeStyle = color; ctx.stroke(); if (minY == maxY) maxY++; // Draw stack area if (stack) { ctx.globalAlpha = 0.65; for (i in values) { if (i > 0) { var f = parseInt(k) + 1; if (noX == true) { f = parseInt(k); } x = (wspace*0.75) + (step*i); value = values[i][f]; y = (canvas.height - 10 - lheight) - (((value-min) / (max-min)) * (canvas.height - 40 - lheight)); y -= stacks[i]; } } var pvalY = y; var pvalX = x; if (y > maxY) maxY = y; if (y < minY) minY = y; for (i in values) { var l = values.length - i - 1; x = (wspace*0.75) + (step*l); y = canvas.height - 10 - lheight - pstacks[l]; if (y > maxY) maxY = y; if (y < minY) minY = y; if (curve) { ctx.bezierCurveTo((pvalX + x) / 2, pvalY, (pvalX + x) / 2, y, x, y); pvalX = x; pvalY = y; } else { ctx.lineTo(x, y); } } ctx.lineTo((wspace*0.75), py - pstacks[0]); ctx.lineWidth = 0; var grad = ctx.createLinearGradient(0, minY, 0, maxY); grad.addColorStop(0.25, colors[k % colors.length][0]) grad.addColorStop(1, colors[k % colors.length][1]) ctx.strokeStyle = colors[k % colors.length][0]; ctx.fillStyle = grad; ctx.fill(); ctx.fillStyle = "#000" ctx.strokeStyle = "#000" ctx.globalAlpha = 1; } ctx.closePath(); } // draw feather base_image = new Image(); base_image.src = ''; base_image.onload = function(){ ctx.globalAlpha = 0.15 ctx.drawImage(base_image, (canvas.width/2) - 64 - (lwidth/2), (canvas.height/2) - 128); ctx.globalAlpha = 1 } } /* Function for drawing line charts * Example usage: * quokkaLines("myCanvas", ['Line a', 'Line b', 'Line c'], [ [x1,a1,b1,c1], [x2,a2,b2,c2], [x3,a3,b3,c3] ], { stacked: true, curve: false, title: "Some title" } ); */ function quokkaBars(id, titles, values, options) { var canvas = document.getElementById(id); var ctx=canvas.getContext("2d"); // clear the canvas first ctx.clearRect(0, 0, canvas.width, canvas.height); var lwidth = 150; var lheight = 75; var stack = options ? options.stack : false; var astack = options ? options.astack : false; var curve = options ? options.curve : false; var title = options ? options.title : null; var noX = options ? options.nox : false; var verts = options ? options.verts : true; if (noX) { lheight = 0; } // Draw a border ctx.lineWidth = 0.5; ctx.strokeRect(25, 30, canvas.width - lwidth - 40, canvas.height - lheight - 40); // Draw a title if set: if (title != null) { ctx.font="15px Arial"; ctx.fillStyle = "#000"; ctx.textAlign = "center"; ctx.fillText(title,(canvas.width-lwidth)/2, 15); } // Draw legend ctx.textAlign = "left"; var posY = 50; for (var k in titles) { var x = parseInt(k) if (!noX) { x = x + 1; } var title = titles[k]; if (title && title.length > 0) { ctx.fillStyle = colors[k % colors.length][0]; ctx.fillRect(canvas.width - lwidth + 20, posY-10, 10, 10); // Add legend text ctx.font="12px Arial"; ctx.fillStyle = "#000"; ctx.fillText(title,canvas.width - lwidth + 40, posY); posY += 15; } } // Find max and min var max = null; var min = 0; var stacked = null; for (x in values) { var s = 0; for (y in values[x]) { if (y > 0 || noX) { s += values[x][y]; if (max == null || max < values[x][y]) { max = values[x][y]; } if (min == null || min > values[x][y]) { min = values[x][y]; } } } if (stacked == null || stacked < s) { stacked = s; } } if (min == max) { max++; } if (stack) { min = 0; max = stacked; } // Set number of lines to draw and each step var numLines = 5; var step = (max-min) / (numLines+1); // Prettify the max value so steps aren't ugly numbers if (step %1 != 0) { step = (Math.round(step+0.5)); max = step * (numLines+1); } // Draw horizontal lines for (x = numLines; x >= 0; x--) { var y = 30 + (((canvas.height-40-lheight) / (numLines+1)) * (x+1)); ctx.moveTo(25, y); ctx.lineTo(canvas.width - lwidth - 15, y); ctx.lineWidth = 0.25; ctx.stroke(); // Add values ctx.font="10px Arial"; ctx.fillStyle = "#000"; ctx.textAlign = "right"; ctx.fillText( Math.round( ((max-min) - (step*(x+1))) * 100 ) / 100,canvas.width - lwidth + 12, y-4); ctx.fillText( Math.round( ((max-min) - (step*(x+1))) * 100 ) / 100,20, y-4); } // Draw vertical lines var sx = 1 var numLines = values.length-1; var step = (canvas.width - lwidth - 40) / values.length; while (step < 24) { step *= 2 sx *= 2 } if (verts) { ctx.beginPath(); for (var x = 1; x < values.length; x++) { if (x % sx == 0) { var y = 35 + (step * (x/sx)); ctx.moveTo(y, 30); ctx.lineTo(y, canvas.height - 10 - lheight); ctx.lineWidth = 0.25; ctx.stroke(); } } } // Some pre-calculations of steps var step = (canvas.width - lwidth - 48) / values.length; var smallstep = (step / titles.length) - 2; // Draw X values if noX isn't set: if (noX != true) { ctx.beginPath(); for (var i = 0; i < values.length; i++) { smallstep = (step / (values[i].length-1)) - 2; zz = 1 var x = 35 + ((step) * i); var y = canvas.height - lheight + 5; if (i % sx == 0) { ctx.translate(x, y); ctx.moveTo(0,0); ctx.lineTo(0,-15); ctx.stroke(); ctx.rotate(45*Math.PI/180); ctx.textAlign = "left"; var val = values[i][0]; if (val.constructor.toString().match("Date()")) { val = val.toDateString(); } ctx.fillText(val.toString(), 0, 0); ctx.rotate(-45*Math.PI/180); ctx.translate(-x,-y); } } } // Draw each line var stacks = []; var pstacks = []; for (k in values) { smallstep = (step / (values[k].length)) - 2; stacks[k] = 0; pstacks[k] = canvas.height - 40 - lheight; var beginX = 0; for (i in values[k]) { if (i > 0 || noX) { var z = parseInt(i); var zz = z; if (!noX) { z = parseInt(i) + 1; zz = z - 2; if (z > values[k].length) { break; } } var value = values[k][i]; var title = titles[i]; var color = colors[zz % colors.length][1]; var fcolor = colors[zz % colors.length][2]; if (values[k][2] && values[k][2].toString().match(/^#.+$/)) { color = values[k][2] fcolor = values[k][2] smallstep = (step / (values[k].length-2)) - 2; } var x = ((step) * k) + ((smallstep+2) * zz) + 5; var y = canvas.height - 10 - lheight; var mdiff = (max-min); mdiff = (mdiff == 0) ? 1 : mdiff; var height = ((canvas.height - 40 - lheight) / (mdiff)) * value * -1; var width = smallstep - 2; if (width <= 1) { width = 1 } if (stack) { width = step - 10; y -= stacks[k]; stacks[k] -= height; x = (step * k) + 4; if (astack) { y = canvas.height - 10 - lheight; } } // Draw bar ctx.beginPath(); ctx.lineWidth = 2; ctx.strokeStyle = color; ctx.strokeRect(27 + x, y, width, height); var alpha = 0.75 if (fcolor.r) { ctx.fillStyle = 'rgba('+ [parseInt(fcolor.r*255),parseInt(fcolor.g*255),parseInt(fcolor.b*255),alpha].join(",") + ')'; } else { ctx.fillStyle = fcolor; } ctx.fillRect(27 + x, y, width, height); } } } } ]==] status_css = [[ html { font-size: 14px; position: relative; background: #272B30; } body { background-color: #272B30; color: #000; margin: 0 auto; min-height: 100%; font-family: Arial, Helvetica, sans-serif; font-weight: normal; } .navbarLeft { background: linear-gradient(to bottom, #F8A900 0%,#D88900 100%); width: 200px; height: 30px; padding-top: 2px; font-size: 1.35rem; color: #FFF; border-bottom: 2px solid #000; float: left; text-align: center; } .navbarRight { background: linear-gradient(to bottom, #EFEFEF 0%,#EEE 100%); width: calc(100% - 240px); height: 28px; color: #333; border-bottom: 2px solid #000; float: left; font-size: 1.3rem; padding-top: 4px; text-align: left; padding-left: 40px; } .wrapper { width: 100%; float: left; background: #33363F; min-height: calc(100% - 80px); position: relative; } .serverinfo { float: left; width: 200px; height: calc(100% - 34px); background: #293D4C; } .skey { background: rgba(30,30,30,0.3); color: #C6E7FF; font-weight: bold; padding: 2px; } .sval { padding: 2px; background: rgba(30,30,30,0.3); color: #FFF; font-size: 0.8rem; border-bottom: 1px solid rgba(200,200,200,0.2); } .charts { padding: 0px; width: calc(100% - 220px); max-width: 1000px; min-height: 100%; margin: 0px auto; position: relative; float: left; margin-left: 20px; } pre, code { font-family: "Courier New", Courier, monospace; } strong { font-weight: bold; } q, em, var { font-style: italic; } /* h1 */ /* ====================== */ h1 { padding: 0.2em; margin: 0; border: 1px solid #405871; background-color: inherit; color: #036; text-decoration: none; font-size: 22px; font-weight: bold; } /* h2 */ /* ====================== */ h2 { padding: 0.2em 0 0.2em 0.7em; margin: 0 0 0.5em 0; text-decoration: none; font-size: 18px; font-weight: bold; text-align: center; } #modules { margin-top:20px; display:none; width:400px; } .servers { width: 1244px; background: #EEE; } tr:nth-child(odd) { background: #F6F6F6; } tr:nth-child(even) { background: #EBEBEB; } td { padding: 2px; } table { border: 1px solid #333; padding: 0px; margin: 5px; min-width: 360px; background: #999; font-size: 0.8rem; } canvas { background: #FFF; margin: 3px; text-align: center; padding: 2px; border-radius: 10px; border: 1px solid #999; } .canvas_wide { position: relative; width: 65%; } .canvas_narrow { position: relative; width: 27%; } a { color: #FFA; } .statsbox { border-radius: 3px; background: #3C3E47; min-width: 150px; height: 60px; float: left; margin: 15px; padding: 10px; } .btn { background: linear-gradient(to bottom, #72ca72 0%,#55bf55 100%); border-radius: 5px; color: #FFF; text-decoration: none; padding-top: 6px; padding-bottom: 6px; padding-left: 3px; padding-right: 3px; font-weight: bold; text-shadow: 1px 1px rgba(0,0,0,0.4); margin: 12px; float: left; clear: none; } .infobox_wrapper { float: left; min-width: 200px; margin: 10px; } .infobox_title { border-top-left-radius: 4px; border-top-right-radius: 4px; background: #FAB227; color: #FFF; border: 2px solid #FAB227; border-bottom: none; font-weight: bold; text-align: center; width: 100%; } .infobox { background: #222222; border: 2px solid #FAB227; border-top: none; color: #EFEFEF; border-bottom-left-radius: 4px; border-bottom-right-radius: 4px; float: left; width: 100%; } .serverinfo ul { margin: 0px; padding: 0px; margin-top: 20px; list-style: none; } .serverinfo ul li .btn { width: calc(100% - 8px); margin: 0px; border: 0px; border-radius: 0px; padding: 0px; padding-top: 8px; padding-left: 8px; height: 24px; background: rgba(0,0,0,0.2); border-bottom: 1px solid rgba(100,100,100,0.3); } .serverinfo ul li:nth-child(1) { border-top: 1px solid rgba(100,100,100,0.3); } .serverinfo ul li .btn.active { background: rgba(30,30,50,0.2); border-left: 4px solid #27FAB2; padding-left: 4px; color: #FFE; } .serverinfo ul li .btn:hover { background: rgba(50,50,50,0.15); border-left: 4px solid #FAB227; padding-left: 4px; color: #FFE; } ]]