Skip to content

Commit

Permalink
httpserve: handle dangling symlinks & show listening ip addresses
Browse files Browse the repository at this point in the history
  • Loading branch information
elohmeier committed Feb 10, 2025
1 parent 50498a4 commit e966411
Showing 1 changed file with 56 additions and 9 deletions.
65 changes: 56 additions & 9 deletions bin/httpserve
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,9 @@ import os
import posixpath
import re
import shutil
import socket
import socketserver
import subprocess
import sys
import time
import urllib.error
Expand Down Expand Up @@ -218,8 +220,7 @@ class SimpleHTTPRequestHandler(http.server.BaseHTTPRequestHandler):
f.write(b"<html>\n")
f.write(
(
'<meta http-equiv="Content-Type" '
'content="text/html; charset=%s">' % enc
'<meta http-equiv="Content-Type" content="text/html; charset=%s">' % enc
).encode(enc)
)
f.write(("<title>Directory listing for %s</title>\n" % displaypath).encode(enc))
Expand Down Expand Up @@ -257,8 +258,15 @@ class SimpleHTTPRequestHandler(http.server.BaseHTTPRequestHandler):
dirimage = ""
fullname = os.path.join(path, name)
displayname = linkname = name
fsize = fbytes(os.path.getsize(fullname))
created_date = time.ctime(os.path.getctime(fullname))

try:
fsize = fbytes(os.path.getsize(fullname))
created_date = time.ctime(os.path.getctime(fullname))
except (FileNotFoundError, OSError):
# Handle dangling symlinks and other file access errors
fsize = "?"
created_date = "?"

# Append / for directories or @ for symbolic links
if os.path.isdir(fullname):
dirimage = ""
Expand Down Expand Up @@ -372,7 +380,7 @@ parser.add_argument(
"-b",
default="",
metavar="ADDRESS",
help="Specify alternate bind address " "[default: all interfaces]",
help="Specify alternate bind address [default: all interfaces]",
)
parser.add_argument(
"port",
Expand All @@ -388,14 +396,53 @@ PORT = args.port
BIND = args.bind
HOST = BIND

if HOST == "":
HOST = "localhost"

def get_ip_addresses():
"""Get all IPv4 addresses for this host."""
addresses = []

try:
# Try ip addr first (modern systems)
output = subprocess.check_output(["ip", "addr"], text=True)
for line in output.split("\n"):
if "inet " in line and "scope global" in line:
ip = line.split()[1].split("/")[0]
addresses.append(ip)
except (subprocess.CalledProcessError, FileNotFoundError):
try:
# Fall back to ifconfig
output = subprocess.check_output(["ifconfig"], text=True)
for line in output.split("\n"):
if "inet " in line and "broadcast" in line:
ip = line.split()[1]
addresses.append(ip)
except (subprocess.CalledProcessError, FileNotFoundError):
# If both fail, just use socket to get hostname IP
try:
hostname_ip = socket.gethostbyname(socket.gethostname())
if hostname_ip != "127.0.0.1":
addresses.append(hostname_ip)
except socket.gaierror:
pass

# Always include localhost
if "127.0.0.1" not in addresses:
addresses.append("127.0.0.1")

return sorted(addresses) # Sort for consistent output


if HOST in ("", "0.0.0.0"):
HOST = "0.0.0.0" # Listen on all interfaces
print("Server accessible at:")
for addr in get_ip_addresses():
print(f" http://{addr}:{PORT}/")
else:
print(f"Server accessible at: http://{HOST}:{PORT}/")

Handler = SimpleHTTPRequestHandler

with socketserver.TCPServer((BIND, PORT), Handler) as httpd:
serve_message = "Serving HTTP on {host} port {port} (http://{host}:{port}/) ..."
print(serve_message.format(host=HOST, port=PORT))
try:
httpd.serve_forever()
except KeyboardInterrupt:
Expand Down

0 comments on commit e966411

Please sign in to comment.