blob: 40ce9bb261a2c25488cde41351f5f092ca841374 [file] [log] [blame]
#!/usr/bin/env python3
"""
Copyright (c) 2014-2024, Regents of the University of California,
Arizona Board of Regents,
Colorado State University,
University Pierre & Marie Curie, Sorbonne University,
Washington University in St. Louis,
Beijing Institute of Technology,
The University of Memphis.
This file is part of NFD (Named Data Networking Forwarding Daemon).
See AUTHORS.md for complete list of NFD authors and contributors.
NFD is free software: you can redistribute it and/or modify it under the terms
of the GNU General Public License as published by the Free Software Foundation,
either version 3 of the License, or (at your option) any later version.
NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
"""
import argparse
import ipaddress
import os
import socket
import subprocess
from http.server import SimpleHTTPRequestHandler, ThreadingHTTPServer
from urllib.parse import urlsplit
class NfdStatusHandler(SimpleHTTPRequestHandler):
"""The handler class to handle HTTP requests."""
def do_GET(self):
path = urlsplit(self.path).path
if path == "/":
self.__serve_report()
elif path == "/robots.txt" and self.server.allow_robots:
self.send_error(404)
else:
super().do_GET()
def __serve_report(self):
"""Obtain XML-formatted NFD status report and send it back as response body."""
try:
proc = subprocess.run(
["nfdc", "status", "report", "xml"], capture_output=True, check=True, text=True, timeout=10
)
output = proc.stdout
except OSError as err:
super().log_message(f"error invoking nfdc: {err}")
self.send_error(500)
except subprocess.SubprocessError as err:
super().log_message(f"error invoking nfdc: {err}")
self.send_error(504, "Cannot connect to NFD")
else:
# add stylesheet processing instruction after the XML document type declaration
# (yes, this is a ugly hack)
if (pos := output.find(">") + 1) != 0:
xml = output[:pos] + '<?xml-stylesheet type="text/xsl" href="nfd-status.xsl"?>' + output[pos:]
self.send_response(200)
self.send_header("Content-Type", "text/xml; charset=UTF-8")
self.end_headers()
self.wfile.write(xml.encode())
else:
super().log_message(f"malformed nfdc output: {output}")
self.send_error(500)
# override
def log_message(self, *args):
if self.server.verbose:
super().log_message(*args)
class NfdStatusHttpServer(ThreadingHTTPServer):
def __init__(self, addr, port, *, robots=False, verbose=False):
# socketserver.BaseServer defaults to AF_INET even if you provide an IPv6 address
# see https://bugs.python.org/issue20215 and https://bugs.python.org/issue24209
if addr.version == 6:
self.address_family = socket.AF_INET6
self.allow_robots = robots
self.verbose = verbose
super().__init__((str(addr), port), NfdStatusHandler)
def main():
def ip_address(arg, /):
"""Validate IP address."""
try:
value = ipaddress.ip_address(arg)
except ValueError:
raise argparse.ArgumentTypeError(f"{arg!r} is not a valid IP address")
return value
def port_number(arg, /):
"""Validate port number."""
try:
value = int(arg)
except ValueError:
value = -1
if value < 0 or value > 65535:
raise argparse.ArgumentTypeError(f"{arg!r} is not a valid port number")
return value
parser = argparse.ArgumentParser(description="Serves NFD status page via HTTP")
parser.add_argument("-V", "--version", action="version", version="@VERSION@")
parser.add_argument("-a", "--address", default="127.0.0.1", type=ip_address, metavar="ADDR",
help="bind to this IP address (default: %(default)s)")
parser.add_argument("-p", "--port", default=6380, type=port_number,
help="bind to this port number (default: %(default)s)")
parser.add_argument("-f", "--workdir", default="@DATAROOTDIR@/ndn", metavar="DIR",
help="server's working directory (default: %(default)s)")
parser.add_argument("-r", "--robots", action="store_true", help="allow crawlers and other HTTP bots")
parser.add_argument("-v", "--verbose", action="store_true", help="turn on verbose logging")
args = parser.parse_args()
os.chdir(args.workdir)
with NfdStatusHttpServer(args.address, args.port, robots=args.robots, verbose=args.verbose) as httpd:
if httpd.address_family == socket.AF_INET6:
url = "http://[{}]:{}"
else:
url = "http://{}:{}"
print("Server started at", url.format(*httpd.server_address))
try:
httpd.serve_forever()
except KeyboardInterrupt:
pass
if __name__ == "__main__":
main()