xref: /PHP-8.3/sapi/fpm/status.html.in (revision f89fe830)
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>&nbsp;</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&darr;</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&darr;</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(/.$/, "&uarr;");
142							order_active_reverse = 1;
143						} else {
144							cell.innerHTML = cell.innerHTML.replace(/.$/, "&darr;");
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 ? "&darr;" : "&uarr;");
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(/.$/, "&uarr;");
161							order_idle_reverse = 1;
162						} else {
163							cell.innerHTML = cell.innerHTML.replace(/.$/, "&darr;");
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 ? "&darr;" : "&uarr;");
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 (isNaN(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 + '&micro;s'
445				}
446
447				r = (t % 1000) + '&micro;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