Skip to content

Commit

Permalink
feat: Add socket-activated systemd service for discovery
Browse files Browse the repository at this point in the history
Introduce a system-wide daemon listening on /run/wsdd.socket,
which is managed by systemd.
When a client first connects to the socket the daemon is started.
Multiple clients can connect and safely use the API simultaneously.

The service runs in discovery-only mode.

Uses multicast source-port 37020.
  • Loading branch information
aleasto committed Oct 14, 2024
1 parent 006a799 commit c25ec56
Show file tree
Hide file tree
Showing 4 changed files with 45 additions and 1 deletion.
17 changes: 17 additions & 0 deletions etc/systemd/wsdd-discovery.service
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
[Unit]
Description=Web Services Dynamic Discovery service
Documentation=man:wsdd(8)
Requires=wsdd-discovery.socket

[Service]
Type=simple
; Use /etc/default/wsdd for defaults, if it exists
EnvironmentFile=-/etc/default/wsdd
; The service is put into an empty runtime directory chroot,
; i.e. the runtime directory which usually resides under /run
ExecStart=/usr/bin/wsdd --shortlog --chroot=/run/wsdd-discovery --source-port=37020 --no-host --discovery $WSDD_DISCOVERY_PARAMS
DynamicUser=yes
User=wsdd-discovery
Group=wsdd-discovery
RuntimeDirectory=wsdd-discovery
AmbientCapabilities=CAP_SYS_CHROOT
9 changes: 9 additions & 0 deletions etc/systemd/wsdd-discovery.socket
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
[Unit]
Description=Web Services Dynamic Discovery API socket
Documentation=man:wsdd(8)

[Socket]
ListenStream=%t/wsdd.socket

[Install]
WantedBy=sockets.target
1 change: 1 addition & 0 deletions etc/systemd/wsdd.defaults
Original file line number Diff line number Diff line change
Expand Up @@ -3,3 +3,4 @@
# Refer to the wsdd(8) man page for details

WSDD_PARAMS=""
WSDD_DISCOVERY_PARAMS=""
19 changes: 18 additions & 1 deletion src/wsdd.py
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,11 @@
except ModuleNotFoundError:
from xml.etree.ElementTree import fromstring as ETfromString

try:
import systemd.daemon
except ModuleNotFoundError:
# Non-systemd host
pass

WSDD_VERSION: str = '0.8'

Expand Down Expand Up @@ -1137,7 +1142,10 @@ async def create_server(self, aio_loop: asyncio.AbstractEventLoop, listen_addres
# It appears mypy is not able to check the argument to create_task and the return value of start_server
# correctly. The docs say start_server returns a coroutine and the create_task takes a coro. And: It works.
# Thus, we ignore type errors here.
if isinstance(listen_address, int) or listen_address.isnumeric():
if isinstance(listen_address, socket.SocketType):
self.server = await aio_loop.create_task(asyncio.start_unix_server( # type: ignore
self.on_connect, sock=listen_address))
elif isinstance(listen_address, int) or listen_address.isnumeric():
self.server = await aio_loop.create_task(asyncio.start_server( # type: ignore
self.on_connect, host='localhost', port=int(listen_address), reuse_address=True,
reuse_port=True))
Expand Down Expand Up @@ -2010,6 +2018,15 @@ def main() -> int:
api_server = None
if args.listen:
api_server = ApiServer(aio_loop, args.listen, nm)
else:
fds = []
try:
fds = systemd.daemon.listen_fds()
except NameError:
# Non-systemd host
pass
if fds:
api_server = ApiServer(aio_loop, socket.socket(fileno=fds[0]), nm)

# get uid:gid before potential chroot'ing
if args.user is not None:
Expand Down

0 comments on commit c25ec56

Please sign in to comment.