1<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 2<html xmlns="http://www.w3.org/1999/xhtml"> 3<!-- 4 (c) 2011 Jerome Loyet 5 The PHP License, version 3.01 6 This is sample real-time status page for FPM. You can change it to better fit your needs. 7--> 8 <head> 9 <meta http-equiv="Content-Type" content="text/html;charset=utf-8" /> 10 <style type="text/css"> 11 body {background-color: #ffffff; color: #000000;} 12 body, td, th, h1, h2 {font-family: sans-serif;} 13 pre {margin: 0px; font-family: monospace;} 14 a:link {color: #000099; text-decoration: none; background-color: #ffffff;} 15 a:hover {text-decoration: underline;} 16 table {border-collapse: collapse;} 17 .center {text-align: center;} 18 .center table { margin-left: auto; margin-right: auto; text-align: left;} 19 .center th { text-align: center !important; } 20 td, th { border: 1px solid #000000; font-size: 75%; vertical-align: baseline;} 21 h1 {font-size: 150%;} 22 h2 {font-size: 125%;} 23 .p {text-align: left;} 24 .e {background-color: #ccccff; font-weight: bold; color: #000000;} 25 .h {background-color: #9999cc; font-weight: bold; color: #000000;} 26 27 .v {background-color: #cccccc; color: #000000;} 28 .w {background-color: #ccccff; color: #000000;} 29 30 .h th { 31 cursor: pointer; 32 } 33 img {float: right; border: 0px;} 34 hr {width: 600px; background-color: #cccccc; border: 0px; height: 1px; color: #000000;} 35 </style> 36 <title>PHP-FPM status page</title> 37 <meta name="ROBOTS" content="NOINDEX,NOFOLLOW,NOARCHIVE" /></head> 38 <body> 39 <div class="center"> 40 <table border="0" cellpadding="3" width="95%"> 41 <tr class="h"> 42 <td> 43 <a href="http://www.php.net/"><img border="0" src="" alt="PHP Logo" /></a><h1 class="p">PHP-FPM real-time status page</h1> 44 </td> 45 </tr> 46 </table> 47 <br /> 48 <table border="0" cellpadding="3" width="95%"> 49 <tr><td class="e">Status URL</td><td class="v"><input type="text" id="url" size="45" /></td></tr> 50 <tr><td class="e">Ajax status</td><td class="v" id="status"></td></tr> 51 <tr><td class="e">Refresh Rate</td><td class="v"><input type="text" id="rate" value="1" /></td></tr> 52 <tr> 53 <td class="e">Actions</td> 54 <td class="v"> 55 <button onclick="javascript:refresh();">Manual Refresh</button> 56 <button id="play" onclick="javascript:playpause();">Play</button> 57 </td> 58 </tr> 59 </table> 60 <h1>Pool Status</h1> 61 <table border="0" cellpadding="3" width="95%" id="short"> 62 <tr style="display: none;"><td> </td></tr> 63 </table> 64 <h1>Active Processes status</h1> 65 <table border="0" cellpadding="3" width="95%" id="active"> 66 <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>Content Length</th><th>User</th><th>Script</th></tr> 67 </table> 68 <h1>Idle Processes status</h1> 69 <table border="0" cellpadding="3" width="95%" id="idle"> 70 <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>Content Length</th><th>User</th><th>Script</th><th>Last Request %CPU</th><th>Last Request Memory</th></tr> 71 </table> 72 </div> 73 <p> 74 <a href="http://validator.w3.org/check?uri=referer"> 75 <img src="http://www.w3.org/Icons/valid-xhtml10" alt="Valid XHTML 1.0 Transitional" height="31" width="88" /> 76 </a> 77 </p> 78 <script type="text/javascript"> 79<!-- 80 var xhr_object = null; 81 var doc_url = document.getElementById("url"); 82 var doc_rate = document.getElementById("rate"); 83 var doc_status = document.getElementById("status"); 84 var doc_play = document.getElementById("play"); 85 var doc_short = document.getElementById("short"); 86 var doc_active = document.getElementById("active"); 87 var doc_idle = document.getElementById("idle"); 88 var rate = 0; 89 var play=0; 90 var delay = 1000; 91 var order_active_index = 0; 92 var order_active_reverse = 0; 93 var order_idle_index = 0; 94 var order_idle_reverse = 0; 95 var sort_index; 96 var sort_order; 97 98 doc_url.value = location.protocol + '//' + location.host + "/status?json&full"; 99 100 ths = document.getElementsByTagName("th"); 101 for (var i=0; i<ths.length; i++) { 102 var th = ths[i]; 103 if (th.parentNode.className == "h") { 104 th.onclick = function() { order(this); return false; }; 105 } 106 } 107 108 xhr_object = create_ajax(); 109 110 function create_ajax() { 111 if (window.XMLHttpRequest) { 112 return new XMLHttpRequest(); 113 } 114 var names = [ 115 "Msxml2.XMLHTTP.6.0", 116 "Msxml2.XMLHTTP.3.0", 117 "Msxml2.XMLHTTP", 118 "Microsoft.XMLHTTP" 119 ]; 120 for(var i in names) 121 { 122 try { 123 return new ActiveXObject(names[i]); 124 } catch(e){} 125 } 126 alert("Browser not compatible ..."); 127 } 128 129 function order(cell) { 130 var table; 131 132 if (cell.constructor != HTMLTableCellElement && cell.constructor != HTMLTableHeaderCellElement) { 133 return; 134 } 135 136 table = cell.parentNode.parentNode.parentNode; 137 138 if (table == doc_active) { 139 if (order_active_index == cell.cellIndex) { 140 if (order_active_reverse == 0) { 141 cell.innerHTML = cell.innerHTML.replace(/.$/, "↑"); 142 order_active_reverse = 1; 143 } else { 144 cell.innerHTML = cell.innerHTML.replace(/.$/, "↓"); 145 order_active_reverse = 0; 146 } 147 } else { 148 var c = doc_active.rows[0].cells[order_active_index]; 149 c.innerHTML = c.innerHTML.replace(/.$/, ""); 150 cell.innerHTML = cell.innerHTML.replace(/$/, order_active_reverse == 0 ? "↓" : "↑"); 151 order_active_index = cell.cellIndex; 152 } 153 reorder(table, order_active_index, order_active_reverse); 154 return; 155 } 156 157 if (table == doc_idle) { 158 if (order_idle_index == cell.cellIndex) { 159 if (order_idle_reverse == 0) { 160 cell.innerHTML = cell.innerHTML.replace(/.$/, "↑"); 161 order_idle_reverse = 1; 162 } else { 163 cell.innerHTML = cell.innerHTML.replace(/.$/, "↓"); 164 order_idle_reverse = 0; 165 } 166 } else { 167 var c = doc_idle.rows[0].cells[order_idle_index]; 168 c.innerHTML = c.innerHTML.replace(/.$/, ""); 169 cell.innerHTML = cell.innerHTML.replace(/$/, order_idle_reverse == 0 ? "↓" : "↑"); 170 order_idle_index = cell.cellIndex; 171 } 172 reorder(table, order_idle_index, order_idle_reverse); 173 return; 174 } 175 } 176 177 function reorder(table, index, order) { 178 var rows = []; 179 while (table.rows.length > 1) { 180 rows.push(table.rows[1]); 181 table.deleteRow(1); 182 } 183 sort_index = index; 184 sort_order = order; 185 rows.sort(sort_table); 186 for (var i in rows) { 187 table.appendChild(rows[i]); 188 } 189 var odd = 1; 190 for (var i=1; i<table.rows.length; i++) { 191 table.rows[i].className = odd++ % 2 == 0 ? "v" : "w"; 192 } 193 return; 194 } 195 196 function sort_table(a, b) { 197 if (a.cells[0].tagName == "TH") return -1; 198 if (b.cells[0].tagName == "TH") return 1; 199 200 if (a.cells[sort_index].__search_t == 0) { /* integer */ 201 if (!sort_order) return a.cells[sort_index].__search_v - b.cells[sort_index].__search_v; 202 return b.cells[sort_index].__search_v - a.cells[sort_index].__search_v;; 203 } 204 205 /* string */ 206 if (!sort_order) return a.cells[sort_index].__search_v.localeCompare(b.cells[sort_index].__search_v); 207 else return b.cells[sort_index].__search_v.localeCompare(a.cells[sort_index].__search_v); 208 } 209 210 function playpause() { 211 rate = 0; 212 if (play) { 213 play = 0; 214 doc_play.innerHTML = "Play"; 215 doc_rate.disabled = false; 216 } else { 217 delay = parseInt(doc_rate.value); 218 if (!delay || delay < 1) { 219 doc_status.innerHTML = "Not valid 'refresh' value"; 220 return; 221 } 222 play = 1; 223 doc_rate.disabled = true; 224 doc_play.innerHTML = "Pause"; 225 setTimeout("callback()", delay * 1000); 226 } 227 } 228 229 function refresh() { 230 if (xhr_object == null) return; 231 if (xhr_object.readyState > 0 && xhr_object.readyState < 4) { 232 return; /* request is running */ 233 } 234 xhr_object.open("GET", doc_url.value, true); 235 xhr_object.onreadystatechange = function() { 236 switch(xhr_object.readyState) { 237 case 0: 238 doc_status.innerHTML = "uninitialized"; 239 break; 240 case 1: 241 doc_status.innerHTML = "loading ..."; 242 break; 243 case 2: 244 doc_status.innerHTML = "loaded"; 245 break; 246 case 3: 247 doc_status.innerHTML = "interactive"; 248 break; 249 case 4: 250 doc_status.innerHTML = "complete"; 251 if (xhr_object.status == 200) { 252 fpm_status(xhr_object.responseText); 253 } else { 254 doc_status.innerHTML = "Error " + xhr_object.status; 255 } 256 break; 257 } 258 } 259 xhr_object.send(); 260 } 261 262 function callback() { 263 if (!play) return; 264 refresh(); 265 setTimeout("callback()", delay * 1000); 266 } 267 268 function fpm_status(txt) { 269 var json = null; 270 271 while (doc_short.rows.length > 0) { 272 doc_short.deleteRow(0); 273 } 274 275 while (doc_active.rows.length > 1) { 276 doc_active.deleteRow(1); 277 } 278 279 while (doc_idle.rows.length > 1) { 280 doc_idle.deleteRow(1); 281 } 282 283 try { 284 json = JSON.parse(txt); 285 } catch (e) { 286 doc_status.innerHTML = "Error while parsing json: '" + e + "': <br /><pre>" + txt + "</pre>"; 287 return; 288 } 289 290 for (var key in json) { 291 if (key == "processes") continue; 292 if (key == "state") continue; 293 var row = doc_short.insertRow(doc_short.rows.length); 294 var value = json[key]; 295 if (key == "start time") { 296 value = new Date(value * 1000).toLocaleString(); 297 } 298 if (key == "start since") { 299 value = time_s(value); 300 } 301 var cell = row.insertCell(row.cells.length); 302 cell.className = "e"; 303 cell.innerHTML = key; 304 305 cell = row.insertCell(row.cells.length); 306 cell.className = "v"; 307 cell.innerHTML = value; 308 } 309 310 if (json.processes) { 311 process_full(json.processes, doc_active, "Idle", 0, 0); 312 reorder(doc_active, order_active_index, order_active_reverse); 313 314 process_full(json.processes, doc_idle, "Idle", 1, 1); 315 reorder(doc_idle, order_idle_index, order_idle_reverse); 316 } 317 } 318 319 function process_full(processes, table, state, equal, cpumem) { 320 var odd = 1; 321 322 for (var i in processes) { 323 var proc = processes[i]; 324 if ((equal && proc.state == state) || (!equal && proc.state != state)) { 325 var c = odd++ % 2 == 0 ? "v" : "w"; 326 var row = table.insertRow(-1); 327 row.className = c; 328 row.insertCell(-1).innerHTML = proc.pid; 329 row.cells[row.cells.length - 1].__search_v = proc.pid; 330 row.cells[row.cells.length - 1].__search_t = 0; 331 332 row.insertCell(-1).innerHTML = date(proc['start time'] * 1000);; 333 row.cells[row.cells.length - 1].__search_v = proc['start time']; 334 row.cells[row.cells.length - 1].__search_t = 0; 335 336 row.insertCell(-1).innerHTML = time_s(proc['start since']); 337 row.cells[row.cells.length - 1].__search_v = proc['start since']; 338 row.cells[row.cells.length - 1].__search_t = 0; 339 340 row.insertCell(-1).innerHTML = proc.requests; 341 row.cells[row.cells.length - 1].__search_v = proc.requests; 342 row.cells[row.cells.length - 1].__search_t = 0; 343 344 row.insertCell(-1).innerHTML = time_u(proc['request duration']); 345 row.cells[row.cells.length - 1].__search_v = proc['request duration']; 346 row.cells[row.cells.length - 1].__search_t = 0; 347 348 row.insertCell(-1).innerHTML = proc['request method']; 349 row.cells[row.cells.length - 1].__search_v = proc['request method']; 350 row.cells[row.cells.length - 1].__search_t = 1; 351 352 row.insertCell(-1).innerHTML = proc['request uri']; 353 row.cells[row.cells.length - 1].__search_v = proc['request uri']; 354 row.cells[row.cells.length - 1].__search_t = 1; 355 356 row.insertCell(-1).innerHTML = proc['content length']; 357 row.cells[row.cells.length - 1].__search_v = proc['content length']; 358 row.cells[row.cells.length - 1].__search_t = 0; 359 360 row.insertCell(-1).innerHTML = proc.user; 361 row.cells[row.cells.length - 1].__search_v = proc.user; 362 row.cells[row.cells.length - 1].__search_t = 1; 363 364 row.insertCell(-1).innerHTML = proc.script; 365 row.cells[row.cells.length - 1].__search_v = proc.script; 366 row.cells[row.cells.length - 1].__search_t = 1; 367 368 if (cpumem) { 369 row.insertCell(-1).innerHTML = cpu(proc['last request cpu']); 370 row.cells[row.cells.length - 1].__search_v = proc['last request cpu']; 371 row.cells[row.cells.length - 1].__search_t = 0; 372 373 row.insertCell(-1).innerHTML = memory(proc['last request memory']); 374 row.cells[row.cells.length - 1].__search_v = proc['last request memory']; 375 row.cells[row.cells.length - 1].__search_t = 0; 376 } 377 } 378 } 379 } 380 381 function date(d) { 382 var t = new Date(d); 383 var r = ""; 384 385 r += (t.getDate() < 10 ? '0' : '') + t.getDate(); 386 r += '/'; 387 r += (t.getMonth() + 1 < 10 ? '0' : '') + (t.getMonth() + 1); 388 r += '/'; 389 r += t.getFullYear(); 390 r += ' '; 391 r += (t.getHours() < 10 ? '0' : '') + t.getHours(); 392 r += ':'; 393 r += (t.getMinutes() < 10 ? '0' : '') + t.getMinutes(); 394 r += ':'; 395 r += (t.getSeconds() < 10 ? '0' : '') + t.getSeconds(); 396 397 398 return r; 399 } 400 401 function cpu(c) { 402 if (c == 0) return 0; 403 return c + "%"; 404 } 405 406 function memory(mem) { 407 if (mem == 0) return 0; 408 if (mem < 1024) { 409 return mem + "B"; 410 } 411 if (mem < 1024 * 1024) { 412 return mem/1024 + "KB"; 413 } 414 if (mem < 1024*1024*1024) { 415 return mem/1024/1024 + "MB"; 416 } 417 } 418 419 function time_s(t) { 420 var r = ""; 421 if (t < 60) { 422 return t + 's'; 423 } 424 425 r = (t % 60) + 's'; 426 t = Math.floor(t / 60); 427 if (t < 60) { 428 return t + 'm ' + r; 429 } 430 431 r = (t % 60) + 'm ' + r; 432 t = Math.floor(t/60); 433 434 if (t < 24) { 435 return t + 'h ' + r; 436 } 437 438 return Math.floor(t/24) + 'd ' + (t % 24) + 'h ' + t; 439 } 440 441 function time_u(t) { 442 var r = ""; 443 if (t < 1000) { 444 return t + 'µs' 445 } 446 447 r = (t % 1000) + 'µs'; 448 t = Math.floor(t / 1000); 449 if (t < 1000) { 450 return t + 'ms ' + r; 451 } 452 453 return time_s(Math.floor(t/1000)) + ' ' + (t%1000) + 'ms ' + r; 454 } 455--> 456 </script> 457 </body> 458</html> 459