Unwrap `UDSAddress` as `tuple[str, str]`, i.e. sin pid

Since in hindsight the real analog of a net-proto's "bindspace"
(normally its routing layer's addresses-port-set) is more akin to the
"location in the file-system" for a UDS socket file (aka the file's
parent directory) determines whether or not the "port" (aka it's
file-name) collides with any other.

So the `._filedir: Path` is like the allocated "address" and,
the `._filename: Path|str` is basically the "port",

at least in my mind.. Bp

Thinking about fs dirs like a "host address" means you can get
essentially the same benefits/behaviour of say an (ip)
addresses-port-space but using the (current process-namespace's)
filesys-tree. Note that for UDS sockets in particular the
network-namespace is what would normally isolate so called "abstract
sockets" (i.e. UDS sockets that do NOT use file-paths by setting `struct
sockaddr_un.sun_path = 'abstract', see `man unix`); using directories is
even easier and definitely more explicit/readable/immediately-obvious as
a human-user.

As such this reworks all the necessary `UDSAddress` meths,
- `.unwrap()` now returns a `tuple(str(._filedir, str(._filename))`,
- `wrap_address()` now matches UDS on a 2nd tuple `str()` element,
- `.get_root()` no longer passes `maybe_pid`.

AND adjusts `MsgpackUDSStream` to,
- use the new `unwrap_sockpath()` on the `socket.get[sock/peer]name()`
  output before passing directly as `UDSAddress.__init__(filedir, filename)`
  instead of via `.from_addr()`.
- also pass `maybe_pid`s to init since no longer included in the
  unwrapped-type form.
leslies_extra_appendix
Tyler Goodlet 2025-04-03 22:24:24 -04:00
parent 69fbe49d37
commit 2d6b3922a6
2 changed files with 56 additions and 39 deletions

View File

@ -313,10 +313,18 @@ class UDSAddress(Address):
# root actor to create a registry address. # root actor to create a registry address.
maybe_pid: int|None = None, maybe_pid: int|None = None,
): ):
fdir = self._filedir = Path(filedir or self.def_bindspace).absolute() fdir = self._filedir = Path(
fpath = self._filepath = Path(filename) filedir
or
self.def_bindspace
).absolute()
fpath = self._filename = Path(filename)
fp: Path = fdir / fpath fp: Path = fdir / fpath
assert fp.is_absolute() assert (
fp.is_absolute()
and
fp == self.sockpath
)
# to track which "side" is the peer process by reading socket # to track which "side" is the peer process by reading socket
# credentials-info. # credentials-info.
@ -324,7 +332,7 @@ class UDSAddress(Address):
@property @property
def sockpath(self) -> Path: def sockpath(self) -> Path:
return self._filedir / self._filepath return self._filedir / self._filename
@property @property
def is_valid(self) -> bool: def is_valid(self) -> bool:
@ -347,19 +355,20 @@ class UDSAddress(Address):
def from_addr( def from_addr(
cls, cls,
addr: ( addr: (
tuple[Path|str|None, int] tuple[Path|str, Path|str]|Path|str
|Path|str
), ),
) -> UDSAddress: ) -> UDSAddress:
match addr: match addr:
case tuple()|list(): case tuple()|list():
sockpath: Path = Path(addr[0]) filedir = Path(addr[0])
filedir, filename = unwrap_sockpath(sockpath) filename = Path(addr[1])
pid: int = addr[1] # sockpath: Path = Path(addr[0])
# filedir, filename = unwrap_sockpath(sockpath)
# pid: int = addr[1]
return UDSAddress( return UDSAddress(
filedir=filedir, filedir=filedir,
filename=filename, filename=filename,
maybe_pid=pid, # maybe_pid=pid,
) )
# NOTE, in case we ever decide to just `.unwrap()` # NOTE, in case we ever decide to just `.unwrap()`
# to a `Path|str`? # to a `Path|str`?
@ -377,8 +386,8 @@ class UDSAddress(Address):
# XXX NOTE, since this gets passed DIRECTLY to # XXX NOTE, since this gets passed DIRECTLY to
# `.ipc._uds.open_unix_socket_w_passcred()` # `.ipc._uds.open_unix_socket_w_passcred()`
return ( return (
str(self.sockpath), str(self._filedir),
self._pid, str(self._filename),
) )
@classmethod @classmethod
@ -409,18 +418,18 @@ class UDSAddress(Address):
@classmethod @classmethod
def get_root(cls) -> Address: def get_root(cls) -> Address:
def_uds_filepath: Path = 'registry@1616.sock' def_uds_filename: Path = 'registry@1616.sock'
return UDSAddress( return UDSAddress(
filedir=None, filedir=None,
filename=def_uds_filepath, filename=def_uds_filename,
maybe_pid=1616, # maybe_pid=1616,
) )
def __repr__(self) -> str: def __repr__(self) -> str:
return ( return (
f'{type(self).__name__}' f'{type(self).__name__}'
f'[' f'['
f'({self.sockpath}, {self._pid})' f'({self._filedir}, {self._filename})'
f']' f']'
) )
@ -430,7 +439,7 @@ class UDSAddress(Address):
f'Can not compare {type(other)} with {type(self)}' f'Can not compare {type(other)} with {type(self)}'
) )
return self._filepath == other._filepath return self.sockpath == other.sockpath
# async def open_listener(self, **kwargs) -> SocketListener: # async def open_listener(self, **kwargs) -> SocketListener:
async def open_listener( async def open_listener(
@ -520,14 +529,6 @@ def wrap_address(
# if 'sock' in addr[0]: # if 'sock' in addr[0]:
# import pdbp; pdbp.set_trace() # import pdbp; pdbp.set_trace()
match addr: match addr:
# TODO! BUT THIS WILL MATCH FOR TCP !...
# -[ ] so prolly go back to what guille had orig XD
# a plain ol' `str`?
# case ((
# str()|Path(),
# int(),
# )):
# cls = UDSAddress
# classic network socket-address as tuple/list # classic network socket-address as tuple/list
case ( case (
@ -537,6 +538,14 @@ def wrap_address(
): ):
cls = TCPAddress cls = TCPAddress
case (
# (str()|Path(), str()|Path()),
# ^TODO? uhh why doesn't this work!?
(_, filename)
) if type(filename) is str:
cls = UDSAddress
# likely an unset UDS or TCP reg address as defaulted in # likely an unset UDS or TCP reg address as defaulted in
# `_state._runtime_vars['_root_mailbox']` # `_state._runtime_vars['_root_mailbox']`
# #
@ -552,8 +561,9 @@ def wrap_address(
case _: case _:
# import pdbp; pdbp.set_trace() # import pdbp; pdbp.set_trace()
raise TypeError( raise TypeError(
f'Can not wrap address {type(addr)}\n' f'Can not wrap unwrapped-address ??\n'
f'{addr!r}\n' f'type(addr): {type(addr)!r}\n'
f'addr: {addr!r}\n'
) )
return cls.from_addr(addr) return cls.from_addr(addr)

View File

@ -38,7 +38,10 @@ from trio._highlevel_open_unix_stream import (
from tractor.msg import MsgCodec from tractor.msg import MsgCodec
from tractor.log import get_logger from tractor.log import get_logger
from tractor._addr import UDSAddress from tractor._addr import (
UDSAddress,
unwrap_sockpath,
)
from tractor.ipc._transport import MsgpackTransport from tractor.ipc._transport import MsgpackTransport
@ -194,16 +197,20 @@ class MsgpackUDSStream(MsgpackTransport):
case (bytes(), str()): case (bytes(), str()):
sock_path: Path = Path(sockname) sock_path: Path = Path(sockname)
( (
pid, peer_pid,
uid, _,
gid, _,
) = get_peer_info(sock) ) = get_peer_info(sock)
laddr = UDSAddress.from_addr((
sock_path, filedir, filename = unwrap_sockpath(sock_path)
os.getpid(), laddr = UDSAddress(
)) filedir=filedir,
raddr = UDSAddress.from_addr(( filename=filename,
sock_path, maybe_pid=os.getpid(),
pid )
)) raddr = UDSAddress(
filedir=filedir,
filename=filename,
maybe_pid=peer_pid
)
return (laddr, raddr) return (laddr, raddr)