about summary refs log blame commit diff stats
path: root/src/io/serversocket.nim
blob: dff6de70e130208b668258efadf25c8b320a8b27 (plain) (tree)
1
2
3
4
5
6
7
8
9
10


                        
 
                    
                  
 
                               

               
          
 
                                    




                                                          
 


                                                                     








                                                                              
 






                                                                               

                                                                              

                                                            
                  
                                   

                                        
                               








                                                                                


                                                       
              
                                                             
 
                                                 
                   





                                                                 
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: "<unistd.h>".}
  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)