Created
February 19, 2023 10:54
-
-
Save EhsanCh/97187902e905a308ce434bda6730073c to your computer and use it in GitHub Desktop.
PHP-FPM real-time status page (Single file without the need for web server configuration)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
// Upload to private url or implement authorization... | |
if (isset($_GET["json"])) { | |
header("Content-type: application/json"); | |
exit(json_encode( fpm_get_status() )); | |
} | |
?> | |
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> | |
<html xmlns="http://www.w3.org/1999/xhtml"> | |
<!-- | |
(c) 2011 Jerome Loyet | |
The PHP License, version 3.01 | |
This is sample real-time status page for FPM. You can change it to better fit your needs. | |
--> | |
<head> | |
<meta http-equiv="Content-Type" content="text/html;charset=utf-8" /> | |
<style type="text/css"> | |
body {background-color: #ffffff; color: #000000;} | |
body, td, th, h1, h2 {font-family: sans-serif;} | |
pre {margin: 0px; font-family: monospace;} | |
a:link {color: #000099; text-decoration: none; background-color: #ffffff;} | |
a:hover {text-decoration: underline;} | |
table {border-collapse: collapse;} | |
.center {text-align: center;} | |
.center table { margin-left: auto; margin-right: auto; text-align: left;} | |
.center th { text-align: center !important; } | |
td, th { border: 1px solid #000000; font-size: 75%; vertical-align: baseline;} | |
h1 {font-size: 150%;} | |
h2 {font-size: 125%;} | |
.p {text-align: left;} | |
.e {background-color: #ccccff; font-weight: bold; color: #000000;} | |
.h {background-color: #9999cc; font-weight: bold; color: #000000;} | |
.v {background-color: #cccccc; color: #000000;} | |
.w {background-color: #ccccff; color: #000000;} | |
.h th { | |
cursor: pointer; | |
} | |
img {float: right; border: 0px;} | |
hr {width: 600px; background-color: #cccccc; border: 0px; height: 1px; color: #000000;} | |
</style> | |
<title>PHP-FPM status page</title> | |
<meta name="ROBOTS" content="NOINDEX,NOFOLLOW,NOARCHIVE" /></head> | |
<body> | |
<div class="center"> | |
<table border="0" cellpadding="3" width="95%"> | |
<tr class="h"> | |
<td> | |
<a href="http://www.php.net/"><img border="0" src="" alt="PHP Logo" /></a><h1 class="p">PHP-FPM real-time status page</h1> | |
</td> | |
</tr> | |
</table> | |
<br /> | |
<table border="0" cellpadding="3" width="95%"> | |
<tr><td class="e">Status URL</td><td class="v"><input type="text" id="url" size="45" /></td></tr> | |
<tr><td class="e">Ajax status</td><td class="v" id="status"></td></tr> | |
<tr><td class="e">Refresh Rate</td><td class="v"><input type="text" id="rate" value="1" /></td></tr> | |
<tr> | |
<td class="e">Actions</td> | |
<td class="v"> | |
<button onclick="javascript:refresh();">Manual Refresh</button> | |
<button id="play" onclick="javascript:playpause();">Play</button> | |
</td> | |
</tr> | |
</table> | |
<h1>Pool Status</h1> | |
<table border="0" cellpadding="3" width="95%" id="short"> | |
<tr style="display: none;"><td> </td></tr> | |
</table> | |
<h1>Active Processes status</h1> | |
<table border="0" cellpadding="3" width="95%" id="active"> | |
<tr class="h"><th>PID↓</th><th>Start Time</th><th>Start Since</th><th>Requests Served</th><th>Request Duration</th><th>Request method</th><th>Request URI</th><th>Query string</th><th>Request Length</th><th>User</th><th>Script</th></tr> | |
</table> | |
<h1>Idle Processes status</h1> | |
<table border="0" cellpadding="3" width="95%" id="idle"> | |
<tr class="h"><th>PID↓</th><th>Start Time</th><th>Start Since</th><th>Requests Served</th><th>Request Duration</th><th>Request method</th><th>Request URI</th><th>Query string</th><th>Request Length</th><th>User</th><th>Script</th><th>Last Request %CPU</th><th>Last Request Memory</th></tr> | |
</table> | |
</div> | |
<p> | |
<a href="http://validator.w3.org/check?uri=referer"> | |
<img src="http://www.w3.org/Icons/valid-xhtml10" alt="Valid XHTML 1.0 Transitional" height="31" width="88" /> | |
</a> | |
</p> | |
<script type="text/javascript"> | |
<!-- | |
var xhr_object = null; | |
var doc_url = document.getElementById("url"); | |
var doc_rate = document.getElementById("rate"); | |
var doc_status = document.getElementById("status"); | |
var doc_play = document.getElementById("play"); | |
var doc_short = document.getElementById("short"); | |
var doc_active = document.getElementById("active"); | |
var doc_idle = document.getElementById("idle"); | |
var rate = 0; | |
var play=0; | |
var delay = 1000; | |
var order_active_index = 0; | |
var order_active_reverse = 0; | |
var order_idle_index = 0; | |
var order_idle_reverse = 0; | |
var sort_index; | |
var sort_order; | |
doc_url.value = location.protocol + '//' + location.host + location.pathname +"?json"; | |
ths = document.getElementsByTagName("th"); | |
for (var i=0; i<ths.length; i++) { | |
var th = ths[i]; | |
if (th.parentNode.className == "h") { | |
th.onclick = function() { order(this); return false; }; | |
} | |
} | |
xhr_object = create_ajax(); | |
function create_ajax() { | |
if (window.XMLHttpRequest) { | |
return new XMLHttpRequest(); | |
} | |
var names = [ | |
"Msxml2.XMLHTTP.6.0", | |
"Msxml2.XMLHTTP.3.0", | |
"Msxml2.XMLHTTP", | |
"Microsoft.XMLHTTP" | |
]; | |
for(var i in names) | |
{ | |
try { | |
return new ActiveXObject(names[i]); | |
} catch(e){} | |
} | |
alert("Browser not compatible ..."); | |
} | |
function order(cell) { | |
var table; | |
if (cell.constructor != HTMLTableCellElement && cell.constructor != HTMLTableHeaderCellElement) { | |
return; | |
} | |
table = cell.parentNode.parentNode.parentNode; | |
if (table == doc_active) { | |
if (order_active_index == cell.cellIndex) { | |
if (order_active_reverse == 0) { | |
cell.innerHTML = cell.innerHTML.replace(/.$/, "↑"); | |
order_active_reverse = 1; | |
} else { | |
cell.innerHTML = cell.innerHTML.replace(/.$/, "↓"); | |
order_active_reverse = 0; | |
} | |
} else { | |
var c = doc_active.rows[0].cells[order_active_index]; | |
c.innerHTML = c.innerHTML.replace(/.$/, ""); | |
cell.innerHTML = cell.innerHTML.replace(/$/, order_active_reverse == 0 ? "↓" : "↑"); | |
order_active_index = cell.cellIndex; | |
} | |
reorder(table, order_active_index, order_active_reverse); | |
return; | |
} | |
if (table == doc_idle) { | |
if (order_idle_index == cell.cellIndex) { | |
if (order_idle_reverse == 0) { | |
cell.innerHTML = cell.innerHTML.replace(/.$/, "↑"); | |
order_idle_reverse = 1; | |
} else { | |
cell.innerHTML = cell.innerHTML.replace(/.$/, "↓"); | |
order_idle_reverse = 0; | |
} | |
} else { | |
var c = doc_idle.rows[0].cells[order_idle_index]; | |
c.innerHTML = c.innerHTML.replace(/.$/, ""); | |
cell.innerHTML = cell.innerHTML.replace(/$/, order_idle_reverse == 0 ? "↓" : "↑"); | |
order_idle_index = cell.cellIndex; | |
} | |
reorder(table, order_idle_index, order_idle_reverse); | |
return; | |
} | |
} | |
function reorder(table, index, order) { | |
var rows = []; | |
while (table.rows.length > 1) { | |
rows.push(table.rows[1]); | |
table.deleteRow(1); | |
} | |
sort_index = index; | |
sort_order = order; | |
rows.sort(sort_table); | |
for (var i in rows) { | |
table.appendChild(rows[i]); | |
} | |
var odd = 1; | |
for (var i=1; i<table.rows.length; i++) { | |
table.rows[i].className = odd++ % 2 == 0 ? "v" : "w"; | |
} | |
return; | |
} | |
function sort_table(a, b) { | |
if (a.cells[0].tagName == "TH") return -1; | |
if (b.cells[0].tagName == "TH") return 1; | |
if (a.cells[sort_index].__search_t == 0) { /* integer */ | |
if (!sort_order) return a.cells[sort_index].__search_v - b.cells[sort_index].__search_v; | |
return b.cells[sort_index].__search_v - a.cells[sort_index].__search_v;; | |
} | |
/* string */ | |
if (!sort_order) return a.cells[sort_index].__search_v.localeCompare(b.cells[sort_index].__search_v); | |
else return b.cells[sort_index].__search_v.localeCompare(a.cells[sort_index].__search_v); | |
} | |
function playpause() { | |
rate = 0; | |
if (play) { | |
play = 0; | |
doc_play.innerHTML = "Play"; | |
doc_rate.disabled = false; | |
} else { | |
delay = parseInt(doc_rate.value); | |
if (!delay || delay < 1) { | |
doc_status.innerHTML = "Not valid 'refresh' value"; | |
return; | |
} | |
play = 1; | |
doc_rate.disabled = true; | |
doc_play.innerHTML = "Pause"; | |
setTimeout("callback()", delay * 1000); | |
} | |
} | |
function refresh() { | |
if (xhr_object == null) return; | |
if (xhr_object.readyState > 0 && xhr_object.readyState < 4) { | |
return; /* request is running */ | |
} | |
xhr_object.open("GET", doc_url.value, true); | |
xhr_object.onreadystatechange = function() { | |
switch(xhr_object.readyState) { | |
case 0: | |
doc_status.innerHTML = "uninitialized"; | |
break; | |
case 1: | |
doc_status.innerHTML = "loading ..."; | |
break; | |
case 2: | |
doc_status.innerHTML = "loaded"; | |
break; | |
case 3: | |
doc_status.innerHTML = "interactive"; | |
break; | |
case 4: | |
doc_status.innerHTML = "complete"; | |
if (xhr_object.status == 200) { | |
fpm_status(xhr_object.responseText); | |
} else { | |
doc_status.innerHTML = "Error " + xhr_object.status; | |
} | |
break; | |
} | |
} | |
xhr_object.send(); | |
} | |
function callback() { | |
if (!play) return; | |
refresh(); | |
setTimeout("callback()", delay * 1000); | |
} | |
function fpm_status(txt) { | |
var json = null; | |
while (doc_short.rows.length > 0) { | |
doc_short.deleteRow(0); | |
} | |
while (doc_active.rows.length > 1) { | |
doc_active.deleteRow(1); | |
} | |
while (doc_idle.rows.length > 1) { | |
doc_idle.deleteRow(1); | |
} | |
try { | |
json = JSON.parse(txt); | |
} catch (e) { | |
doc_status.innerHTML = "Error while parsing json: '" + e + "': <br /><pre>" + txt + "</pre>"; | |
return; | |
} | |
for (var key in json) { | |
if (key == "procs") continue; | |
if (key == "state") continue; | |
var row = doc_short.insertRow(doc_short.rows.length); | |
var value = json[key]; | |
if (key == "start-time") { | |
value = new Date(value * 1000).toLocaleString(); | |
} | |
if (key == "start-since") { | |
value = time_s(value); | |
} | |
var cell = row.insertCell(row.cells.length); | |
cell.className = "e"; | |
cell.innerHTML = key; | |
cell = row.insertCell(row.cells.length); | |
cell.className = "v"; | |
cell.innerHTML = value; | |
} | |
if (json.procs) { | |
process_full(json.procs, doc_active, "Idle", 0, 0); | |
reorder(doc_active, order_active_index, order_active_reverse); | |
process_full(json.procs, doc_idle, "Idle", 1, 1); | |
reorder(doc_idle, order_idle_index, order_idle_reverse); | |
} | |
} | |
function process_full(processes, table, state, equal, cpumem) { | |
var odd = 1; | |
for (var i in processes) { | |
var proc = processes[i]; | |
if ((equal && proc.state == state) || (!equal && proc.state != state)) { | |
var c = odd++ % 2 == 0 ? "v" : "w"; | |
var row = table.insertRow(-1); | |
row.className = c; | |
row.insertCell(-1).innerHTML = proc.pid; | |
row.cells[row.cells.length - 1].__search_v = proc.pid; | |
row.cells[row.cells.length - 1].__search_t = 0; | |
row.insertCell(-1).innerHTML = date(proc['start-time'] * 1000);; | |
row.cells[row.cells.length - 1].__search_v = proc['start-time']; | |
row.cells[row.cells.length - 1].__search_t = 0; | |
row.insertCell(-1).innerHTML = time_s(proc['start-since']); | |
row.cells[row.cells.length - 1].__search_v = proc['start since']; | |
row.cells[row.cells.length - 1].__search_t = 0; | |
row.insertCell(-1).innerHTML = proc.requests; | |
row.cells[row.cells.length - 1].__search_v = proc.requests; | |
row.cells[row.cells.length - 1].__search_t = 0; | |
row.insertCell(-1).innerHTML = time_u(proc['request-duration']); | |
row.cells[row.cells.length - 1].__search_v = proc['request duration']; | |
row.cells[row.cells.length - 1].__search_t = 0; | |
row.insertCell(-1).innerHTML = proc['request-method']; | |
row.cells[row.cells.length - 1].__search_v = proc['request-method']; | |
row.cells[row.cells.length - 1].__search_t = 1; | |
row.insertCell(-1).innerHTML = proc['request-uri']; | |
row.cells[row.cells.length - 1].__search_v = proc['request-uri']; | |
row.cells[row.cells.length - 1].__search_t = 1; | |
row.insertCell(-1).innerHTML = proc['query-string']; | |
row.cells[row.cells.length - 1].__search_v = proc['query-string']; | |
row.cells[row.cells.length - 1].__search_t = 1; | |
row.insertCell(-1).innerHTML = proc['request-length']; | |
row.cells[row.cells.length - 1].__search_v = proc['request-length']; | |
row.cells[row.cells.length - 1].__search_t = 0; | |
row.insertCell(-1).innerHTML = proc.user; | |
row.cells[row.cells.length - 1].__search_v = proc.user; | |
row.cells[row.cells.length - 1].__search_t = 1; | |
row.insertCell(-1).innerHTML = proc.script; | |
row.cells[row.cells.length - 1].__search_v = proc.script; | |
row.cells[row.cells.length - 1].__search_t = 1; | |
if (cpumem) { | |
row.insertCell(-1).innerHTML = cpu(proc['last-request-cpu']); | |
row.cells[row.cells.length - 1].__search_v = proc['last-request-cpu']; | |
row.cells[row.cells.length - 1].__search_t = 0; | |
row.insertCell(-1).innerHTML = memory(proc['last-request-memory']); | |
row.cells[row.cells.length - 1].__search_v = proc['last-request-memory']; | |
row.cells[row.cells.length - 1].__search_t = 0; | |
} | |
} | |
} | |
} | |
function date(d) { | |
var t = new Date(d); | |
var r = ""; | |
r += (t.getDate() < 10 ? '0' : '') + t.getDate(); | |
r += '/'; | |
r += (t.getMonth() + 1 < 10 ? '0' : '') + (t.getMonth() + 1); | |
r += '/'; | |
r += t.getFullYear(); | |
r += ' '; | |
r += (t.getHours() < 10 ? '0' : '') + t.getHours(); | |
r += ':'; | |
r += (t.getMinutes() < 10 ? '0' : '') + t.getMinutes(); | |
r += ':'; | |
r += (t.getSeconds() < 10 ? '0' : '') + t.getSeconds(); | |
return r; | |
} | |
function cpu(c) { | |
if (c == 0) return 0; | |
return Math.round(c) + "%"; | |
} | |
function memory(mem) { | |
if (mem == 0) return 0; | |
if (mem < 1024) { | |
return mem + "B"; | |
} | |
if (mem < 1024 * 1024) { | |
return mem/1024 + "KB"; | |
} | |
if (mem < 1024*1024*1024) { | |
return mem/1024/1024 + "MB"; | |
} | |
} | |
function time_s(t) { | |
var r = ""; | |
if (t < 60) { | |
return t + 's'; | |
} | |
r = (t % 60) + 's'; | |
t = Math.floor(t / 60); | |
if (t < 60) { | |
return t + 'm ' + r; | |
} | |
r = (t % 60) + 'm ' + r; | |
t = Math.floor(t/60); | |
if (t < 24) { | |
return t + 'h ' + r; | |
} | |
return Math.floor(t/24) + 'd ' + (t % 24) + 'h ' + t; | |
} | |
function time_u(t) { | |
var r = ""; | |
if (t < 1000) { | |
return t + 'µs' | |
} | |
// r = (t % 1000) + 'µs'; | |
t = Math.floor(t / 1000); | |
if (t < 1000) { | |
return t + 'ms ' + r; | |
} | |
return time_s(Math.floor(t/1000)) + ' ' + (t%1000) + 'ms ' + r; | |
} | |
--> | |
</script> | |
</body> | |
</html> |
Is there some way to get this to give more detail about exactly which script is running? This doesn't help identify what's actually causing a slowdown. For instance, if I'm loading index.php and it's very slow, it lists "index.php" as the slow script. But obviously that's not it, it's some sub-script being called by the page that's holding everything up. This output gives me no idea what that might be.
The php-fpm status does not give that detail, however if you configure php-fpm slow log you get a more detailed trace
Add below to begining of file(after <?php
, duh!) for simple authorization.
// Use HTTP Basic Authentication for verification
$username = 'admin';
$password = 'Passw0rd!'; // Please use your own password!!
// Check if the client has provided correct credentials
if (!isset($_SERVER['PHP_AUTH_USER']) || $_SERVER['PHP_AUTH_USER'] !== $username || $_SERVER['PHP_AUTH_PW'] !== $password) {
// Send authentication request headers
header('WWW-Authenticate: Basic realm="Restricted Area"');
header('HTTP/1.0 401 Unauthorized');
echo 'Authentication failed, please provide valid credentials';
exit;
}
Ref: https://gist.github.com/Cojad/578dc0f5db71608341dd05ce86540150
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Is there some way to get this to give more detail about exactly which script is running? This doesn't help identify what's actually causing a slowdown. For instance, if I'm loading index.php and it's very slow, it lists "index.php" as the slow script. But obviously that's not it, it's some sub-script being called by the page that's holding everything up. This output gives me no idea what that might be.
I tried changing the parameter from json to html or full but then it just gives a parsing error.