import std/nativesockets import std/net import std/os when defined(posix): import std/posix type ServerSocket* = ref object sock*: Socket path*: string dfd: int const SocketPathPrefix = "cha_sock_" proc getSocketName*(pid: int): string = SocketPathPrefix & $pid proc getSocketPath*(socketDir: string; pid: int): string = socketDir / getSocketName(pid) # The way stdlib does bindUnix is utterly broken at least on FreeBSD. # It seems that just writing it in C is the easiest solution. {.compile: "bind_unix.c".} proc bind_unix_from_c(fd: cint; path: cstring; pathlen: cint): cint {.importc.} when defined(freebsd): # capsicum stuff proc unlinkat(dfd: cint; path: cstring; flag: cint): cint {.importc, header: "".} proc bindat_unix_from_c(dfd, sock: cint; path: cstring; pathlen: cint): cint {.importc.} proc initServerSocket*(fd: SocketHandle; sockDir: string; pid, sockDirFd: int): ServerSocket = let sock = newSocket(fd, Domain.AF_UNIX, SockType.SOCK_STREAM, Protocol.IPPROTO_IP, buffered = false) let path = getSocketPath(sockDir, pid) return ServerSocket(sock: sock, path: path, dfd: sockDirFd) proc initServerSocket*(sockDir: string; sockDirFd, pid: int; blocking = true): ServerSocket = let sock = newSocket(Domain.AF_UNIX, SockType.SOCK_STREAM, Protocol.IPPROTO_IP, buffered = false) if not blocking: sock.getFd().setBlocking(false) let path = getSocketPath(sockDir, pid) if sockDirFd == -1: discard tryRemoveFile(path) if bind_unix_from_c(cint(sock.getFd()), cstring(path), cint(path.len)) != 0: raiseOSError(osLastError()) else: when defined(freebsd): let name = getSocketName(pid) discard unlinkat(cint(sockDirFd), cstring(name), 0) if bindat_unix_from_c(cint(sockDirFd), cint(sock.getFd()), cstring(name), cint(name.len)) != 0: raiseOSError(osLastError()) else: # shouldn't have sockDirFd on other architectures doAssert false listen(sock) return ServerSocket(sock: sock, path: path, dfd: sockDirFd) proc close*(ssock: ServerSocket; unlink = true) = close(ssock.sock) if unlink: when defined(freebsd): if ssock.dfd != -1: discard unlinkat(cint(ssock.dfd), cstring(ssock.path), 0) return discard tryRemoveFile(ssock.path)