@@ -20,7 +20,9 @@ import os
20
20
import posixpath
21
21
import re
22
22
import shutil
23
+ import socket
23
24
import socketserver
25
+ import subprocess
24
26
import sys
25
27
import time
26
28
import urllib .error
@@ -218,8 +220,7 @@ class SimpleHTTPRequestHandler(http.server.BaseHTTPRequestHandler):
218
220
f .write (b"<html>\n " )
219
221
f .write (
220
222
(
221
- '<meta http-equiv="Content-Type" '
222
- 'content="text/html; charset=%s">' % enc
223
+ '<meta http-equiv="Content-Type" content="text/html; charset=%s">' % enc
223
224
).encode (enc )
224
225
)
225
226
f .write (("<title>Directory listing for %s</title>\n " % displaypath ).encode (enc ))
@@ -257,8 +258,15 @@ class SimpleHTTPRequestHandler(http.server.BaseHTTPRequestHandler):
257
258
dirimage = ""
258
259
fullname = os .path .join (path , name )
259
260
displayname = linkname = name
260
- fsize = fbytes (os .path .getsize (fullname ))
261
- created_date = time .ctime (os .path .getctime (fullname ))
261
+
262
+ try :
263
+ fsize = fbytes (os .path .getsize (fullname ))
264
+ created_date = time .ctime (os .path .getctime (fullname ))
265
+ except (FileNotFoundError , OSError ):
266
+ # Handle dangling symlinks and other file access errors
267
+ fsize = "?"
268
+ created_date = "?"
269
+
262
270
# Append / for directories or @ for symbolic links
263
271
if os .path .isdir (fullname ):
264
272
dirimage = ""
@@ -372,7 +380,7 @@ parser.add_argument(
372
380
"-b" ,
373
381
default = "" ,
374
382
metavar = "ADDRESS" ,
375
- help = "Specify alternate bind address " " [default: all interfaces]" ,
383
+ help = "Specify alternate bind address [default: all interfaces]" ,
376
384
)
377
385
parser .add_argument (
378
386
"port" ,
@@ -388,14 +396,53 @@ PORT = args.port
388
396
BIND = args .bind
389
397
HOST = BIND
390
398
391
- if HOST == "" :
392
- HOST = "localhost"
399
+
400
+ def get_ip_addresses ():
401
+ """Get all IPv4 addresses for this host."""
402
+ addresses = []
403
+
404
+ try :
405
+ # Try ip addr first (modern systems)
406
+ output = subprocess .check_output (["ip" , "addr" ], text = True )
407
+ for line in output .split ("\n " ):
408
+ if "inet " in line and "scope global" in line :
409
+ ip = line .split ()[1 ].split ("/" )[0 ]
410
+ addresses .append (ip )
411
+ except (subprocess .CalledProcessError , FileNotFoundError ):
412
+ try :
413
+ # Fall back to ifconfig
414
+ output = subprocess .check_output (["ifconfig" ], text = True )
415
+ for line in output .split ("\n " ):
416
+ if "inet " in line and "broadcast" in line :
417
+ ip = line .split ()[1 ]
418
+ addresses .append (ip )
419
+ except (subprocess .CalledProcessError , FileNotFoundError ):
420
+ # If both fail, just use socket to get hostname IP
421
+ try :
422
+ hostname_ip = socket .gethostbyname (socket .gethostname ())
423
+ if hostname_ip != "127.0.0.1" :
424
+ addresses .append (hostname_ip )
425
+ except socket .gaierror :
426
+ pass
427
+
428
+ # Always include localhost
429
+ if "127.0.0.1" not in addresses :
430
+ addresses .append ("127.0.0.1" )
431
+
432
+ return sorted (addresses ) # Sort for consistent output
433
+
434
+
435
+ if HOST in ("" , "0.0.0.0" ):
436
+ HOST = "0.0.0.0" # Listen on all interfaces
437
+ print ("Server accessible at:" )
438
+ for addr in get_ip_addresses ():
439
+ print (f" http://{ addr } :{ PORT } /" )
440
+ else :
441
+ print (f"Server accessible at: http://{ HOST } :{ PORT } /" )
393
442
394
443
Handler = SimpleHTTPRequestHandler
395
444
396
445
with socketserver .TCPServer ((BIND , PORT ), Handler ) as httpd :
397
- serve_message = "Serving HTTP on {host} port {port} (http://{host}:{port}/) ..."
398
- print (serve_message .format (host = HOST , port = PORT ))
399
446
try :
400
447
httpd .serve_forever ()
401
448
except KeyboardInterrupt :
0 commit comments