mirror of
https://gitlab.isc.org/isc-projects/bind9
synced 2025-09-01 06:55:30 +00:00
[master] stats improvements
3700. [func] Allow access to subgroups of XML statistics via special URLs http://<server>:<port>/xml/v3/server, /zones, /net, /tasks, /mem, and /status. [RT #35115] 3699. [bug] Improvements to statistics channel XSL stylesheet: the stylesheet can now be cached by the browser; section headers are omitted from the stats display when there is no data in those sections to be displayed; counters are now right-justified for easier readability. [RT #35117]
This commit is contained in:
11
CHANGES
11
CHANGES
@@ -1,3 +1,14 @@
|
||||
3700. [func] Allow access to subgroups of XML statistics via
|
||||
special URLs http://<server>:<port>/xml/v3/server,
|
||||
/zones, /net, /tasks, /mem, and /status. [RT #35115]
|
||||
|
||||
3699. [bug] Improvements to statistics channel XSL stylesheet:
|
||||
the stylesheet can now be cached by the browser;
|
||||
section headers are omitted from the stats display
|
||||
when there is no data in those sections to be
|
||||
displayed; counters are now right-justified for
|
||||
easier readability. [RT #35117]
|
||||
|
||||
3698. [cleanup] Replaced all uses of memcpy() with memmove().
|
||||
[RT #35120]
|
||||
|
||||
|
@@ -20,7 +20,7 @@
|
||||
<!-- %Id: bind9.xsl,v 1.21 2009/01/27 23:47:54 tbox Exp % -->
|
||||
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns="http://www.w3.org/1999/xhtml" version="1.0">
|
||||
<xsl:output method="html" indent="yes" version="4.0"/>
|
||||
<xsl:template match="statistics[@version="3.1"]">
|
||||
<xsl:template match="statistics[@version="3.4"]">
|
||||
<html>
|
||||
<head>
|
||||
<xsl:if test="system-property('xsl:vendor')!='Transformiix'">
|
||||
@@ -33,47 +33,52 @@
|
||||
|
||||
var graphs=[];
|
||||
|
||||
function drawChart(chart_title,target,data) {
|
||||
function drawChart(chart_title,target,style,data) {
|
||||
var data = google.visualization.arrayToDataTable(data);
|
||||
|
||||
var options = {
|
||||
title: chart_title
|
||||
};
|
||||
|
||||
var chart = new google.visualization.BarChart(document.getElementById(target));
|
||||
var chart;
|
||||
if (style == "barchart") {
|
||||
chart = new google.visualization.BarChart(document.getElementById(target));
|
||||
chart.draw(data, options);
|
||||
} else if (style == "piechart") {
|
||||
chart = new google.visualization.PieChart(document.getElementById(target));
|
||||
chart.draw(data, options);
|
||||
}
|
||||
}
|
||||
|
||||
function loadGraphs(){
|
||||
//alert("here we are!");
|
||||
var g;
|
||||
|
||||
// Server Incoming query Types
|
||||
while(g = graphs.shift()){
|
||||
// alert("going for: " + g.target);
|
||||
if(g.data.length > 1){
|
||||
drawChart(g.title,g.target,g.data);
|
||||
drawChart(g.title,g.target,g.style,g.data);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Server Incoming Queries Types
|
||||
<xsl:if test="server/counters[@type="qtype"]/counter">
|
||||
// Server Incoming Query Types
|
||||
graphs.push({
|
||||
'title' : "Server Incoming Query Types",
|
||||
'target': 'chart_incoming_qtypes',
|
||||
'style': 'barchart',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="server/counters[@type="qtype"]/counter">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
});
|
||||
</xsl:if>
|
||||
|
||||
|
||||
// Server Incoming Requests
|
||||
<xsl:if test="server/counters[@type="opcode"]/counter">
|
||||
// Server Incoming Requests by opcode
|
||||
graphs.push({
|
||||
'title' : "Server Incoming Requests",
|
||||
'target': 'chart_incoming_requests',
|
||||
'data': [['Requests','Counter'],<xsl:for-each select="server/counters[@type="opcode"]/counter">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]});
|
||||
|
||||
|
||||
|
||||
|
||||
'title' : "Server Incoming Requests by DNS Opcode",
|
||||
'target': 'chart_incoming_opcodes',
|
||||
'style': 'barchart',
|
||||
'data': [['Opcode','Counter'],<xsl:for-each select="server/counters[@type="opcode"]/counter[. > 0 or substring(@name,1,3) != 'RES']">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]});
|
||||
</xsl:if>
|
||||
</script>
|
||||
</xsl:if>
|
||||
<style type="text/css">
|
||||
@@ -106,18 +111,98 @@
|
||||
border: 1px solid grey;
|
||||
width: 500px;
|
||||
}
|
||||
|
||||
table.counters th {
|
||||
text-align: center;
|
||||
text-align: right;
|
||||
border: 1px solid grey;
|
||||
width: 120px;
|
||||
width: 150px;
|
||||
}
|
||||
table.counters td {
|
||||
text-align:center;
|
||||
|
||||
text-align: right;
|
||||
font-family: monospace;
|
||||
}
|
||||
table.counters tr:hover{
|
||||
background-color: #99ddff;
|
||||
}
|
||||
|
||||
table.counters tr:hover{
|
||||
table.info {
|
||||
border: 1px solid grey;
|
||||
width: 500px;
|
||||
}
|
||||
table.info th {
|
||||
text-align: center;
|
||||
border: 1px solid grey;
|
||||
width: 150px;
|
||||
}
|
||||
table.info td {
|
||||
text-align: center;
|
||||
}
|
||||
table.info tr:hover{
|
||||
background-color: #99ddff;
|
||||
}
|
||||
|
||||
table.tasks {
|
||||
border: 1px solid grey;
|
||||
width: 500px;
|
||||
}
|
||||
table.tasks th {
|
||||
text-align: center;
|
||||
border: 1px solid grey;
|
||||
width: 150px;
|
||||
}
|
||||
table.tasks td {
|
||||
text-align: right;
|
||||
font-family: monospace;
|
||||
}
|
||||
table.tasks td:nth-child(2) {
|
||||
text-align: center;
|
||||
}
|
||||
table.tasks td:nth-child(4) {
|
||||
text-align: center;
|
||||
}
|
||||
table.tasks tr:hover{
|
||||
background-color: #99ddff;
|
||||
}
|
||||
|
||||
table.netstat {
|
||||
border: 1px solid grey;
|
||||
width: 500px;
|
||||
}
|
||||
table.netstat th {
|
||||
text-align: center;
|
||||
border: 1px solid grey;
|
||||
width: 150px;
|
||||
}
|
||||
table.netstat td {
|
||||
text-align: center;
|
||||
}
|
||||
table.netstat td:nth-child(4) {
|
||||
text-align: right;
|
||||
font-family: monospace;
|
||||
}
|
||||
table.netstat td:nth-child(7) {
|
||||
text-align: left;
|
||||
}
|
||||
table.netstat tr:hover{
|
||||
background-color: #99ddff;
|
||||
}
|
||||
|
||||
table.mctx {
|
||||
border: 1px solid grey;
|
||||
width: 500px;
|
||||
}
|
||||
table.mctx th {
|
||||
text-align: center;
|
||||
border: 1px solid grey;
|
||||
}
|
||||
table.mctx td {
|
||||
text-align: right;
|
||||
font-family: monospace;
|
||||
}
|
||||
table.mctx td:nth-child(-n+2) {
|
||||
text-align: left;
|
||||
width: 100px;
|
||||
}
|
||||
table.mctx tr:hover{
|
||||
background-color: #99ddff;
|
||||
}
|
||||
|
||||
@@ -166,14 +251,12 @@
|
||||
font-size: 12pt;
|
||||
width:500px;
|
||||
text-align:center;
|
||||
|
||||
}
|
||||
h4 {
|
||||
color: rgb(1,169,206);
|
||||
font-size: 10pt;
|
||||
width:500px;
|
||||
text-align:center;
|
||||
|
||||
}
|
||||
|
||||
.pie {
|
||||
@@ -189,8 +272,8 @@
|
||||
<h1>ISC Bind 9 Configuration and Statistics</h1>
|
||||
</div>
|
||||
<hr/>
|
||||
<h2>Server Times</h2>
|
||||
<table class="counters">
|
||||
<h2>Server Status</h2>
|
||||
<table class="info">
|
||||
<tr>
|
||||
<th>Boot time:</th>
|
||||
<td>
|
||||
@@ -211,13 +294,16 @@
|
||||
</tr>
|
||||
</table>
|
||||
<br/>
|
||||
<h2>Incoming Requests</h2>
|
||||
<xsl:if test="server/counters[@type="opcode"]/counter[. > 0]">
|
||||
<xsl:if test="system-property('xsl:vendor')!='Transformiix'">
|
||||
<h2>Incoming Requests by DNS Opcode</h2>
|
||||
<!-- Non Mozilla specific markup -->
|
||||
<div class="pie" id="chart_incoming_requests">[no incoming requests]</div>
|
||||
<div class="pie" id="chart_incoming_opcodes">
|
||||
[cannot display chart]
|
||||
</div>
|
||||
</xsl:if>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="server/counters[@type="opcode"]/counter">
|
||||
<xsl:for-each select="server/counters[@type="opcode"]/counter[. > 0 or substring(@name,1,3) != 'RES']">
|
||||
<xsl:sort select="." data-type="number" order="descending"/>
|
||||
<tr>
|
||||
<th>
|
||||
@@ -236,10 +322,14 @@
|
||||
</tr>
|
||||
</table>
|
||||
<br/>
|
||||
<h3>Incoming Queries by Type</h3>
|
||||
</xsl:if>
|
||||
<xsl:if test="server/counters[@type="qtype"]/counter">
|
||||
<xsl:if test="system-property('xsl:vendor')!='Transformiix'">
|
||||
<!-- Non Mozilla specific markup -->
|
||||
<div class="pie" id="chart_incoming_qtypes">[no incoming queries]</div>
|
||||
<h3>Incoming Queries by Query Type</h3>
|
||||
<div class="pie" id="chart_incoming_qtypes">
|
||||
[cannot display chart]
|
||||
</div>
|
||||
</xsl:if>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="server/counters[@type="qtype"]/counter">
|
||||
@@ -267,6 +357,8 @@
|
||||
</tr>
|
||||
</table>
|
||||
<br/>
|
||||
</xsl:if>
|
||||
<xsl:if test="views/view[count(counters[@type="resqtype"]/counter) > 0]">
|
||||
<h2>Outgoing Queries per view</h2>
|
||||
<xsl:for-each select="views/view[count(counters[@type="resqtype"]/counter) > 0]">
|
||||
<h3>View <xsl:value-of select="@name"/></h3>
|
||||
@@ -274,16 +366,16 @@
|
||||
<!-- Non Mozilla specific markup -->
|
||||
<script type="text/javascript">
|
||||
graphs.push({
|
||||
'title': "Outgoing queries for view: <xsl:value-of select="@name"/>",
|
||||
'title': "Outgoing Queries for view: <xsl:value-of select="@name"/>",
|
||||
'target': 'chart_outgoing_queries_view_<xsl:value-of select="@name"/>',
|
||||
'style': 'barchart',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="counters[@type="resqtype"]/counter">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
});
|
||||
|
||||
</script>
|
||||
<xsl:variable name="target">
|
||||
<xsl:value-of select="@name"/>
|
||||
</xsl:variable>
|
||||
<div class="pie" id="chart_outgoing_queries_view_{$target}"/>
|
||||
<div class="pie" id="chart_outgoing_queries_view_{$target}">[no data to display]</div>
|
||||
</xsl:if>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="counters[@type="resqtype"]/counter">
|
||||
@@ -306,6 +398,8 @@
|
||||
</table>
|
||||
<br/>
|
||||
</xsl:for-each>
|
||||
</xsl:if>
|
||||
<xsl:if test="server/counters[@type="nsstat"]/counter[.>0]">
|
||||
<h2>Server Statistics</h2>
|
||||
<xsl:if test="system-property('xsl:vendor')!='Transformiix'">
|
||||
<!-- Non Mozilla specific markup -->
|
||||
@@ -313,11 +407,11 @@
|
||||
graphs.push({
|
||||
'title' : "Server Counters",
|
||||
'target': 'chart_server_nsstat_restype',
|
||||
'style': 'barchart',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="server/counters[@type="nsstat"]/counter[.>0]">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
});
|
||||
|
||||
</script>
|
||||
<div class="pie" id="chart_server_nsstat_restype"/>
|
||||
<div class="pie" id="chart_server_nsstat_restype">[no data to display]</div>
|
||||
</xsl:if>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="server/counters[@type="nsstat"]/counter[.>0]">
|
||||
@@ -339,18 +433,20 @@
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
<br/>
|
||||
<h2>Zone Maintenance Statistics</h2>
|
||||
</xsl:if>
|
||||
<xsl:if test="server/counters[@type="zonestat"]/counter[.>0]">
|
||||
<xsl:if test="system-property('xsl:vendor')!='Transformiix'">
|
||||
<h2>Zone Maintenance Statistics</h2>
|
||||
<script type="text/javascript">
|
||||
graphs.push({
|
||||
'title' : "Zone Maintenance Stats",
|
||||
'target': 'chart_server_zone_maint',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="server/counters[@type="zonestat"]/counter">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
'style': 'barchart',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="server/counters[@type="zonestat"]/counter[.>0]">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
});
|
||||
|
||||
</script>
|
||||
<!-- Non Mozilla specific markup -->
|
||||
<div class="pie" id="chart_server_zone_maint"/>
|
||||
<div class="pie" id="chart_server_zone_maint">[no data to display]</div>
|
||||
</xsl:if>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="server/counters[@type="zonestat"]/counter">
|
||||
@@ -371,6 +467,8 @@
|
||||
</tr>
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
</xsl:if>
|
||||
<xsl:if test="server/counters[@type="resstat"]/counter[.>0]">
|
||||
<h2>Resolver Statistics (Common)</h2>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="server/counters[@type="resstat"]/counter">
|
||||
@@ -391,7 +489,9 @@
|
||||
</tr>
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
</xsl:if>
|
||||
<xsl:for-each select="views/view">
|
||||
<xsl:if test="counters[@type="resstats"]/counter[.>0]">
|
||||
<h3>Resolver Statistics for View <xsl:value-of select="@name"/></h3>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="counters[@type="resstats"]/counter[.>0]">
|
||||
@@ -412,10 +512,10 @@
|
||||
</tr>
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
</xsl:if>
|
||||
</xsl:for-each>
|
||||
|
||||
|
||||
<xsl:for-each select="views/view">
|
||||
<xsl:if test="counters[@type="adbstat"]/counter[.>0]">
|
||||
<h3>ADB Statistics for View <xsl:value-of select="@name"/></h3>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="counters[@type="adbstat"]/counter[.>0]">
|
||||
@@ -436,9 +536,11 @@
|
||||
</tr>
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
</xsl:if>
|
||||
</xsl:for-each>
|
||||
|
||||
<xsl:for-each select="views/view">
|
||||
<xsl:if test="counters[@type="cachestats"]/counter[.>0]">
|
||||
<h3>Cache Statistics for View <xsl:value-of select="@name"/></h3>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="counters[@type="cachestats"]/counter[.>0]">
|
||||
@@ -459,11 +561,12 @@
|
||||
</tr>
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
</xsl:if>
|
||||
</xsl:for-each>
|
||||
|
||||
|
||||
<h3>Cache DB RRsets for View <xsl:value-of select="@name"/></h3>
|
||||
<xsl:for-each select="views/view">
|
||||
<xsl:if test="cache/rrset">
|
||||
<h3>Cache DB RRsets for View <xsl:value-of select="@name"/></h3>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="cache/rrset">
|
||||
<xsl:variable name="css-class6">
|
||||
@@ -483,7 +586,10 @@
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
<br/>
|
||||
</xsl:if>
|
||||
</xsl:for-each>
|
||||
|
||||
<xsl:if test="server/counters[@type="sockstat"]/counter[.>0]">
|
||||
<h2>Socket I/O Statistics</h2>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="server/counters[@type="sockstat"]/counter[.>0]">
|
||||
@@ -504,33 +610,35 @@
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
<br/>
|
||||
<br/>
|
||||
<h2>Response Codes per view/zone</h2>
|
||||
<xsl:for-each select="views/view[zones/zone/counters[@type="rcode"]/counter >0]">
|
||||
</xsl:if>
|
||||
<xsl:if test="views/view[zones/zone/counters[@type="qtype"]/counter >0]">
|
||||
<h2>Received QTYPES per view/zone</h2>
|
||||
<xsl:for-each select="views/view[zones/zone/counters[@type="qtype"]/counter >0]">
|
||||
<h3>View <xsl:value-of select="@name"/></h3>
|
||||
<xsl:variable name="thisview">
|
||||
<xsl:value-of select="@name"/>
|
||||
</xsl:variable>
|
||||
<xsl:for-each select="zones/zone">
|
||||
<xsl:if test="counters[@type="rcode"]/counter[. > 0]">
|
||||
<xsl:if test="counters[@type="qtype"]/counter[count(.) > 0]">
|
||||
<h4>Zone <xsl:value-of select="@name"/></h4>
|
||||
<xsl:if test="system-property('xsl:vendor')!='Transformiix'">
|
||||
<!-- Non Mozilla specific markup -->
|
||||
<script type="text/javascript">
|
||||
graphs.push({
|
||||
'title': "Response Codes for zone <xsl:value-of select="@name"/>",
|
||||
'target': 'chart_rescode_<xsl:value-of select="../../@name"/>_<xsl:value-of select="@name"/>',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="counters[@type="rcode"]/counter[.>0 and @name != "QryAuthAns"]">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
'title': "Query types for zone <xsl:value-of select="@name"/>",
|
||||
'target': 'chart_qtype_<xsl:value-of select="../../@name"/>_<xsl:value-of select="@name"/>',
|
||||
'style': 'barchart',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="counters[@type="qtype"]/counter[.>0 and @name != "QryAuthAns"]">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
});
|
||||
|
||||
</script>
|
||||
<xsl:variable name="target">
|
||||
<xsl:value-of select="@name"/>
|
||||
</xsl:variable>
|
||||
<div class="pie" id="chart_rescode_{$thisview}_{$target}"/>
|
||||
<div class="pie" id="chart_qtype_{$thisview}_{$target}">[no data to display]</div>
|
||||
</xsl:if>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="counters[@type="rcode"]/counter[.>0 and @name != "QryAuthAns"]">
|
||||
<xsl:for-each select="counters[@type="qtype"]/counter">
|
||||
<xsl:sort select="."/>
|
||||
<xsl:variable name="css-class10">
|
||||
<xsl:choose>
|
||||
@@ -551,32 +659,35 @@
|
||||
</xsl:if>
|
||||
</xsl:for-each>
|
||||
</xsl:for-each>
|
||||
<h2>Received QTYPES per view/zone</h2>
|
||||
<xsl:for-each select="views/view[zones/zone/counters[@type="qtype"]/counter >0]">
|
||||
</xsl:if>
|
||||
<xsl:if test="views/view[zones/zone/counters[@type="rcode"]/counter >0]">
|
||||
<h2>Response Codes per view/zone</h2>
|
||||
<xsl:for-each select="views/view[zones/zone/counters[@type="rcode"]/counter >0]">
|
||||
<h3>View <xsl:value-of select="@name"/></h3>
|
||||
<xsl:variable name="thisview2">
|
||||
<xsl:value-of select="@name"/>
|
||||
</xsl:variable>
|
||||
<xsl:for-each select="zones/zone">
|
||||
<xsl:if test="counters[@type="qtype"]/counter[count(.) > 0]">
|
||||
<xsl:if test="counters[@type="rcode"]/counter[. > 0]">
|
||||
<h4>Zone <xsl:value-of select="@name"/></h4>
|
||||
<xsl:if test="system-property('xsl:vendor')!='Transformiix'">
|
||||
<!-- Non Mozilla specific markup -->
|
||||
<script type="text/javascript">
|
||||
graphs.push({
|
||||
'title': "Query Types for zone <xsl:value-of select="@name"/>",
|
||||
'target': 'chart_qtype_<xsl:value-of select="../../@name"/>_<xsl:value-of select="@name"/>',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="counters[@type="qtype"]/counter[.>0 and @name != "QryAuthAns"]">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
'title': "Response codes for zone <xsl:value-of select="@name"/>",
|
||||
'target': 'chart_rescode_<xsl:value-of select="../../@name"/>_<xsl:value-of select="@name"/>',
|
||||
'style': 'barchart',
|
||||
'data': [['Type','Counter'],<xsl:for-each select="counters[@type="rcode"]/counter[.>0 and @name != "QryAuthAns"]">['<xsl:value-of select="@name"/>',<xsl:value-of select="."/>],</xsl:for-each>]
|
||||
});
|
||||
|
||||
</script>
|
||||
<xsl:variable name="target">
|
||||
<xsl:value-of select="@name"/>
|
||||
</xsl:variable>
|
||||
<div class="pie" id="chart_qtype_{$thisview2}_{$target}"/>
|
||||
<div class="pie" id="chart_rescode_{$thisview2}_{$target}">[no data to display]</div>
|
||||
</xsl:if>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="counters[@type="qtype"]/counter">
|
||||
<xsl:for-each select="counters[@type="rcode"]/counter[.>0 and @name != "QryAuthAns"]">
|
||||
<xsl:sort select="."/>
|
||||
<xsl:variable name="css-class11">
|
||||
<xsl:choose>
|
||||
@@ -597,8 +708,10 @@
|
||||
</xsl:if>
|
||||
</xsl:for-each>
|
||||
</xsl:for-each>
|
||||
</xsl:if>
|
||||
<xsl:if test="socketmgr/sockets/socket">
|
||||
<h2>Network Status</h2>
|
||||
<table class="counters">
|
||||
<table class="netstat">
|
||||
<tr>
|
||||
<th>ID</th>
|
||||
<th>Name</th>
|
||||
@@ -644,6 +757,8 @@
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
<br/>
|
||||
</xsl:if>
|
||||
<xsl:if test="taskmgr/thread-model/type">
|
||||
<h2>Task Manager Configuration</h2>
|
||||
<table class="counters">
|
||||
<tr>
|
||||
@@ -670,7 +785,7 @@
|
||||
<xsl:value-of select="taskmgr/thread-model/tasks-running"/>
|
||||
</td>
|
||||
</tr>
|
||||
<tr class="odd">
|
||||
<tr class="even">
|
||||
<th>Tasks Ready</th>
|
||||
<td>
|
||||
<xsl:value-of select="taskmgr/thread-model/tasks-ready"/>
|
||||
@@ -678,8 +793,10 @@
|
||||
</tr>
|
||||
</table>
|
||||
<br/>
|
||||
</xsl:if>
|
||||
<xsl:if test="taskmgr/tasks/task">
|
||||
<h2>Tasks</h2>
|
||||
<table class="counters">
|
||||
<table class="tasks">
|
||||
<tr>
|
||||
<th>ID</th>
|
||||
<th>Name</th>
|
||||
@@ -719,6 +836,8 @@
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
<br/>
|
||||
</xsl:if>
|
||||
<xsl:if test="memory/summary">
|
||||
<h2>Memory Usage Summary</h2>
|
||||
<table class="counters">
|
||||
<xsl:for-each select="memory/summary/*">
|
||||
@@ -739,8 +858,10 @@
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
<br/>
|
||||
</xsl:if>
|
||||
<xsl:if test="memory/contexts/context">
|
||||
<h2>Memory Contexts</h2>
|
||||
<table class="counters">
|
||||
<table class="mctx">
|
||||
<tr>
|
||||
<th>ID</th>
|
||||
<th>Name</th>
|
||||
@@ -795,6 +916,7 @@
|
||||
</tr>
|
||||
</xsl:for-each>
|
||||
</table>
|
||||
</xsl:if>
|
||||
<hr/>
|
||||
<p class="footer">Internet Systems Consortium Inc.<br/><a href="http://www.isc.org">http://www.isc.org</a></p>
|
||||
</body>
|
||||
|
@@ -38,47 +38,52 @@ static char xslmsg[] =
|
||||
"\n"
|
||||
" var graphs=[];\n"
|
||||
"\n"
|
||||
" function drawChart(chart_title,target,data) {\n"
|
||||
" function drawChart(chart_title,target,style,data) {\n"
|
||||
" var data = google.visualization.arrayToDataTable(data);\n"
|
||||
"\n"
|
||||
" var options = {\n"
|
||||
" title: chart_title\n"
|
||||
" };\n"
|
||||
"\n"
|
||||
" var chart = new google.visualization.BarChart(document.getElementById(target));\n"
|
||||
" var chart;\n"
|
||||
" if (style == \"barchart\") {\n"
|
||||
" chart = new google.visualization.BarChart(document.getElementById(target));\n"
|
||||
" chart.draw(data, options);\n"
|
||||
" } else if (style == \"piechart\") {\n"
|
||||
" chart = new google.visualization.PieChart(document.getElementById(target));\n"
|
||||
" chart.draw(data, options);\n"
|
||||
" }\n"
|
||||
" }\n"
|
||||
"\n"
|
||||
" function loadGraphs(){\n"
|
||||
" //alert(\"here we are!\");\n"
|
||||
" var g;\n"
|
||||
"\n"
|
||||
" // Server Incoming query Types\n"
|
||||
" while(g = graphs.shift()){\n"
|
||||
" // alert(\"going for: \" + g.target);\n"
|
||||
" if(g.data.length > 1){\n"
|
||||
" drawChart(g.title,g.target,g.data);\n"
|
||||
" drawChart(g.title,g.target,g.style,g.data);\n"
|
||||
" }\n"
|
||||
" }\n"
|
||||
" }\n"
|
||||
"\n"
|
||||
" // Server Incoming Queries Types \n"
|
||||
" <xsl:if test=\"server/counters[@type="qtype"]/counter\">\n"
|
||||
" // Server Incoming Query Types\n"
|
||||
" graphs.push({\n"
|
||||
" 'title' : \"Server Incoming Query Types\",\n"
|
||||
" 'target': 'chart_incoming_qtypes',\n"
|
||||
" 'style': 'barchart',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"server/counters[@type="qtype"]/counter\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" });\n"
|
||||
" </xsl:if>\n"
|
||||
"\n"
|
||||
"\n"
|
||||
" // Server Incoming Requests \n"
|
||||
" <xsl:if test=\"server/counters[@type="opcode"]/counter\">\n"
|
||||
" // Server Incoming Requests by opcode\n"
|
||||
" graphs.push({\n"
|
||||
" 'title' : \"Server Incoming Requests\",\n"
|
||||
" 'target': 'chart_incoming_requests',\n"
|
||||
" 'data': [['Requests','Counter'],<xsl:for-each select=\"server/counters[@type="opcode"]/counter\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]});\n"
|
||||
" \n"
|
||||
" \n"
|
||||
" \n"
|
||||
" \n"
|
||||
" 'title' : \"Server Incoming Requests by DNS Opcode\",\n"
|
||||
" 'target': 'chart_incoming_opcodes',\n"
|
||||
" 'style': 'barchart',\n"
|
||||
" 'data': [['Opcode','Counter'],<xsl:for-each select=\"server/counters[@type="opcode"]/counter[. > 0 or substring(@name,1,3) != 'RES']\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]});\n"
|
||||
" </xsl:if>\n"
|
||||
" </script>\n"
|
||||
" </xsl:if>\n"
|
||||
" <style type=\"text/css\">\n"
|
||||
@@ -111,18 +116,98 @@ static char xslmsg[] =
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 500px;\n"
|
||||
" }\n"
|
||||
" \n"
|
||||
" table.counters th {\n"
|
||||
" text-align: center;\n"
|
||||
" text-align: right;\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 120px;\n"
|
||||
" width: 150px;\n"
|
||||
" }\n"
|
||||
" table.counters td {\n"
|
||||
" text-align:center;\n"
|
||||
" \n"
|
||||
" text-align: right;\n"
|
||||
" font-family: monospace;\n"
|
||||
" }\n"
|
||||
" table.counters tr:hover{\n"
|
||||
" background-color: #99ddff;\n"
|
||||
" }\n"
|
||||
"\n"
|
||||
" table.counters tr:hover{\n"
|
||||
" table.info {\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 500px;\n"
|
||||
" }\n"
|
||||
" table.info th {\n"
|
||||
" text-align: center;\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 150px;\n"
|
||||
" }\n"
|
||||
" table.info td {\n"
|
||||
" text-align: center;\n"
|
||||
" }\n"
|
||||
" table.info tr:hover{\n"
|
||||
" background-color: #99ddff;\n"
|
||||
" }\n"
|
||||
"\n"
|
||||
" table.tasks {\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 500px;\n"
|
||||
" }\n"
|
||||
" table.tasks th {\n"
|
||||
" text-align: center;\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 150px;\n"
|
||||
" }\n"
|
||||
" table.tasks td {\n"
|
||||
" text-align: right;\n"
|
||||
" font-family: monospace;\n"
|
||||
" }\n"
|
||||
" table.tasks td:nth-child(2) {\n"
|
||||
" text-align: center;\n"
|
||||
" }\n"
|
||||
" table.tasks td:nth-child(4) {\n"
|
||||
" text-align: center;\n"
|
||||
" }\n"
|
||||
" table.tasks tr:hover{\n"
|
||||
" background-color: #99ddff;\n"
|
||||
" }\n"
|
||||
"\n"
|
||||
" table.netstat {\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 500px;\n"
|
||||
" }\n"
|
||||
" table.netstat th {\n"
|
||||
" text-align: center;\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 150px;\n"
|
||||
" }\n"
|
||||
" table.netstat td {\n"
|
||||
" text-align: center;\n"
|
||||
" }\n"
|
||||
" table.netstat td:nth-child(4) {\n"
|
||||
" text-align: right;\n"
|
||||
" font-family: monospace;\n"
|
||||
" }\n"
|
||||
" table.netstat td:nth-child(7) {\n"
|
||||
" text-align: left;\n"
|
||||
" }\n"
|
||||
" table.netstat tr:hover{\n"
|
||||
" background-color: #99ddff;\n"
|
||||
" }\n"
|
||||
"\n"
|
||||
" table.mctx {\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" width: 500px;\n"
|
||||
" }\n"
|
||||
" table.mctx th {\n"
|
||||
" text-align: center;\n"
|
||||
" border: 1px solid grey;\n"
|
||||
" }\n"
|
||||
" table.mctx td {\n"
|
||||
" text-align: right;\n"
|
||||
" font-family: monospace;\n"
|
||||
" }\n"
|
||||
" table.mctx td:nth-child(-n+2) {\n"
|
||||
" text-align: left;\n"
|
||||
" width: 100px;\n"
|
||||
" }\n"
|
||||
" table.mctx tr:hover{\n"
|
||||
" background-color: #99ddff;\n"
|
||||
" }\n"
|
||||
"\n"
|
||||
@@ -171,14 +256,12 @@ static char xslmsg[] =
|
||||
" font-size: 12pt;\n"
|
||||
" width:500px;\n"
|
||||
" text-align:center;\n"
|
||||
" \n"
|
||||
" }\n"
|
||||
" h4 {\n"
|
||||
" color: rgb(1,169,206);\n"
|
||||
" font-size: 10pt;\n"
|
||||
" width:500px;\n"
|
||||
" text-align:center;\n"
|
||||
" \n"
|
||||
" }\n"
|
||||
"\n"
|
||||
" .pie {\n"
|
||||
@@ -194,8 +277,8 @@ static char xslmsg[] =
|
||||
" <h1>ISC Bind 9 Configuration and Statistics</h1>\n"
|
||||
" </div>\n"
|
||||
" <hr/>\n"
|
||||
" <h2>Server Times</h2>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <h2>Server Status</h2>\n"
|
||||
" <table class=\"info\">\n"
|
||||
" <tr>\n"
|
||||
" <th>Boot time:</th>\n"
|
||||
" <td>\n"
|
||||
@@ -216,13 +299,16 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" <h2>Incoming Requests</h2>\n"
|
||||
" <xsl:if test=\"server/counters[@type="opcode"]/counter[. > 0]\">\n"
|
||||
" <xsl:if test=\"system-property('xsl:vendor')!='Transformiix'\">\n"
|
||||
" <h2>Incoming Requests by DNS Opcode</h2>\n"
|
||||
" <!-- Non Mozilla specific markup -->\n"
|
||||
" <div class=\"pie\" id=\"chart_incoming_requests\">[no incoming requests]</div>\n"
|
||||
" <div class=\"pie\" id=\"chart_incoming_opcodes\">\n"
|
||||
" [cannot display chart]\n"
|
||||
" </div>\n"
|
||||
" </xsl:if>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"server/counters[@type="opcode"]/counter\">\n"
|
||||
" <xsl:for-each select=\"server/counters[@type="opcode"]/counter[. > 0 or substring(@name,1,3) != 'RES']\">\n"
|
||||
" <xsl:sort select=\".\" data-type=\"number\" order=\"descending\"/>\n"
|
||||
" <tr>\n"
|
||||
" <th>\n"
|
||||
@@ -241,10 +327,14 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" <h3>Incoming Queries by Type</h3>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"server/counters[@type="qtype"]/counter\">\n"
|
||||
" <xsl:if test=\"system-property('xsl:vendor')!='Transformiix'\">\n"
|
||||
" <!-- Non Mozilla specific markup -->\n"
|
||||
" <div class=\"pie\" id=\"chart_incoming_qtypes\">[no incoming queries]</div>\n"
|
||||
" <h3>Incoming Queries by Query Type</h3>\n"
|
||||
" <div class=\"pie\" id=\"chart_incoming_qtypes\">\n"
|
||||
" [cannot display chart]\n"
|
||||
" </div>\n"
|
||||
" </xsl:if>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"server/counters[@type="qtype"]/counter\">\n"
|
||||
@@ -272,6 +362,8 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"views/view[count(counters[@type="resqtype"]/counter) > 0]\">\n"
|
||||
" <h2>Outgoing Queries per view</h2>\n"
|
||||
" <xsl:for-each select=\"views/view[count(counters[@type="resqtype"]/counter) > 0]\">\n"
|
||||
" <h3>View <xsl:value-of select=\"@name\"/></h3>\n"
|
||||
@@ -279,16 +371,16 @@ static char xslmsg[] =
|
||||
" <!-- Non Mozilla specific markup -->\n"
|
||||
" <script type=\"text/javascript\">\n"
|
||||
" graphs.push({\n"
|
||||
" 'title': \"Outgoing queries for view: <xsl:value-of select=\"@name\"/>\",\n"
|
||||
" 'title': \"Outgoing Queries for view: <xsl:value-of select=\"@name\"/>\",\n"
|
||||
" 'target': 'chart_outgoing_queries_view_<xsl:value-of select=\"@name\"/>',\n"
|
||||
" 'style': 'barchart',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"counters[@type="resqtype"]/counter\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" });\n"
|
||||
" \n"
|
||||
" </script>\n"
|
||||
" <xsl:variable name=\"target\">\n"
|
||||
" <xsl:value-of select=\"@name\"/>\n"
|
||||
" </xsl:variable>\n"
|
||||
" <div class=\"pie\" id=\"chart_outgoing_queries_view_{$target}\"/>\n"
|
||||
" <div class=\"pie\" id=\"chart_outgoing_queries_view_{$target}\">[no data to display]</div>\n"
|
||||
" </xsl:if>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"counters[@type="resqtype"]/counter\">\n"
|
||||
@@ -311,6 +403,8 @@ static char xslmsg[] =
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"server/counters[@type="nsstat"]/counter[.>0]\">\n"
|
||||
" <h2>Server Statistics</h2>\n"
|
||||
" <xsl:if test=\"system-property('xsl:vendor')!='Transformiix'\">\n"
|
||||
" <!-- Non Mozilla specific markup -->\n"
|
||||
@@ -318,11 +412,11 @@ static char xslmsg[] =
|
||||
" graphs.push({\n"
|
||||
" 'title' : \"Server Counters\",\n"
|
||||
" 'target': 'chart_server_nsstat_restype',\n"
|
||||
" 'style': 'barchart',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"server/counters[@type="nsstat"]/counter[.>0]\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" });\n"
|
||||
" \n"
|
||||
" </script>\n"
|
||||
" <div class=\"pie\" id=\"chart_server_nsstat_restype\"/>\n"
|
||||
" <div class=\"pie\" id=\"chart_server_nsstat_restype\">[no data to display]</div>\n"
|
||||
" </xsl:if>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"server/counters[@type="nsstat"]/counter[.>0]\">\n"
|
||||
@@ -344,18 +438,20 @@ static char xslmsg[] =
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" <h2>Zone Maintenance Statistics</h2>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"server/counters[@type="zonestat"]/counter[.>0]\">\n"
|
||||
" <xsl:if test=\"system-property('xsl:vendor')!='Transformiix'\">\n"
|
||||
" <h2>Zone Maintenance Statistics</h2>\n"
|
||||
" <script type=\"text/javascript\">\n"
|
||||
" graphs.push({\n"
|
||||
" 'title' : \"Zone Maintenance Stats\",\n"
|
||||
" 'target': 'chart_server_zone_maint',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"server/counters[@type="zonestat"]/counter\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" 'style': 'barchart',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"server/counters[@type="zonestat"]/counter[.>0]\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" });\n"
|
||||
"\n"
|
||||
" </script>\n"
|
||||
" <!-- Non Mozilla specific markup -->\n"
|
||||
" <div class=\"pie\" id=\"chart_server_zone_maint\"/>\n"
|
||||
" <div class=\"pie\" id=\"chart_server_zone_maint\">[no data to display]</div>\n"
|
||||
" </xsl:if>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"server/counters[@type="zonestat"]/counter\">\n"
|
||||
@@ -376,6 +472,8 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"server/counters[@type="resstat"]/counter[.>0]\">\n"
|
||||
" <h2>Resolver Statistics (Common)</h2>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"server/counters[@type="resstat"]/counter\">\n"
|
||||
@@ -396,7 +494,9 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:for-each select=\"views/view\">\n"
|
||||
" <xsl:if test=\"counters[@type="resstats"]/counter[.>0]\">\n"
|
||||
" <h3>Resolver Statistics for View <xsl:value-of select=\"@name\"/></h3>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"counters[@type="resstats"]/counter[.>0]\">\n"
|
||||
@@ -417,10 +517,10 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" </xsl:if>\n"
|
||||
" </xsl:for-each>\n"
|
||||
"\n"
|
||||
"\n"
|
||||
" <xsl:for-each select=\"views/view\">\n"
|
||||
" <xsl:if test=\"counters[@type="adbstat"]/counter[.>0]\">\n"
|
||||
" <h3>ADB Statistics for View <xsl:value-of select=\"@name\"/></h3>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"counters[@type="adbstat"]/counter[.>0]\">\n"
|
||||
@@ -441,9 +541,11 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" </xsl:if>\n"
|
||||
" </xsl:for-each>\n"
|
||||
"\n"
|
||||
" <xsl:for-each select=\"views/view\">\n"
|
||||
" <xsl:if test=\"counters[@type="cachestats"]/counter[.>0]\">\n"
|
||||
" <h3>Cache Statistics for View <xsl:value-of select=\"@name\"/></h3>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"counters[@type="cachestats"]/counter[.>0]\">\n"
|
||||
@@ -464,11 +566,12 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" </xsl:if>\n"
|
||||
" </xsl:for-each>\n"
|
||||
"\n"
|
||||
"\n"
|
||||
" <h3>Cache DB RRsets for View <xsl:value-of select=\"@name\"/></h3>\n"
|
||||
" <xsl:for-each select=\"views/view\">\n"
|
||||
" <xsl:if test=\"cache/rrset\">\n"
|
||||
" <h3>Cache DB RRsets for View <xsl:value-of select=\"@name\"/></h3>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"cache/rrset\">\n"
|
||||
" <xsl:variable name=\"css-class6\">\n"
|
||||
@@ -488,7 +591,10 @@ static char xslmsg[] =
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" </xsl:if>\n"
|
||||
" </xsl:for-each>\n"
|
||||
"\n"
|
||||
" <xsl:if test=\"server/counters[@type="sockstat"]/counter[.>0]\">\n"
|
||||
" <h2>Socket I/O Statistics</h2>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"server/counters[@type="sockstat"]/counter[.>0]\">\n"
|
||||
@@ -509,33 +615,35 @@ static char xslmsg[] =
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" <br/>\n"
|
||||
" <h2>Response Codes per view/zone</h2>\n"
|
||||
" <xsl:for-each select=\"views/view[zones/zone/counters[@type="rcode"]/counter >0]\">\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"views/view[zones/zone/counters[@type="qtype"]/counter >0]\">\n"
|
||||
" <h2>Received QTYPES per view/zone</h2>\n"
|
||||
" <xsl:for-each select=\"views/view[zones/zone/counters[@type="qtype"]/counter >0]\">\n"
|
||||
" <h3>View <xsl:value-of select=\"@name\"/></h3>\n"
|
||||
" <xsl:variable name=\"thisview\">\n"
|
||||
" <xsl:value-of select=\"@name\"/>\n"
|
||||
" </xsl:variable>\n"
|
||||
" <xsl:for-each select=\"zones/zone\">\n"
|
||||
" <xsl:if test=\"counters[@type="rcode"]/counter[. > 0]\">\n"
|
||||
" <xsl:if test=\"counters[@type="qtype"]/counter[count(.) > 0]\">\n"
|
||||
" <h4>Zone <xsl:value-of select=\"@name\"/></h4>\n"
|
||||
" <xsl:if test=\"system-property('xsl:vendor')!='Transformiix'\">\n"
|
||||
" <!-- Non Mozilla specific markup -->\n"
|
||||
" <script type=\"text/javascript\">\n"
|
||||
" graphs.push({\n"
|
||||
" 'title': \"Response Codes for zone <xsl:value-of select=\"@name\"/>\",\n"
|
||||
" 'target': 'chart_rescode_<xsl:value-of select=\"../../@name\"/>_<xsl:value-of select=\"@name\"/>',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"counters[@type="rcode"]/counter[.>0 and @name != "QryAuthAns"]\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" 'title': \"Query types for zone <xsl:value-of select=\"@name\"/>\",\n"
|
||||
" 'target': 'chart_qtype_<xsl:value-of select=\"../../@name\"/>_<xsl:value-of select=\"@name\"/>',\n"
|
||||
" 'style': 'barchart',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"counters[@type="qtype"]/counter[.>0 and @name != "QryAuthAns"]\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" });\n"
|
||||
"\n"
|
||||
" </script>\n"
|
||||
" <xsl:variable name=\"target\">\n"
|
||||
" <xsl:value-of select=\"@name\"/>\n"
|
||||
" </xsl:variable>\n"
|
||||
" <div class=\"pie\" id=\"chart_rescode_{$thisview}_{$target}\"/>\n"
|
||||
" <div class=\"pie\" id=\"chart_qtype_{$thisview}_{$target}\">[no data to display]</div>\n"
|
||||
" </xsl:if>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"counters[@type="rcode"]/counter[.>0 and @name != "QryAuthAns"]\">\n"
|
||||
" <xsl:for-each select=\"counters[@type="qtype"]/counter\">\n"
|
||||
" <xsl:sort select=\".\"/>\n"
|
||||
" <xsl:variable name=\"css-class10\">\n"
|
||||
" <xsl:choose>\n"
|
||||
@@ -556,32 +664,35 @@ static char xslmsg[] =
|
||||
" </xsl:if>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" <h2>Received QTYPES per view/zone</h2>\n"
|
||||
" <xsl:for-each select=\"views/view[zones/zone/counters[@type="qtype"]/counter >0]\">\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"views/view[zones/zone/counters[@type="rcode"]/counter >0]\">\n"
|
||||
" <h2>Response Codes per view/zone</h2>\n"
|
||||
" <xsl:for-each select=\"views/view[zones/zone/counters[@type="rcode"]/counter >0]\">\n"
|
||||
" <h3>View <xsl:value-of select=\"@name\"/></h3>\n"
|
||||
" <xsl:variable name=\"thisview2\">\n"
|
||||
" <xsl:value-of select=\"@name\"/>\n"
|
||||
" </xsl:variable>\n"
|
||||
" <xsl:for-each select=\"zones/zone\">\n"
|
||||
" <xsl:if test=\"counters[@type="qtype"]/counter[count(.) > 0]\">\n"
|
||||
" <xsl:if test=\"counters[@type="rcode"]/counter[. > 0]\">\n"
|
||||
" <h4>Zone <xsl:value-of select=\"@name\"/></h4>\n"
|
||||
" <xsl:if test=\"system-property('xsl:vendor')!='Transformiix'\">\n"
|
||||
" <!-- Non Mozilla specific markup -->\n"
|
||||
" <script type=\"text/javascript\">\n"
|
||||
" graphs.push({\n"
|
||||
" 'title': \"Query Types for zone <xsl:value-of select=\"@name\"/>\",\n"
|
||||
" 'target': 'chart_qtype_<xsl:value-of select=\"../../@name\"/>_<xsl:value-of select=\"@name\"/>',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"counters[@type="qtype"]/counter[.>0 and @name != "QryAuthAns"]\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" 'title': \"Response codes for zone <xsl:value-of select=\"@name\"/>\",\n"
|
||||
" 'target': 'chart_rescode_<xsl:value-of select=\"../../@name\"/>_<xsl:value-of select=\"@name\"/>',\n"
|
||||
" 'style': 'barchart',\n"
|
||||
" 'data': [['Type','Counter'],<xsl:for-each select=\"counters[@type="rcode"]/counter[.>0 and @name != "QryAuthAns"]\">['<xsl:value-of select=\"@name\"/>',<xsl:value-of select=\".\"/>],</xsl:for-each>]\n"
|
||||
" });\n"
|
||||
"\n"
|
||||
" </script>\n"
|
||||
" <xsl:variable name=\"target\">\n"
|
||||
" <xsl:value-of select=\"@name\"/>\n"
|
||||
" </xsl:variable>\n"
|
||||
" <div class=\"pie\" id=\"chart_qtype_{$thisview2}_{$target}\"/>\n"
|
||||
" <div class=\"pie\" id=\"chart_rescode_{$thisview2}_{$target}\">[no data to display]</div>\n"
|
||||
" </xsl:if>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"counters[@type="qtype"]/counter\">\n"
|
||||
" <xsl:for-each select=\"counters[@type="rcode"]/counter[.>0 and @name != "QryAuthAns"]\">\n"
|
||||
" <xsl:sort select=\".\"/>\n"
|
||||
" <xsl:variable name=\"css-class11\">\n"
|
||||
" <xsl:choose>\n"
|
||||
@@ -602,8 +713,10 @@ static char xslmsg[] =
|
||||
" </xsl:if>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"socketmgr/sockets/socket\">\n"
|
||||
" <h2>Network Status</h2>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <table class=\"netstat\">\n"
|
||||
" <tr>\n"
|
||||
" <th>ID</th>\n"
|
||||
" <th>Name</th>\n"
|
||||
@@ -649,6 +762,8 @@ static char xslmsg[] =
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"taskmgr/thread-model/type\">\n"
|
||||
" <h2>Task Manager Configuration</h2>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <tr>\n"
|
||||
@@ -675,7 +790,7 @@ static char xslmsg[] =
|
||||
" <xsl:value-of select=\"taskmgr/thread-model/tasks-running\"/>\n"
|
||||
" </td>\n"
|
||||
" </tr>\n"
|
||||
" <tr class=\"odd\">\n"
|
||||
" <tr class=\"even\">\n"
|
||||
" <th>Tasks Ready</th>\n"
|
||||
" <td>\n"
|
||||
" <xsl:value-of select=\"taskmgr/thread-model/tasks-ready\"/>\n"
|
||||
@@ -683,8 +798,10 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"taskmgr/tasks/task\">\n"
|
||||
" <h2>Tasks</h2>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <table class=\"tasks\">\n"
|
||||
" <tr>\n"
|
||||
" <th>ID</th>\n"
|
||||
" <th>Name</th>\n"
|
||||
@@ -724,6 +841,8 @@ static char xslmsg[] =
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"memory/summary\">\n"
|
||||
" <h2>Memory Usage Summary</h2>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <xsl:for-each select=\"memory/summary/*\">\n"
|
||||
@@ -744,8 +863,10 @@ static char xslmsg[] =
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" <br/>\n"
|
||||
" </xsl:if>\n"
|
||||
" <xsl:if test=\"memory/contexts/context\">\n"
|
||||
" <h2>Memory Contexts</h2>\n"
|
||||
" <table class=\"counters\">\n"
|
||||
" <table class=\"mctx\">\n"
|
||||
" <tr>\n"
|
||||
" <th>ID</th>\n"
|
||||
" <th>Name</th>\n"
|
||||
@@ -800,6 +921,7 @@ static char xslmsg[] =
|
||||
" </tr>\n"
|
||||
" </xsl:for-each>\n"
|
||||
" </table>\n"
|
||||
" </xsl:if>\n"
|
||||
" <hr/>\n"
|
||||
" <p class=\"footer\">Internet Systems Consortium Inc.<br/><a href=\"http://www.isc.org\">http://www.isc.org</a></p>\n"
|
||||
" </body>\n"
|
||||
|
@@ -37,6 +37,7 @@ while (<>) {
|
||||
s/\>\ \</\>\</g;
|
||||
s/\"/\\\"/g;
|
||||
s/^/\t\"/;
|
||||
s/[\ \t]+$//g;
|
||||
s/$/\\n\"/;
|
||||
if ($lines eq "") {
|
||||
$lines .= $_;
|
||||
|
@@ -8779,6 +8779,9 @@ ns_server_add_zone(ns_server_t *server, char *args) {
|
||||
"zone %s added to view %s via addzone",
|
||||
zonename, viewname);
|
||||
|
||||
/* Adding a zone counts as reconfiguration */
|
||||
CHECK(isc_time_now(&ns_g_configtime));
|
||||
|
||||
result = ISC_R_SUCCESS;
|
||||
|
||||
cleanup:
|
||||
@@ -9038,6 +9041,9 @@ ns_server_del_zone(ns_server_t *server, char *args, isc_buffer_t *text) {
|
||||
NS_LOGMODULE_SERVER, ISC_LOG_INFO,
|
||||
"zone %s removed via delzone", zonename);
|
||||
|
||||
/* Removing a zone counts as reconfiguration */
|
||||
CHECK(isc_time_now(&ns_g_configtime));
|
||||
|
||||
result = ISC_R_SUCCESS;
|
||||
|
||||
cleanup:
|
||||
|
@@ -856,8 +856,16 @@ opcodestat_dump(dns_opcode_t code, isc_uint64_t val, void *arg) {
|
||||
}
|
||||
|
||||
#ifdef HAVE_LIBXML2
|
||||
|
||||
/* XXXMLG below here sucks. (not so much) */
|
||||
/*
|
||||
* Which statistics to include when rendering to XML
|
||||
*/
|
||||
#define STATS_XML_STATUS 0x00 /* display only common statistics */
|
||||
#define STATS_XML_SERVER 0x01
|
||||
#define STATS_XML_ZONES 0x02
|
||||
#define STATS_XML_TASKS 0x04
|
||||
#define STATS_XML_NET 0x08
|
||||
#define STATS_XML_MEM 0x10
|
||||
#define STATS_XML_ALL 0xff
|
||||
|
||||
static isc_result_t
|
||||
zone_xmlrender(dns_zone_t *zone, void *arg) {
|
||||
@@ -943,7 +951,9 @@ zone_xmlrender(dns_zone_t *zone, void *arg) {
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
generatexml(ns_server_t *server, int *buflen, xmlChar **buf) {
|
||||
generatexml(ns_server_t *server, isc_uint32_t flags,
|
||||
int *buflen, xmlChar **buf)
|
||||
{
|
||||
char boottime[sizeof "yyyy-mm-ddThh:mm:ssZ"];
|
||||
char configtime[sizeof "yyyy-mm-ddThh:mm:ssZ"];
|
||||
char nowstr[sizeof "yyyy-mm-ddThh:mm:ssZ"];
|
||||
@@ -974,29 +984,139 @@ generatexml(ns_server_t *server, int *buflen, xmlChar **buf) {
|
||||
ISC_XMLCHAR "type=\"text/xsl\" href=\"/bind9.xsl\""));
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "statistics"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "version",
|
||||
ISC_XMLCHAR "3.1"));
|
||||
ISC_XMLCHAR "3.4"));
|
||||
|
||||
/* Set common fields for statistics dump */
|
||||
dumparg.type = isc_statsformat_xml;
|
||||
dumparg.arg = writer;
|
||||
|
||||
/* Render server information */
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "server"));
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "boot-time"));
|
||||
TRY0(xmlTextWriterWriteString(writer, ISC_XMLCHAR boottime));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* boot-time */
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "config-time"));
|
||||
TRY0(xmlTextWriterWriteString(writer, ISC_XMLCHAR configtime));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* config-time */
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "current-time"));
|
||||
TRY0(xmlTextWriterWriteString(writer, ISC_XMLCHAR nowstr));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* current-time */
|
||||
|
||||
if ((flags & STATS_XML_SERVER) != 0) {
|
||||
dumparg.result = ISC_R_SUCCESS;
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "opcode"));
|
||||
|
||||
dns_opcodestats_dump(server->opcodestats, opcodestat_dump,
|
||||
&dumparg, ISC_STATSDUMP_VERBOSE);
|
||||
if (dumparg.result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer));
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "qtype"));
|
||||
|
||||
dumparg.result = ISC_R_SUCCESS;
|
||||
dns_rdatatypestats_dump(server->rcvquerystats, rdtypestat_dump,
|
||||
&dumparg, 0);
|
||||
if (dumparg.result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* counters */
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "nsstat"));
|
||||
|
||||
result = dump_counters(server->nsstats, isc_statsformat_xml,
|
||||
writer, NULL, nsstats_xmldesc,
|
||||
dns_nsstatscounter_max,
|
||||
nsstats_index, nsstat_values,
|
||||
ISC_STATSDUMP_VERBOSE);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /nsstat */
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "zonestat"));
|
||||
|
||||
result = dump_counters(server->zonestats, isc_statsformat_xml,
|
||||
writer, NULL, zonestats_xmldesc,
|
||||
dns_zonestatscounter_max,
|
||||
zonestats_index, zonestat_values,
|
||||
ISC_STATSDUMP_VERBOSE);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /zonestat */
|
||||
|
||||
/*
|
||||
* Start by rendering the views we know of here. For each view we
|
||||
* know of, call its rendering function.
|
||||
* Most of the common resolver statistics entries are 0, so
|
||||
* we don't use the verbose dump here.
|
||||
*/
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "resstat"));
|
||||
result = dump_counters(server->resolverstats,
|
||||
isc_statsformat_xml, writer,
|
||||
NULL, resstats_xmldesc,
|
||||
dns_resstatscounter_max,
|
||||
resstats_index, resstat_values, 0);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* resstat */
|
||||
}
|
||||
|
||||
if ((flags & STATS_XML_NET) != 0) {
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "sockstat"));
|
||||
|
||||
result = dump_counters(server->sockstats, isc_statsformat_xml,
|
||||
writer, NULL, sockstats_xmldesc,
|
||||
isc_sockstatscounter_max,
|
||||
sockstats_index, sockstat_values,
|
||||
ISC_STATSDUMP_VERBOSE);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /sockstat */
|
||||
}
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /server */
|
||||
|
||||
/*
|
||||
* Render views. For each view we know of, call its
|
||||
* rendering function.
|
||||
*/
|
||||
view = ISC_LIST_HEAD(server->viewlist);
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "views"));
|
||||
while (view != NULL) {
|
||||
while (view != NULL &&
|
||||
((flags & (STATS_XML_SERVER | STATS_XML_ZONES)) != 0))
|
||||
{
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "view"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "name",
|
||||
ISC_XMLCHAR view->name));
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "zones"));
|
||||
result = dns_zt_apply(view->zonetable, ISC_TRUE, zone_xmlrender,
|
||||
writer);
|
||||
if ((flags & STATS_XML_ZONES) != 0) {
|
||||
TRY0(xmlTextWriterStartElement(writer,
|
||||
ISC_XMLCHAR "zones"));
|
||||
result = dns_zt_apply(view->zonetable, ISC_TRUE,
|
||||
zone_xmlrender, writer);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* zones */
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /zones */
|
||||
}
|
||||
|
||||
if ((flags & STATS_XML_SERVER) == 0) {
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /view */
|
||||
view = ISC_LIST_NEXT(view, link);
|
||||
continue;
|
||||
}
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer,
|
||||
ISC_XMLCHAR "counters"));
|
||||
@@ -1071,115 +1191,28 @@ generatexml(ns_server_t *server, int *buflen, xmlChar **buf) {
|
||||
|
||||
view = ISC_LIST_NEXT(view, link);
|
||||
}
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* views */
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /views */
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "socketmgr"));
|
||||
if ((flags & STATS_XML_NET) != 0) {
|
||||
TRY0(xmlTextWriterStartElement(writer,
|
||||
ISC_XMLCHAR "socketmgr"));
|
||||
TRY0(isc_socketmgr_renderxml(ns_g_socketmgr, writer));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* socketmgr */
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /socketmgr */
|
||||
}
|
||||
|
||||
if ((flags & STATS_XML_TASKS) != 0) {
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "taskmgr"));
|
||||
TRY0(isc_taskmgr_renderxml(ns_g_taskmgr, writer));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* taskmgr */
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "server"));
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "boot-time"));
|
||||
TRY0(xmlTextWriterWriteString(writer, ISC_XMLCHAR boottime));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* boot-time */
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "config-time"));
|
||||
TRY0(xmlTextWriterWriteString(writer, ISC_XMLCHAR configtime));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* config-time */
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "current-time"));
|
||||
TRY0(xmlTextWriterWriteString(writer, ISC_XMLCHAR nowstr));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* current-time */
|
||||
|
||||
dumparg.result = ISC_R_SUCCESS;
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "opcode"));
|
||||
|
||||
dns_opcodestats_dump(server->opcodestats, opcodestat_dump, &dumparg,
|
||||
0);
|
||||
if (dumparg.result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* counters type=opcode */
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "qtype"));
|
||||
|
||||
dumparg.result = ISC_R_SUCCESS;
|
||||
dns_rdatatypestats_dump(server->rcvquerystats, rdtypestat_dump,
|
||||
&dumparg, 0);
|
||||
if (dumparg.result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* counters */
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "nsstat"));
|
||||
|
||||
result = dump_counters(server->nsstats, isc_statsformat_xml,
|
||||
writer, NULL, nsstats_xmldesc,
|
||||
dns_nsstatscounter_max,
|
||||
nsstats_index, nsstat_values,
|
||||
ISC_STATSDUMP_VERBOSE);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* counters type=nsstat */
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "zonestat"));
|
||||
|
||||
result = dump_counters(server->zonestats, isc_statsformat_xml, writer,
|
||||
NULL, zonestats_xmldesc,
|
||||
dns_zonestatscounter_max, zonestats_index,
|
||||
zonestat_values, ISC_STATSDUMP_VERBOSE);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* counters type=zonestat */
|
||||
|
||||
/*
|
||||
* Most of the common resolver statistics entries are 0, so we don't
|
||||
* use the verbose dump here.
|
||||
*/
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "resstat"));
|
||||
result = dump_counters(server->resolverstats, isc_statsformat_xml,
|
||||
writer, NULL, resstats_xmldesc,
|
||||
dns_resstatscounter_max, resstats_index,
|
||||
resstat_values, 0);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* counters type=resstat */
|
||||
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "counters"));
|
||||
TRY0(xmlTextWriterWriteAttribute(writer, ISC_XMLCHAR "type",
|
||||
ISC_XMLCHAR "sockstat"));
|
||||
|
||||
result = dump_counters(server->sockstats, isc_statsformat_xml,
|
||||
writer, NULL, sockstats_xmldesc,
|
||||
isc_sockstatscounter_max, sockstats_index,
|
||||
sockstat_values, ISC_STATSDUMP_VERBOSE);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto error;
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* counters type=sockstat */
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* server */
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /taskmgr */
|
||||
}
|
||||
|
||||
if ((flags & STATS_XML_MEM) != 0) {
|
||||
TRY0(xmlTextWriterStartElement(writer, ISC_XMLCHAR "memory"));
|
||||
TRY0(isc_mem_renderxml(writer));
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* memory */
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* statistics */
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /memory */
|
||||
}
|
||||
|
||||
TRY0(xmlTextWriterEndElement(writer)); /* /statistics */
|
||||
TRY0(xmlTextWriterEndDocument(writer));
|
||||
|
||||
xmlFreeTextWriter(writer);
|
||||
@@ -1208,10 +1241,11 @@ wrap_xmlfree(isc_buffer_t *buffer, void *arg) {
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_index(const char *url, const char *querystring, void *arg,
|
||||
unsigned int *retcode, const char **retmsg, const char **mimetype,
|
||||
isc_buffer_t *b, isc_httpdfree_t **freecb,
|
||||
void **freecb_args)
|
||||
render_xml(isc_uint32_t flags, const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
unsigned char *msg = NULL;
|
||||
int msglen;
|
||||
@@ -1219,9 +1253,11 @@ render_index(const char *url, const char *querystring, void *arg,
|
||||
isc_result_t result;
|
||||
|
||||
UNUSED(url);
|
||||
UNUSED(urlinfo);
|
||||
UNUSED(headers);
|
||||
UNUSED(querystring);
|
||||
|
||||
result = generatexml(server, &msglen, &msg);
|
||||
result = generatexml(server, flags, &msglen, &msg);
|
||||
|
||||
if (result == ISC_R_SUCCESS) {
|
||||
*retcode = 200;
|
||||
@@ -1239,12 +1275,104 @@ render_index(const char *url, const char *querystring, void *arg,
|
||||
return (result);
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_xml_all(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_xml(STATS_XML_ALL, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_xml_status(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_xml(STATS_XML_STATUS, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_xml_server(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_xml(STATS_XML_SERVER, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_xml_zones(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_xml(STATS_XML_ZONES, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_xml_net(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_xml(STATS_XML_NET, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_xml_tasks(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_xml(STATS_XML_TASKS, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_xml_mem(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_xml(STATS_XML_MEM, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
#endif /* HAVE_LIBXML2 */
|
||||
|
||||
#ifdef HAVE_JSON
|
||||
/*
|
||||
* Which statistics to include when rendering to JSON
|
||||
*/
|
||||
#define STATS_JSON_STATUS 0x00 /* display only common statistics */
|
||||
#define STATS_JSON_SERVER 0x01
|
||||
#define STATS_JSON_ZONES 0x02
|
||||
#define STATS_JSON_TASKS 0x04
|
||||
@@ -1372,7 +1500,7 @@ zone_jsonrender(dns_zone_t *zone, void *arg) {
|
||||
|
||||
static isc_result_t
|
||||
generatejson(ns_server_t *server, size_t *msglen,
|
||||
const char **msg, json_object **rootp, isc_uint8_t flags)
|
||||
const char **msg, json_object **rootp, isc_uint32_t flags)
|
||||
{
|
||||
dns_view_t *view;
|
||||
isc_result_t result = ISC_R_SUCCESS;
|
||||
@@ -1431,7 +1559,8 @@ generatejson(ns_server_t *server, size_t *msglen,
|
||||
dumparg.arg = counters;
|
||||
|
||||
dns_opcodestats_dump(server->opcodestats,
|
||||
opcodestat_dump, &dumparg, 0);
|
||||
opcodestat_dump, &dumparg,
|
||||
ISC_STATSDUMP_VERBOSE);
|
||||
if (dumparg.result != ISC_R_SUCCESS) {
|
||||
json_object_put(counters);
|
||||
goto error;
|
||||
@@ -1745,7 +1874,9 @@ generatejson(ns_server_t *server, size_t *msglen,
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_json(isc_uint8_t flags, const char *url, const char *querystring,
|
||||
render_json(isc_uint32_t flags,
|
||||
const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers,
|
||||
void *arg, unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
@@ -1758,6 +1889,8 @@ render_json(isc_uint8_t flags, const char *url, const char *querystring,
|
||||
char *p;
|
||||
|
||||
UNUSED(url);
|
||||
UNUSED(urlinfo);
|
||||
UNUSED(headers);
|
||||
UNUSED(querystring);
|
||||
|
||||
result = generatejson(server, &msglen, &msg, &bindstats, flags);
|
||||
@@ -1779,88 +1912,147 @@ render_json(isc_uint8_t flags, const char *url, const char *querystring,
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_json_all(const char *url, const char *querystring, void *arg,
|
||||
render_json_all(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_json(STATS_JSON_ALL, url, querystring, arg,
|
||||
return (render_json(STATS_JSON_ALL, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_json_server(const char *url, const char *querystring, void *arg,
|
||||
render_json_status(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_json(STATS_JSON_SERVER, url, querystring, arg,
|
||||
return (render_json(STATS_JSON_STATUS, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_json_zones(const char *url, const char *querystring, void *arg,
|
||||
render_json_server(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_json(STATS_JSON_ZONES, url, querystring, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
static isc_result_t
|
||||
render_json_mem(const char *url, const char *querystring, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_json(STATS_JSON_MEM, url, querystring, arg,
|
||||
return (render_json(STATS_JSON_SERVER, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_json_tasks(const char *url, const char *querystring, void *arg,
|
||||
render_json_zones(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_json(STATS_JSON_TASKS, url, querystring, arg,
|
||||
return (render_json(STATS_JSON_ZONES, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
static isc_result_t
|
||||
render_json_mem(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_json(STATS_JSON_MEM, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_json_net(const char *url, const char *querystring, void *arg,
|
||||
render_json_tasks(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_json(STATS_JSON_NET, url, querystring, arg,
|
||||
return (render_json(STATS_JSON_TASKS, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_json_net(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
return (render_json(STATS_JSON_NET, url, urlinfo,
|
||||
querystring, headers, arg,
|
||||
retcode, retmsg, mimetype, b,
|
||||
freecb, freecb_args));
|
||||
}
|
||||
#endif /* HAVE_JSON */
|
||||
|
||||
static isc_result_t
|
||||
render_xsl(const char *url, const char *querystring, void *args,
|
||||
unsigned int *retcode, const char **retmsg, const char **mimetype,
|
||||
isc_buffer_t *b, isc_httpdfree_t **freecb,
|
||||
void **freecb_args)
|
||||
render_xsl(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers,
|
||||
void *args, unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
{
|
||||
isc_result_t result;
|
||||
|
||||
UNUSED(url);
|
||||
UNUSED(querystring);
|
||||
UNUSED(args);
|
||||
|
||||
*retcode = 200;
|
||||
*retmsg = "OK";
|
||||
*mimetype = "text/xslt+xml";
|
||||
isc_buffer_reinit(b, xslmsg, strlen(xslmsg));
|
||||
isc_buffer_add(b, strlen(xslmsg));
|
||||
*freecb = NULL;
|
||||
*freecb_args = NULL;
|
||||
*mimetype = "text/xslt+xml";
|
||||
|
||||
if (urlinfo->isstatic) {
|
||||
isc_time_t when;
|
||||
char *p = strcasestr(headers, "If-Modified-Since: ");
|
||||
|
||||
if (p != NULL) {
|
||||
time_t t1, t2;
|
||||
p += strlen("If-Modified-Since: ");
|
||||
result = isc_time_parsehttptimestamp(p, &when);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto send;
|
||||
|
||||
result = isc_time_secondsastimet(&when, &t1);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto send;
|
||||
|
||||
result = isc_time_secondsastimet(&urlinfo->loadtime,
|
||||
&t2);
|
||||
if (result != ISC_R_SUCCESS)
|
||||
goto send;
|
||||
|
||||
if (t1 < t2)
|
||||
goto send;
|
||||
|
||||
*retcode = 304;
|
||||
*retmsg = "Not modified";
|
||||
return (ISC_R_SUCCESS);
|
||||
}
|
||||
}
|
||||
|
||||
send:
|
||||
*retcode = 200;
|
||||
*retmsg = "OK";
|
||||
isc_buffer_reinit(b, xslmsg, strlen(xslmsg));
|
||||
isc_buffer_add(b, strlen(xslmsg));
|
||||
|
||||
return (ISC_R_SUCCESS);
|
||||
}
|
||||
@@ -1986,27 +2178,45 @@ add_listener(ns_server_t *server, ns_statschannel_t **listenerp,
|
||||
goto cleanup;
|
||||
|
||||
#ifdef HAVE_LIBXML2
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/", render_index, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml", render_index, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml/v3", render_index,
|
||||
server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/",
|
||||
render_xml_all, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml",
|
||||
render_xml_all, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml/v3",
|
||||
render_xml_all, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml/v3/status",
|
||||
render_xml_status, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml/v3/server",
|
||||
render_xml_server, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml/v3/zones",
|
||||
render_xml_zones, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml/v3/net",
|
||||
render_xml_net, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml/v3/tasks",
|
||||
render_xml_tasks, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/xml/v3/mem",
|
||||
render_xml_mem, server);
|
||||
#endif
|
||||
#ifdef HAVE_JSON
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json",
|
||||
render_json_all, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/server",
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/v1",
|
||||
render_json_all, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/v1/status",
|
||||
render_json_status, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/v1/server",
|
||||
render_json_server, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/zones",
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/v1/zones",
|
||||
render_json_zones, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/tasks",
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/v1/tasks",
|
||||
render_json_tasks, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/net",
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/v1/net",
|
||||
render_json_net, server);
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/mem",
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/json/v1/mem",
|
||||
render_json_mem, server);
|
||||
#endif
|
||||
isc_httpdmgr_addurl(listener->httpdmgr, "/bind9.xsl", render_xsl,
|
||||
server);
|
||||
isc_httpdmgr_addurl2(listener->httpdmgr, "/bind9.xsl", ISC_TRUE,
|
||||
render_xsl, server);
|
||||
|
||||
*listenerp = listener;
|
||||
isc_log_write(ns_g_lctx, NS_LOGCATEGORY_GENERAL,
|
||||
|
@@ -10256,25 +10256,50 @@ example.com CNAME rpz-tcp-only.
|
||||
error.
|
||||
</para>
|
||||
|
||||
<para>
|
||||
Broken-out subsets of the statistics can be viewed at
|
||||
<ulink url="http://127.0.0.1:8888/xml/v3/status"
|
||||
>http://127.0.0.1:8888/xml/v3/status</ulink>
|
||||
(server uptime and last reconfiguration time),
|
||||
<ulink url="http://127.0.0.1:8888/xml/v3/server"
|
||||
>http://127.0.0.1:8888/xml/v3/server</ulink>
|
||||
(server and resolver statistics),
|
||||
<ulink url="http://127.0.0.1:8888/xml/v3/zones"
|
||||
>http://127.0.0.1:8888/xml/v3/zones</ulink>
|
||||
(zone statistics),
|
||||
<ulink url="http://127.0.0.1:8888/xml/v3/net"
|
||||
>http://127.0.0.1:8888/xml/v3/net</ulink>
|
||||
(network status and socket statistics),
|
||||
<ulink url="http://127.0.0.1:8888/xml/v3/mem"
|
||||
>http://127.0.0.1:8888/xml/v3/mem</ulink>
|
||||
(memory manager statistics),
|
||||
<ulink url="http://127.0.0.1:8888/xml/v3/tasks"
|
||||
>http://127.0.0.1:8888/xml/v3/tasks</ulink>
|
||||
(task manager statistics).
|
||||
</para>
|
||||
|
||||
<para>
|
||||
The full set of statistics can also be read in JSON format at
|
||||
<ulink url="http://127.0.0.1:8888/json"
|
||||
>http://127.0.0.1:8888/json</ulink>.
|
||||
Broken-out subsets of the statistics can be viewed at
|
||||
<ulink url="http://127.0.0.1:8888/json/server"
|
||||
>http://127.0.0.1:8888/json/server</ulink>
|
||||
>http://127.0.0.1:8888/json</ulink>,
|
||||
with the broken-out subsets at
|
||||
<ulink url="http://127.0.0.1:8888/json/v1/status"
|
||||
>http://127.0.0.1:8888/json/v1/status</ulink>
|
||||
(server uptime and last reconfiguration time),
|
||||
<ulink url="http://127.0.0.1:8888/json/v1/server"
|
||||
>http://127.0.0.1:8888/json/v1/server</ulink>
|
||||
(server and resolver statistics),
|
||||
<ulink url="http://127.0.0.1:8888/json/zones"
|
||||
>http://127.0.0.1:8888/json/zones</ulink>
|
||||
<ulink url="http://127.0.0.1:8888/json/v1/zones"
|
||||
>http://127.0.0.1:8888/json/v1/zones</ulink>
|
||||
(zone statistics),
|
||||
<ulink url="http://127.0.0.1:8888/json/net"
|
||||
>http://127.0.0.1:8888/json/net</ulink>
|
||||
<ulink url="http://127.0.0.1:8888/json/v1/net"
|
||||
>http://127.0.0.1:8888/json/v1/net</ulink>
|
||||
(network status and socket statistics),
|
||||
<ulink url="http://127.0.0.1:8888/json/mem"
|
||||
>http://127.0.0.1:8888/json/mem</ulink>
|
||||
<ulink url="http://127.0.0.1:8888/json/v1/mem"
|
||||
>http://127.0.0.1:8888/json/v1/mem</ulink>
|
||||
(memory manager statistics),
|
||||
<ulink url="http://127.0.0.1:8888/json/tasks"
|
||||
>http://127.0.0.1:8888/json/tasks</ulink>
|
||||
<ulink url="http://127.0.0.1:8888/json/v1/tasks"
|
||||
>http://127.0.0.1:8888/json/v1/tasks</ulink>
|
||||
(task manager statistics).
|
||||
</para>
|
||||
</sect2>
|
||||
|
114
lib/isc/httpd.c
114
lib/isc/httpd.c
@@ -26,6 +26,7 @@
|
||||
#include <isc/socket.h>
|
||||
#include <isc/string.h>
|
||||
#include <isc/task.h>
|
||||
#include <isc/time.h>
|
||||
#include <isc/util.h>
|
||||
|
||||
#include <string.h>
|
||||
@@ -58,20 +59,6 @@
|
||||
#define HTTP_SENDGROW 1024
|
||||
#define HTTP_SEND_MAXLEN 10240
|
||||
|
||||
/*%
|
||||
* HTTP urls. These are the URLs we manage, and the function to call to
|
||||
* provide the data for it. We pass in the base url (so the same function
|
||||
* can handle multiple requests), and a structure to fill in to return a
|
||||
* result to the client. We also pass in a pointer to be filled in for
|
||||
* the data cleanup function.
|
||||
*/
|
||||
struct isc_httpdurl {
|
||||
char *url;
|
||||
isc_httpdaction_t *action;
|
||||
void *action_arg;
|
||||
ISC_LINK(isc_httpdurl_t) link;
|
||||
};
|
||||
|
||||
#define HTTPD_CLOSE 0x0001 /* Got a Connection: close header */
|
||||
#define HTTPD_FOUNDHOST 0x0002 /* Got a Host: header */
|
||||
|
||||
@@ -87,6 +74,7 @@ struct isc_httpd {
|
||||
*/
|
||||
char recvbuf[HTTP_RECVLEN]; /*%< receive buffer */
|
||||
isc_uint32_t recvlen; /*%< length recv'd */
|
||||
char *headers; /*%< set in process_request() */
|
||||
unsigned int method;
|
||||
char *url;
|
||||
char *querystring;
|
||||
@@ -217,20 +205,12 @@ static isc_result_t process_request(isc_httpd_t *, int);
|
||||
static void httpdmgr_destroy(isc_httpdmgr_t *);
|
||||
static isc_result_t grow_headerspace(isc_httpd_t *);
|
||||
static void reset_client(isc_httpd_t *httpd);
|
||||
static isc_result_t render_404(const char *, const char *,
|
||||
void *,
|
||||
unsigned int *, const char **,
|
||||
const char **, isc_buffer_t *,
|
||||
isc_httpdfree_t **, void **);
|
||||
static isc_result_t render_500(const char *, const char *,
|
||||
void *,
|
||||
unsigned int *, const char **,
|
||||
const char **, isc_buffer_t *,
|
||||
isc_httpdfree_t **, void **);
|
||||
|
||||
static isc_httpdaction_t render_404;
|
||||
static isc_httpdaction_t render_500;
|
||||
|
||||
static void
|
||||
destroy_client(isc_httpd_t **httpdp)
|
||||
{
|
||||
destroy_client(isc_httpd_t **httpdp) {
|
||||
isc_httpd_t *httpd = *httpdp;
|
||||
isc_httpdmgr_t *httpdmgr = httpd->mgr;
|
||||
|
||||
@@ -321,8 +301,7 @@ isc_httpdmgr_create(isc_mem_t *mctx, isc_socket_t *sock, isc_task_t *task,
|
||||
}
|
||||
|
||||
static void
|
||||
httpdmgr_destroy(isc_httpdmgr_t *httpdmgr)
|
||||
{
|
||||
httpdmgr_destroy(isc_httpdmgr_t *httpdmgr) {
|
||||
isc_mem_t *mctx;
|
||||
isc_httpdurl_t *url;
|
||||
|
||||
@@ -379,8 +358,7 @@ httpdmgr_destroy(isc_httpdmgr_t *httpdmgr)
|
||||
#define BUFLENOK(s) (httpd->recvbuf - (s) < HTTP_RECVLEN)
|
||||
|
||||
static isc_result_t
|
||||
process_request(isc_httpd_t *httpd, int length)
|
||||
{
|
||||
process_request(isc_httpd_t *httpd, int length) {
|
||||
char *s;
|
||||
char *p;
|
||||
int delim;
|
||||
@@ -390,6 +368,7 @@ process_request(isc_httpd_t *httpd, int length)
|
||||
httpd->recvlen += length;
|
||||
|
||||
httpd->recvbuf[httpd->recvlen] = 0;
|
||||
httpd->headers = NULL;
|
||||
|
||||
/*
|
||||
* If we don't find a blank line in our buffer, return that we need
|
||||
@@ -494,6 +473,8 @@ process_request(isc_httpd_t *httpd, int length)
|
||||
p = s + 1;
|
||||
s = p;
|
||||
|
||||
httpd->headers = s;
|
||||
|
||||
if (strstr(s, "Connection: close") != NULL)
|
||||
httpd->flags |= HTTPD_CLOSE;
|
||||
|
||||
@@ -513,8 +494,7 @@ process_request(isc_httpd_t *httpd, int length)
|
||||
}
|
||||
|
||||
static void
|
||||
isc_httpd_accept(isc_task_t *task, isc_event_t *ev)
|
||||
{
|
||||
isc_httpd_accept(isc_task_t *task, isc_event_t *ev) {
|
||||
isc_result_t result;
|
||||
isc_httpdmgr_t *httpdmgr = ev->ev_arg;
|
||||
isc_httpd_t *httpd;
|
||||
@@ -609,8 +589,8 @@ isc_httpd_accept(isc_task_t *task, isc_event_t *ev)
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_404(const char *url, const char *querystring,
|
||||
void *arg,
|
||||
render_404(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
@@ -618,7 +598,9 @@ render_404(const char *url, const char *querystring,
|
||||
static char msg[] = "No such URL.";
|
||||
|
||||
UNUSED(url);
|
||||
UNUSED(urlinfo);
|
||||
UNUSED(querystring);
|
||||
UNUSED(headers);
|
||||
UNUSED(arg);
|
||||
|
||||
*retcode = 404;
|
||||
@@ -633,8 +615,8 @@ render_404(const char *url, const char *querystring,
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
render_500(const char *url, const char *querystring,
|
||||
void *arg,
|
||||
render_500(const char *url, isc_httpdurl_t *urlinfo,
|
||||
const char *querystring, const char *headers, void *arg,
|
||||
unsigned int *retcode, const char **retmsg,
|
||||
const char **mimetype, isc_buffer_t *b,
|
||||
isc_httpdfree_t **freecb, void **freecb_args)
|
||||
@@ -642,7 +624,9 @@ render_500(const char *url, const char *querystring,
|
||||
static char msg[] = "Internal server failure.";
|
||||
|
||||
UNUSED(url);
|
||||
UNUSED(urlinfo);
|
||||
UNUSED(querystring);
|
||||
UNUSED(headers);
|
||||
UNUSED(arg);
|
||||
|
||||
*retcode = 500;
|
||||
@@ -657,8 +641,7 @@ render_500(const char *url, const char *querystring,
|
||||
}
|
||||
|
||||
static void
|
||||
isc_httpd_recvdone(isc_task_t *task, isc_event_t *ev)
|
||||
{
|
||||
isc_httpd_recvdone(isc_task_t *task, isc_event_t *ev) {
|
||||
isc_region_t r;
|
||||
isc_result_t result;
|
||||
isc_httpd_t *httpd = ev->ev_arg;
|
||||
@@ -710,8 +693,9 @@ isc_httpd_recvdone(isc_task_t *task, isc_event_t *ev)
|
||||
url = ISC_LIST_NEXT(url, link);
|
||||
}
|
||||
if (url == NULL)
|
||||
result = httpd->mgr->render_404(httpd->url, httpd->querystring,
|
||||
NULL,
|
||||
result = httpd->mgr->render_404(httpd->url, NULL,
|
||||
httpd->querystring,
|
||||
NULL, NULL,
|
||||
&httpd->retcode,
|
||||
&httpd->retmsg,
|
||||
&httpd->mimetype,
|
||||
@@ -719,14 +703,18 @@ isc_httpd_recvdone(isc_task_t *task, isc_event_t *ev)
|
||||
&httpd->freecb,
|
||||
&httpd->freecb_arg);
|
||||
else
|
||||
result = url->action(httpd->url, httpd->querystring,
|
||||
result = url->action(httpd->url, url,
|
||||
httpd->querystring,
|
||||
httpd->headers,
|
||||
url->action_arg,
|
||||
&httpd->retcode, &httpd->retmsg,
|
||||
&httpd->mimetype, &httpd->bodybuffer,
|
||||
&httpd->freecb, &httpd->freecb_arg);
|
||||
if (result != ISC_R_SUCCESS) {
|
||||
result = httpd->mgr->render_500(httpd->url, httpd->querystring,
|
||||
NULL, &httpd->retcode,
|
||||
result = httpd->mgr->render_500(httpd->url, url,
|
||||
httpd->querystring,
|
||||
NULL, NULL,
|
||||
&httpd->retcode,
|
||||
&httpd->retmsg,
|
||||
&httpd->mimetype,
|
||||
&httpd->bodybuffer,
|
||||
@@ -739,9 +727,19 @@ isc_httpd_recvdone(isc_task_t *task, isc_event_t *ev)
|
||||
isc_httpd_addheader(httpd, "Content-Type", httpd->mimetype);
|
||||
isc_httpd_addheader(httpd, "Date", datebuf);
|
||||
isc_httpd_addheader(httpd, "Expires", datebuf);
|
||||
|
||||
if (url != NULL && url->isstatic) {
|
||||
char loadbuf[32];
|
||||
isc_time_formathttptimestamp(&url->loadtime,
|
||||
loadbuf, sizeof(loadbuf));
|
||||
isc_httpd_addheader(httpd, "Last-Modified", loadbuf);
|
||||
isc_httpd_addheader(httpd, "Cache-Control: public", NULL);
|
||||
} else {
|
||||
isc_httpd_addheader(httpd, "Last-Modified", datebuf);
|
||||
isc_httpd_addheader(httpd, "Pragma: no-cache", NULL);
|
||||
isc_httpd_addheader(httpd, "Cache-Control: no-cache", NULL);
|
||||
}
|
||||
|
||||
isc_httpd_addheader(httpd, "Server: libisc", NULL);
|
||||
isc_httpd_addheaderuint(httpd, "Content-Length",
|
||||
isc_buffer_usedlength(&httpd->bodybuffer));
|
||||
@@ -766,8 +764,7 @@ isc_httpd_recvdone(isc_task_t *task, isc_event_t *ev)
|
||||
}
|
||||
|
||||
void
|
||||
isc_httpdmgr_shutdown(isc_httpdmgr_t **httpdmgrp)
|
||||
{
|
||||
isc_httpdmgr_shutdown(isc_httpdmgr_t **httpdmgrp) {
|
||||
isc_httpdmgr_t *httpdmgr;
|
||||
isc_httpd_t *httpd;
|
||||
httpdmgr = *httpdmgrp;
|
||||
@@ -794,8 +791,7 @@ isc_httpdmgr_shutdown(isc_httpdmgr_t **httpdmgrp)
|
||||
}
|
||||
|
||||
static isc_result_t
|
||||
grow_headerspace(isc_httpd_t *httpd)
|
||||
{
|
||||
grow_headerspace(isc_httpd_t *httpd) {
|
||||
char *newspace;
|
||||
unsigned int newlen;
|
||||
isc_region_t r;
|
||||
@@ -816,8 +812,7 @@ grow_headerspace(isc_httpd_t *httpd)
|
||||
}
|
||||
|
||||
isc_result_t
|
||||
isc_httpd_response(isc_httpd_t *httpd)
|
||||
{
|
||||
isc_httpd_response(isc_httpd_t *httpd) {
|
||||
isc_result_t result;
|
||||
unsigned int needlen;
|
||||
|
||||
@@ -869,8 +864,7 @@ isc_httpd_addheader(isc_httpd_t *httpd, const char *name,
|
||||
}
|
||||
|
||||
isc_result_t
|
||||
isc_httpd_endheaders(isc_httpd_t *httpd)
|
||||
{
|
||||
isc_httpd_endheaders(isc_httpd_t *httpd) {
|
||||
isc_result_t result;
|
||||
|
||||
while (isc_buffer_availablelength(&httpd->headerbuffer) < 2) {
|
||||
@@ -912,8 +906,7 @@ isc_httpd_addheaderuint(isc_httpd_t *httpd, const char *name, int val) {
|
||||
}
|
||||
|
||||
static void
|
||||
isc_httpd_senddone(isc_task_t *task, isc_event_t *ev)
|
||||
{
|
||||
isc_httpd_senddone(isc_task_t *task, isc_event_t *ev) {
|
||||
isc_httpd_t *httpd = ev->ev_arg;
|
||||
isc_region_t r;
|
||||
isc_socketevent_t *sev = (isc_socketevent_t *)ev;
|
||||
@@ -977,8 +970,7 @@ out:
|
||||
}
|
||||
|
||||
static void
|
||||
reset_client(isc_httpd_t *httpd)
|
||||
{
|
||||
reset_client(isc_httpd_t *httpd) {
|
||||
/*
|
||||
* Catch errors here. We MUST be in RECV mode, and we MUST NOT have
|
||||
* any outstanding buffers. If we have buffers, we have a leak.
|
||||
@@ -989,6 +981,7 @@ reset_client(isc_httpd_t *httpd)
|
||||
|
||||
httpd->recvbuf[0] = 0;
|
||||
httpd->recvlen = 0;
|
||||
httpd->headers = NULL;
|
||||
httpd->method = ISC_HTTPD_METHODUNKNOWN;
|
||||
httpd->url = NULL;
|
||||
httpd->querystring = NULL;
|
||||
@@ -1002,6 +995,14 @@ reset_client(isc_httpd_t *httpd)
|
||||
isc_result_t
|
||||
isc_httpdmgr_addurl(isc_httpdmgr_t *httpdmgr, const char *url,
|
||||
isc_httpdaction_t *func, void *arg)
|
||||
{
|
||||
return (isc_httpdmgr_addurl2(httpdmgr, url, ISC_FALSE, func, arg));
|
||||
}
|
||||
|
||||
isc_result_t
|
||||
isc_httpdmgr_addurl2(isc_httpdmgr_t *httpdmgr, const char *url,
|
||||
isc_boolean_t isstatic,
|
||||
isc_httpdaction_t *func, void *arg)
|
||||
{
|
||||
isc_httpdurl_t *item;
|
||||
|
||||
@@ -1022,6 +1023,9 @@ isc_httpdmgr_addurl(isc_httpdmgr_t *httpdmgr, const char *url,
|
||||
|
||||
item->action = func;
|
||||
item->action_arg = arg;
|
||||
item->isstatic = isstatic;
|
||||
isc_time_now(&item->loadtime);
|
||||
|
||||
ISC_LINK_INIT(item, link);
|
||||
ISC_LIST_APPEND(httpdmgr->urls, item, link);
|
||||
|
||||
|
@@ -26,6 +26,23 @@
|
||||
#include <isc/types.h>
|
||||
#include <isc/mutex.h>
|
||||
#include <isc/task.h>
|
||||
#include <isc/time.h>
|
||||
|
||||
/*%
|
||||
* HTTP urls. These are the URLs we manage, and the function to call to
|
||||
* provide the data for it. We pass in the base url (so the same function
|
||||
* can handle multiple requests), and a structure to fill in to return a
|
||||
* result to the client. We also pass in a pointer to be filled in for
|
||||
* the data cleanup function.
|
||||
*/
|
||||
struct isc_httpdurl {
|
||||
char *url;
|
||||
isc_httpdaction_t *action;
|
||||
void *action_arg;
|
||||
isc_boolean_t isstatic;
|
||||
isc_time_t loadtime;
|
||||
ISC_LINK(isc_httpdurl_t) link;
|
||||
};
|
||||
|
||||
#define HTTPD_EVENTCLASS ISC_EVENTCLASS(4300)
|
||||
#define HTTPD_SHUTDOWN (HTTPD_EVENTCLASS + 0x0001)
|
||||
@@ -49,6 +66,11 @@ isc_result_t
|
||||
isc_httpdmgr_addurl(isc_httpdmgr_t *httpdmgr, const char *url,
|
||||
isc_httpdaction_t *func, void *arg);
|
||||
|
||||
isc_result_t
|
||||
isc_httpdmgr_addurl2(isc_httpdmgr_t *httpdmgr, const char *url,
|
||||
isc_boolean_t isstatic,
|
||||
isc_httpdaction_t *func, void *arg);
|
||||
|
||||
isc_result_t
|
||||
isc_httpd_response(isc_httpd_t *httpd);
|
||||
|
||||
|
@@ -103,7 +103,9 @@ typedef int (*isc_sockfdwatch_t)(isc_task_t *, isc_socket_t *, void *, int);
|
||||
|
||||
/* The following cannot be listed alphabetically due to forward reference */
|
||||
typedef isc_result_t (isc_httpdaction_t)(const char *url,
|
||||
isc_httpdurl_t *urlinfo,
|
||||
const char *querystring,
|
||||
const char *headers,
|
||||
void *arg,
|
||||
unsigned int *retcode,
|
||||
const char **retmsg,
|
||||
|
@@ -315,6 +315,16 @@ isc_time_formathttptimestamp(const isc_time_t *t, char *buf, unsigned int len);
|
||||
*
|
||||
*/
|
||||
|
||||
isc_result_t
|
||||
isc_time_parsehttptimestamp(char *input, isc_time_t *t);
|
||||
/*%<
|
||||
* Parse the time in 'input' into the isc_time_t pointed to by 't',
|
||||
* expecting a format like "Mon, 30 Aug 2000 04:06:47 GMT"
|
||||
*
|
||||
* Requires:
|
||||
*\li 'buf' and 't' are not NULL.
|
||||
*/
|
||||
|
||||
void
|
||||
isc_time_formatISO8601(const isc_time_t *t, char *buf, unsigned int len);
|
||||
/*%<
|
||||
|
@@ -407,6 +407,24 @@ isc_time_formathttptimestamp(const isc_time_t *t, char *buf, unsigned int len) {
|
||||
INSIST(flen < len);
|
||||
}
|
||||
|
||||
isc_result_t
|
||||
isc_time_parsehttptimestamp(char *buf, isc_time_t *t) {
|
||||
struct tm t_tm;
|
||||
time_t when;
|
||||
char *p;
|
||||
|
||||
REQUIRE(buf != NULL);
|
||||
REQUIRE(t != NULL);
|
||||
p = strptime(buf, "%a, %d %b %Y %H:%M:%S %Z", &t_tm);
|
||||
if (p == NULL)
|
||||
return (ISC_R_UNEXPECTED);
|
||||
when = timegm(&t_tm);
|
||||
if (when == -1)
|
||||
return (ISC_R_UNEXPECTED);
|
||||
isc_time_set(t, when, 0);
|
||||
return (ISC_R_SUCCESS);
|
||||
}
|
||||
|
||||
void
|
||||
isc_time_formatISO8601(const isc_time_t *t, char *buf, unsigned int len) {
|
||||
time_t now;
|
||||
|
Reference in New Issue
Block a user