mirror of
https://github.com/waku-org/nwaku.git
synced 2025-01-15 09:26:38 +00:00
285 lines
9.9 KiB
Nim
285 lines
9.9 KiB
Nim
# Copyright (c) 2019 Status Research & Development GmbH
|
|
# Licensed under either of
|
|
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
|
|
# * MIT license ([LICENSE-MIT](LICENSE-MIT))
|
|
# at your option.
|
|
# This file may not be copied, modified, or distributed except according to
|
|
# those terms.
|
|
|
|
################################
|
|
# headers and library location #
|
|
################################
|
|
|
|
import os
|
|
when defined(windows):
|
|
import winlean
|
|
else:
|
|
import posix
|
|
|
|
when defined(libnatpmpUseSystemLibs):
|
|
{.passL: "-lnatpmp".}
|
|
else:
|
|
const includePath = currentSourcePath.parentDir().parentDir() / "vendor" / "libnatpmp-upstream"
|
|
{.passC: "-I" & includePath.}
|
|
{.passL: includePath / "libnatpmp.a".}
|
|
|
|
when defined(windows):
|
|
import nativesockets # for that wsaStartup() call at the end
|
|
{.passC: "-DNATPMP_STATICLIB".}
|
|
{.passL: "-lws2_32 -liphlpapi".}
|
|
|
|
############
|
|
# natpmp.h #
|
|
############
|
|
|
|
{.passC: "-DENABLE_STRNATPMPERR".}
|
|
|
|
## NAT-PMP Port as defined by the NAT-PMP draft
|
|
const NATPMP_PORT* = cint(5351)
|
|
|
|
type
|
|
PublicAddressStruct* {.importc: "struct no_name", header: "natpmp.h", bycopy.} = object
|
|
address* {.importc: "addr".}: InAddr ## culong addr;
|
|
|
|
NewPortMappingStruct* {.importc: "struct no_name", header: "natpmp.h", bycopy.} = object
|
|
privateport* {.importc: "privateport".}: cushort
|
|
mappedpublicport* {.importc: "mappedpublicport".}: cushort
|
|
lifetime* {.importc: "lifetime".}: culong
|
|
|
|
PnuUnion* {.importc: "struct no_name", header: "natpmp.h", bycopy, union.} = object
|
|
publicaddress* {.importc: "publicaddress".}: PublicAddressStruct
|
|
newportmapping* {.importc: "newportmapping".}: NewPortMappingStruct
|
|
|
|
natpmp_t* {.importc: "natpmp_t", header: "natpmp.h", bycopy.} = object
|
|
s* {.importc: "s".}: cint ## socket
|
|
gateway* {.importc: "gateway".}: culong ## default gateway (IPv4)
|
|
has_pending_request* {.importc: "has_pending_request".}: cint
|
|
pending_request* {.importc: "pending_request".}: array[12, cuchar]
|
|
pending_request_len* {.importc: "pending_request_len".}: cint
|
|
try_number* {.importc: "try_number".}: cint
|
|
retry_time* {.importc: "retry_time".}: Timeval
|
|
|
|
natpmpresp_t* {.importc: "natpmpresp_t", header: "natpmp.h", bycopy.} = object
|
|
resptype* {.importc: "type".}: cushort ## NATPMP_RESPTYPE_*
|
|
resultcode* {.importc: "resultcode".}: cushort ## NAT-PMP response code
|
|
epoch* {.importc: "epoch".}: culong ## Seconds since start of epoch
|
|
pnu* {.importc: "pnu".}: PnuUnion
|
|
|
|
|
|
## possible values for type field of natpmpresp_t
|
|
const
|
|
NATPMP_RESPTYPE_PUBLICADDRESS* = cushort(0)
|
|
NATPMP_RESPTYPE_UDPPORTMAPPING* = cushort(1)
|
|
NATPMP_RESPTYPE_TCPPORTMAPPING* = cushort(2)
|
|
|
|
## Values to pass to sendnewportmappingrequest()
|
|
const
|
|
NATPMP_PROTOCOL_UDP* = 1
|
|
NATPMP_PROTOCOL_TCP* = 2
|
|
|
|
## return values
|
|
|
|
## NATPMP_ERR_INVALIDARGS : invalid arguments passed to the function
|
|
const NATPMP_ERR_INVALIDARGS* = cint(-1)
|
|
## NATPMP_ERR_SOCKETERROR : socket() failed. check errno for details
|
|
const NATPMP_ERR_SOCKETERROR* = cint(-2)
|
|
## NATPMP_ERR_CANNOTGETGATEWAY : can't get default gateway IP
|
|
const NATPMP_ERR_CANNOTGETGATEWAY* = cint(-3)
|
|
## NATPMP_ERR_CLOSEERR : close() failed. check errno for details
|
|
const NATPMP_ERR_CLOSEERR* = cint(-4)
|
|
## NATPMP_ERR_RECVFROM : recvfrom() failed. check errno for details
|
|
const NATPMP_ERR_RECVFROM* = cint(-5)
|
|
## NATPMP_ERR_NOPENDINGREQ : readnatpmpresponseorretry() called while no NAT-PMP request was pending
|
|
const NATPMP_ERR_NOPENDINGREQ* = cint(-6)
|
|
## NATPMP_ERR_NOGATEWAYSUPPORT : the gateway does not support NAT-PMP
|
|
const NATPMP_ERR_NOGATEWAYSUPPORT* = cint(-7)
|
|
## NATPMP_ERR_CONNECTERR : connect() failed. check errno for details
|
|
const NATPMP_ERR_CONNECTERR* = cint(-8)
|
|
## NATPMP_ERR_WRONGPACKETSOURCE : packet not received from the network gateway
|
|
const NATPMP_ERR_WRONGPACKETSOURCE* = cint(-9)
|
|
## NATPMP_ERR_SENDERR : send() failed. check errno for details
|
|
const NATPMP_ERR_SENDERR* = cint(-10)
|
|
## NATPMP_ERR_FCNTLERROR : fcntl() failed. check errno for details
|
|
const NATPMP_ERR_FCNTLERROR* = cint(-11)
|
|
## NATPMP_ERR_GETTIMEOFDAYERR : gettimeofday() failed. check errno for details
|
|
const NATPMP_ERR_GETTIMEOFDAYERR* = cint(-12)
|
|
|
|
const NATPMP_ERR_UNSUPPORTEDVERSION* = cint(-14)
|
|
const NATPMP_ERR_UNSUPPORTEDOPCODE* = cint(-15)
|
|
|
|
## Errors from the server :
|
|
const NATPMP_ERR_UNDEFINEDERROR* = cint(-49)
|
|
const NATPMP_ERR_NOTAUTHORIZED* = cint(-51)
|
|
const NATPMP_ERR_NETWORKFAILURE* = cint(-52)
|
|
const NATPMP_ERR_OUTOFRESOURCES* = cint(-53)
|
|
## NATPMP_TRYAGAIN : no data available for the moment. try again later
|
|
const NATPMP_TRYAGAIN* = cint(-100)
|
|
|
|
## initnatpmp()
|
|
## initialize a natpmp_t object
|
|
## With forcegw=1 the gateway is not detected automaticaly.
|
|
## Return values :
|
|
## 0 = OK
|
|
## NATPMP_ERR_INVALIDARGS
|
|
## NATPMP_ERR_SOCKETERROR
|
|
## NATPMP_ERR_FCNTLERROR
|
|
## NATPMP_ERR_CANNOTGETGATEWAY
|
|
## NATPMP_ERR_CONNECTERR
|
|
proc initnatpmp*(p: ptr natpmp_t; forcegw: cint; forcedgw: culong): cint {.importc: "initnatpmp", header: "natpmp.h".}
|
|
|
|
## closenatpmp()
|
|
## close resources associated with a natpmp_t object
|
|
## Return values :
|
|
## 0 = OK
|
|
## NATPMP_ERR_INVALIDARGS
|
|
## NATPMP_ERR_CLOSEERR
|
|
proc closenatpmp*(p: ptr natpmp_t): cint {.importc: "closenatpmp", header: "natpmp.h".}
|
|
|
|
## sendpublicaddressrequest()
|
|
## send a public address NAT-PMP request to the network gateway
|
|
## Return values :
|
|
## 2 = OK (size of the request)
|
|
## NATPMP_ERR_INVALIDARGS
|
|
## NATPMP_ERR_SENDERR
|
|
proc sendpublicaddressrequest*(p: ptr natpmp_t): cint {.importc: "sendpublicaddressrequest", header: "natpmp.h".}
|
|
|
|
## sendnewportmappingrequest()
|
|
## send a new port mapping NAT-PMP request to the network gateway
|
|
## Arguments :
|
|
## protocol is either NATPMP_PROTOCOL_TCP or NATPMP_PROTOCOL_UDP,
|
|
## lifetime is in seconds.
|
|
## To remove a port mapping, set lifetime to zero.
|
|
## To remove all port mappings to the host, set lifetime and both ports
|
|
## to zero.
|
|
## Return values :
|
|
## 12 = OK (size of the request)
|
|
## NATPMP_ERR_INVALIDARGS
|
|
## NATPMP_ERR_SENDERR
|
|
proc sendnewportmappingrequest*(p: ptr natpmp_t; protocol: cint;
|
|
privateport: cushort; publicport: cushort;
|
|
lifetime: culong): cint {.importc: "sendnewportmappingrequest", header: "natpmp.h".}
|
|
|
|
## getnatpmprequesttimeout()
|
|
## fills the timeval structure with the timeout duration of the
|
|
## currently pending NAT-PMP request.
|
|
## Return values :
|
|
## 0 = OK
|
|
## NATPMP_ERR_INVALIDARGS
|
|
## NATPMP_ERR_GETTIMEOFDAYERR
|
|
## NATPMP_ERR_NOPENDINGREQ
|
|
proc getnatpmprequesttimeout*(p: ptr natpmp_t; timeout: ptr Timeval): cint {.importc: "getnatpmprequesttimeout", header: "natpmp.h".}
|
|
|
|
## readnatpmpresponseorretry()
|
|
## fills the natpmpresp_t structure if possible
|
|
## Return values :
|
|
## 0 = OK
|
|
## NATPMP_TRYAGAIN
|
|
## NATPMP_ERR_INVALIDARGS
|
|
## NATPMP_ERR_NOPENDINGREQ
|
|
## NATPMP_ERR_NOGATEWAYSUPPORT
|
|
## NATPMP_ERR_RECVFROM
|
|
## NATPMP_ERR_WRONGPACKETSOURCE
|
|
## NATPMP_ERR_UNSUPPORTEDVERSION
|
|
## NATPMP_ERR_UNSUPPORTEDOPCODE
|
|
## NATPMP_ERR_NOTAUTHORIZED
|
|
## NATPMP_ERR_NETWORKFAILURE
|
|
## NATPMP_ERR_OUTOFRESOURCES
|
|
## NATPMP_ERR_UNSUPPORTEDOPCODE
|
|
## NATPMP_ERR_UNDEFINEDERROR
|
|
proc readnatpmpresponseorretry*(p: ptr natpmp_t; response: ptr natpmpresp_t): cint {.importc: "readnatpmpresponseorretry", header: "natpmp.h".}
|
|
|
|
proc strnatpmperr*(t: cint): cstring {.importc: "strnatpmperr", header: "natpmp.h".}
|
|
|
|
###################
|
|
# custom wrappers #
|
|
###################
|
|
|
|
import stew/result
|
|
|
|
type NatPmp* {.packed.} = ref object
|
|
cstruct*: natpmp_t
|
|
|
|
proc natpmpFinalizer(x: NatPmp) =
|
|
discard closenatpmp(addr(x.cstruct))
|
|
|
|
proc newNatPmp*(): NatPmp =
|
|
new(result, natpmpFinalizer)
|
|
|
|
proc init*(self: NatPmp): Result[bool, cstring] =
|
|
let res = initnatpmp(addr(self.cstruct), 0, 0)
|
|
if res == 0:
|
|
result.ok(true)
|
|
else:
|
|
result.err(strnatpmperr(res))
|
|
|
|
proc `=deepCopy`(x: NatPmp): NatPmp =
|
|
doAssert(false, "not implemented")
|
|
|
|
proc getNatPmpResponse(self: NatPmp, natPmpResponsePtr: ptr natpmpresp_t): Result[bool, cstring] =
|
|
var
|
|
res: cint
|
|
timeout: Timeval
|
|
fds: TFdSet
|
|
|
|
while true:
|
|
FD_ZERO(fds);
|
|
FD_SET(SocketHandle(self.cstruct.s), fds)
|
|
res = getnatpmprequesttimeout(addr(self.cstruct), addr(timeout))
|
|
if res != 0:
|
|
result.err(strnatpmperr(res))
|
|
return
|
|
res = select(FD_SETSIZE, addr(fds), nil, nil, addr(timeout))
|
|
if res == -1:
|
|
result.err(osErrorMsg(osLastError()))
|
|
return
|
|
res = readnatpmpresponseorretry(addr(self.cstruct), natPmpResponsePtr)
|
|
if res < 0 and res != NATPMP_TRYAGAIN:
|
|
result.err(strnatpmperr(res))
|
|
return
|
|
if res != NATPMP_TRYAGAIN:
|
|
break
|
|
|
|
result.ok(true)
|
|
|
|
|
|
proc externalIPAddress*(self: NatPmp): Result[cstring, cstring] =
|
|
var
|
|
res: cint
|
|
natPmpResponse: natpmpresp_t
|
|
|
|
res = sendpublicaddressrequest(addr(self.cstruct))
|
|
if res < 0:
|
|
result.err(strnatpmperr(res))
|
|
return
|
|
if (let r = self.getNatPmpResponse(addr(natPmpResponse)); r.isErr):
|
|
result.err(r.error)
|
|
return
|
|
result.ok(inet_ntoa(natPmpResponse.pnu.publicaddress.address))
|
|
|
|
type NatPmpProtocol* = enum
|
|
UDP = NATPMP_PROTOCOL_UDP
|
|
TCP = NATPMP_PROTOCOL_TCP
|
|
|
|
proc doMapping(self: NatPmp, eport: cushort, iport: cushort, protocol: NatPmpProtocol, lifetime: culong): Result[cushort, cstring] =
|
|
var
|
|
res: cint
|
|
natPmpResponse: natpmpresp_t
|
|
|
|
res = sendnewportmappingrequest(addr(self.cstruct), protocol.cint, iport, eport, lifetime)
|
|
if res < 0:
|
|
result.err(strnatpmperr(res))
|
|
return
|
|
if (let r = self.getNatPmpResponse(addr(natPmpResponse)); r.isErr):
|
|
result.err(r.error)
|
|
return
|
|
result.ok(natPmpResponse.pnu.newportmapping.mappedpublicport)
|
|
|
|
## returns the mapped external port (might be different than the one requested)
|
|
## "lifetime" is in seconds
|
|
proc addPortMapping*(self: NatPmp, eport: cushort, iport: cushort, protocol: NatPmpProtocol, lifetime: culong): Result[cushort, cstring] =
|
|
return self.doMapping(eport, iport, protocol, lifetime)
|
|
|
|
proc deletePortMapping*(self: NatPmp, eport: cushort, iport: cushort, protocol: NatPmpProtocol): Result[cushort, cstring] =
|
|
return self.doMapping(eport, iport, protocol, 0)
|
|
|