2019-04-15 04:27:12 +03:00
|
|
|
#
|
|
|
|
# Chronos OS utilities
|
|
|
|
# (c) Copyright 2018-Present
|
|
|
|
# Status Research & Development GmbH
|
|
|
|
#
|
|
|
|
# Licensed under either of
|
|
|
|
# Apache License, version 2.0, (LICENSE-APACHEv2)
|
|
|
|
# MIT license (LICENSE-MIT)
|
|
|
|
|
|
|
|
## This module implements cross-platform network interfaces list.
|
|
|
|
## Currently supported OSes are Windows, Linux, MacOS, BSD(not tested).
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
|
2023-06-05 22:21:50 +02:00
|
|
|
{.push raises: [].}
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
|
|
|
|
import std/algorithm
|
2023-02-21 12:48:36 +02:00
|
|
|
import ".."/osdefs
|
|
|
|
import "."/ipnet
|
2019-04-15 04:27:12 +03:00
|
|
|
export ipnet
|
|
|
|
|
|
|
|
const
|
2023-02-21 12:48:36 +02:00
|
|
|
MaxAdapterAddressLength* =
|
|
|
|
when defined(windows):
|
|
|
|
MAX_ADAPTER_ADDRESS_LENGTH
|
|
|
|
else:
|
|
|
|
8
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
type
|
|
|
|
InterfaceType* = enum
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
IfError = 0, # This is workaround element for ProveInit warnings.
|
2019-04-15 04:27:12 +03:00
|
|
|
IfOther = 1,
|
|
|
|
IfRegular1822 = 2,
|
|
|
|
IfHdh1822 = 3,
|
|
|
|
IfDdnX25 = 4,
|
|
|
|
IfRfc877X25 = 5,
|
|
|
|
IfEthernetCsmacd = 6,
|
|
|
|
IfIso88023Csmacd = 7,
|
|
|
|
IfIso88024TokenBus = 8,
|
|
|
|
IfIso88025TokenRing = 9,
|
|
|
|
IfIso88026MAN = 10,
|
|
|
|
IfStarlan = 11,
|
|
|
|
IfProteon10Mbit = 12,
|
|
|
|
IfProteon80Mbit = 13,
|
|
|
|
IfHyperChannel = 14,
|
|
|
|
IfFddi = 15,
|
|
|
|
IfLapB = 16,
|
|
|
|
IfSdlc = 17,
|
|
|
|
IfDs1 = 18,
|
|
|
|
IfE1 = 19,
|
|
|
|
IfBasicIsdn = 20,
|
|
|
|
IfPrimaryIsdn = 21,
|
|
|
|
IfPropPoint2PointSerial = 22,
|
|
|
|
IfPpp = 23,
|
|
|
|
IfSoftwareLoopback = 24,
|
|
|
|
IfEon = 25,
|
|
|
|
IfEthernet3Mbit = 26,
|
|
|
|
IfNsip = 27,
|
|
|
|
IfSlip = 28,
|
|
|
|
IfUltra = 29,
|
|
|
|
IfDs3 = 30,
|
|
|
|
IfSip = 31,
|
|
|
|
IfFrameRelay = 32,
|
|
|
|
IfRs232 = 33,
|
|
|
|
IfPara = 34,
|
|
|
|
IfArcNet = 35,
|
|
|
|
IfArcNetPlus = 36,
|
|
|
|
IfAtm = 37,
|
|
|
|
IfMioX25 = 38,
|
|
|
|
IfSonet = 39,
|
|
|
|
IfX25Ple = 40,
|
|
|
|
IfIso88022Llc = 41,
|
|
|
|
IfLocalTalk = 42,
|
|
|
|
IfSmdsDxi = 43,
|
|
|
|
IfFrameRelayService = 44,
|
|
|
|
IfV35 = 45,
|
|
|
|
IfHssi = 46,
|
|
|
|
IfHippi = 47,
|
|
|
|
IfModem = 48,
|
|
|
|
IfAal5 = 49,
|
|
|
|
IfSonetPath = 50,
|
|
|
|
IfSonetVt = 51,
|
|
|
|
IfSmdsIcip = 52,
|
|
|
|
IfPropVirtual = 53,
|
|
|
|
IfPropMultiplexor = 54,
|
|
|
|
IfIeee80212 = 55,
|
|
|
|
IfFibreChannel = 56,
|
|
|
|
IfHippiInterface = 57,
|
|
|
|
IfFrameRelayInterconnect = 58,
|
|
|
|
IfAflane8023 = 59,
|
|
|
|
IfAflane8025 = 60,
|
|
|
|
IfCctemul = 61,
|
|
|
|
IfFastEther = 62,
|
|
|
|
IfIsdn = 63,
|
|
|
|
IfV11 = 64,
|
|
|
|
IfV36 = 65,
|
|
|
|
IfG70364K = 66,
|
|
|
|
IfG7032MB = 67,
|
|
|
|
IfQllc = 68,
|
|
|
|
IfFastEtherFx = 69,
|
|
|
|
IfChannel = 70,
|
|
|
|
IfIeee80211 = 71,
|
|
|
|
IfIbm370Parchan = 72,
|
|
|
|
IfEscon = 73,
|
|
|
|
IfDlsw = 74,
|
|
|
|
IfIsdnS = 75,
|
|
|
|
IfIsdnU = 76,
|
|
|
|
IfLapD = 77,
|
|
|
|
IfIpSwitch = 78,
|
|
|
|
IfRsrb = 79,
|
|
|
|
IfAtmLogical = 80,
|
|
|
|
IfDs0 = 81,
|
|
|
|
IfDs0Bundle = 82,
|
|
|
|
IfBsc = 83,
|
|
|
|
IfAsync = 84,
|
|
|
|
IfCnr = 85,
|
|
|
|
IfIso88025rDtr = 86,
|
|
|
|
IfEplrs = 87,
|
|
|
|
IfArap = 88,
|
|
|
|
IfPropCnls = 89,
|
|
|
|
IfHostPad = 90,
|
|
|
|
IfTermPad = 91,
|
|
|
|
IfFrameRelayMpi = 92,
|
|
|
|
IfX213 = 93,
|
|
|
|
IfAdsl = 94,
|
|
|
|
IfRadsl = 95,
|
|
|
|
IfSdsl = 96,
|
|
|
|
IfVdsl = 97,
|
|
|
|
IfIso88025Crfprint = 98,
|
|
|
|
IfMyrInet = 99,
|
|
|
|
IfVoiceEm = 100,
|
|
|
|
IfVoiceFxo = 101,
|
|
|
|
IfVoiceFxs = 102,
|
|
|
|
IfVoiceEncap = 103,
|
|
|
|
IfVoiceOverip = 104,
|
|
|
|
IfAtmDxi = 105,
|
|
|
|
IfAtmFuni = 106,
|
|
|
|
IfAtmIma = 107,
|
|
|
|
IfPppMultilinkBundle = 108,
|
|
|
|
IfIpoverCdlc = 109,
|
|
|
|
IfIpoverClaw = 110,
|
|
|
|
IfStackToStack = 111,
|
|
|
|
IfVirtualIpAddress = 112,
|
|
|
|
IfMpc = 113,
|
|
|
|
IfIpoverAtm = 114,
|
|
|
|
IfIso88025Fiber = 115,
|
|
|
|
IfTdlc = 116,
|
|
|
|
IfGigabitEthernet = 117,
|
|
|
|
IfHdlc = 118,
|
|
|
|
IfLapF = 119,
|
|
|
|
IfV37 = 120,
|
|
|
|
IfX25Mlp = 121,
|
|
|
|
IfX25HuntGroup = 122,
|
|
|
|
IfTransPhdlc = 123,
|
|
|
|
IfInterleave = 124,
|
|
|
|
IfFast = 125,
|
|
|
|
IfIp = 126,
|
|
|
|
IfDocScableMaclayer = 127,
|
|
|
|
IfDocScableDownstream = 128,
|
|
|
|
IfDocScableUpstream = 129,
|
|
|
|
IfA12MppSwitch = 130,
|
|
|
|
IfTunnel = 131,
|
|
|
|
IfCoffee = 132,
|
|
|
|
IfCes = 133,
|
|
|
|
IfAtmSubInterface = 134,
|
|
|
|
IfL2Vlan = 135,
|
|
|
|
IfL3IpVlan = 136,
|
|
|
|
IfL3IpxVlan = 137,
|
|
|
|
IfDigitalPowerline = 138,
|
|
|
|
IfMediaMailOverIp = 139,
|
|
|
|
IfDtm = 140,
|
|
|
|
IfDcn = 141,
|
|
|
|
IfIpForward = 142,
|
|
|
|
IfMsdsl = 143,
|
|
|
|
IfIeee1394 = 144,
|
|
|
|
IfIfGsn = 145,
|
|
|
|
IfDvbrccMaclayer = 146,
|
|
|
|
IfDvbrccDownstream = 147,
|
|
|
|
IfDvbrccUpstream = 148,
|
|
|
|
IfAtmVirtual = 149,
|
|
|
|
IfMplsTunnel = 150,
|
|
|
|
IfSrp = 151,
|
|
|
|
IfVoiceOverAtm = 152,
|
|
|
|
IfVoiceOverFrameRelay = 153,
|
|
|
|
IfIdsl = 154,
|
|
|
|
IfCompositeLink = 155,
|
|
|
|
IfSs7SigLink = 156,
|
|
|
|
IfPropWirelessP2p = 157,
|
|
|
|
IfFrForward = 158,
|
|
|
|
IfRfc1483 = 159,
|
|
|
|
IfUsb = 160,
|
|
|
|
IfIeee8023AdLag = 161,
|
|
|
|
IfBgpPolicyAccounting = 162,
|
|
|
|
IfFrf16MfrBundle = 163,
|
|
|
|
IfH323Gatekeeper = 164,
|
|
|
|
IfH323Proxy = 165,
|
|
|
|
IfMpls = 166,
|
|
|
|
IfMfSigLink = 167,
|
|
|
|
IfHdsl2 = 168,
|
|
|
|
IfShdsl = 169,
|
|
|
|
IfDs1Fdl = 170,
|
|
|
|
IfPos = 171,
|
|
|
|
IfDvbAsiIn = 172,
|
|
|
|
IfDvbAsiOut = 173,
|
|
|
|
IfPlc = 174,
|
|
|
|
IfNfas = 175,
|
|
|
|
IfTr008 = 176,
|
|
|
|
IfGr303Rdt = 177,
|
|
|
|
IfGr303Idt = 178,
|
|
|
|
IfIsup = 179,
|
|
|
|
IfPropDocsWirelessMaclayer = 180,
|
|
|
|
IfPropDocsWirelessDownstream = 181,
|
|
|
|
IfPropDocsWirelessUpstream = 182,
|
|
|
|
IfHiperLan2 = 183,
|
|
|
|
IfPropBwaP2mp = 184,
|
|
|
|
IfSonetOverheadChannel = 185,
|
|
|
|
IfDigitalWrapperOverheadChannel = 186,
|
|
|
|
IfAal2 = 187,
|
|
|
|
IfRadioMac = 188,
|
|
|
|
IfAtmRadio = 189,
|
|
|
|
IfImt = 190,
|
|
|
|
IfMvl = 191,
|
|
|
|
IfReachDsl = 192,
|
|
|
|
IfFrDlciEndpt = 193,
|
|
|
|
IfAtmVciEndpt = 194,
|
|
|
|
IfOpticalChannel = 195,
|
|
|
|
IfOpticalTransport = 196,
|
|
|
|
IfIeee80216Wman = 237,
|
|
|
|
IfWwanPp = 243,
|
|
|
|
IfWwanPp2 = 244,
|
|
|
|
IfIeee802154 = 259,
|
|
|
|
IfXboxWireless = 281
|
|
|
|
|
|
|
|
InterfaceState* = enum
|
|
|
|
StatusError = 0, # This is workaround element for ProoveInit warnings.
|
|
|
|
StatusUp,
|
|
|
|
StatusDown,
|
|
|
|
StatusTesting,
|
|
|
|
StatusUnknown,
|
|
|
|
StatusDormant,
|
|
|
|
StatusNotPresent,
|
|
|
|
StatusLowerLayerDown
|
|
|
|
|
|
|
|
InterfaceAddress* = object
|
|
|
|
host*: TransportAddress
|
|
|
|
net*: IpNet
|
|
|
|
|
|
|
|
NetworkInterface* = object
|
|
|
|
ifIndex*: int
|
|
|
|
ifType*: InterfaceType
|
|
|
|
name*: string
|
|
|
|
desc*: string
|
2022-11-02 09:09:15 +02:00
|
|
|
mtu*: int64
|
2019-04-15 04:27:12 +03:00
|
|
|
flags*: uint64
|
|
|
|
state*: InterfaceState
|
|
|
|
mac*: array[MaxAdapterAddressLength, byte]
|
|
|
|
maclen*: int
|
|
|
|
addresses*: seq[InterfaceAddress]
|
|
|
|
|
|
|
|
Route* = object
|
|
|
|
ifIndex*: int
|
|
|
|
dest*: TransportAddress
|
|
|
|
source*: TransportAddress
|
|
|
|
gateway*: TransportAddress
|
|
|
|
metric*: int
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc broadcast*(ifa: InterfaceAddress): TransportAddress =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return broadcast address for ``ifa``.
|
2021-02-25 23:04:56 +02:00
|
|
|
ifa.net.broadcast()
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc network*(ifa: InterfaceAddress): TransportAddress =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return network address for ``ifa``.
|
2021-02-25 23:04:56 +02:00
|
|
|
ifa.net.network()
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc netmask*(ifa: InterfaceAddress): TransportAddress =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return network mask for ``ifa``.
|
2021-02-25 23:04:56 +02:00
|
|
|
ifa.net.subnetMask()
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc init*(ift: typedesc[InterfaceAddress], address: TransportAddress,
|
|
|
|
prefix: int): InterfaceAddress =
|
|
|
|
## Initialize ``InterfaceAddress`` using ``address`` and prefix length
|
|
|
|
## ``prefix``.
|
2021-02-25 23:04:56 +02:00
|
|
|
InterfaceAddress(host: address, net: IpNet.init(address, prefix))
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc `$`*(ifa: InterfaceAddress): string =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return string representation of ``ifa``.
|
|
|
|
if ifa.host.family == AddressFamily.IPv4:
|
2021-02-25 23:04:56 +02:00
|
|
|
$ifa.net
|
2019-04-15 04:27:12 +03:00
|
|
|
elif ifa.host.family == AddressFamily.IPv6:
|
2021-02-25 23:04:56 +02:00
|
|
|
$ifa.net
|
|
|
|
else:
|
|
|
|
"Unknown"
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2023-05-01 00:46:15 +03:00
|
|
|
proc hexDigit(x: uint8, lowercase: bool = false): char =
|
|
|
|
char(0x30'u8 + x + (uint32(7) and not((uint32(x) - 10) shr 8)))
|
|
|
|
|
2019-04-15 04:27:12 +03:00
|
|
|
proc `$`*(iface: NetworkInterface): string =
|
|
|
|
## Return string representation of network interface ``iface``.
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = $iface.ifIndex
|
|
|
|
if len(res) == 1:
|
|
|
|
res.add(". ")
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add(". ")
|
|
|
|
res.add(iface.name)
|
2019-04-15 04:27:12 +03:00
|
|
|
when defined(windows):
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add(" [")
|
|
|
|
res.add(iface.desc)
|
|
|
|
res.add("]")
|
|
|
|
res.add(": flags = ")
|
|
|
|
res.add($iface.flags)
|
|
|
|
res.add(" mtu ")
|
|
|
|
res.add($iface.mtu)
|
|
|
|
res.add(" state ")
|
|
|
|
res.add($iface.state)
|
|
|
|
res.add("\n ")
|
|
|
|
res.add($iface.ifType)
|
|
|
|
res.add(" ")
|
2019-04-15 04:27:12 +03:00
|
|
|
if iface.maclen > 0:
|
2022-11-02 09:09:15 +02:00
|
|
|
for i in 0 ..< iface.maclen:
|
2023-05-01 00:46:15 +03:00
|
|
|
res.add(hexDigit(iface.mac[i] shr 4))
|
|
|
|
res.add(hexDigit(iface.mac[i] and 15))
|
2019-04-15 04:27:12 +03:00
|
|
|
if i < iface.maclen - 1:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add(":")
|
2019-04-15 04:27:12 +03:00
|
|
|
for item in iface.addresses:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add("\n ")
|
2019-04-15 04:27:12 +03:00
|
|
|
if item.host.family == AddressFamily.IPv4:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add("inet ")
|
2019-04-15 04:27:12 +03:00
|
|
|
elif item.host.family == AddressFamily.IPv6:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add("inet6 ")
|
|
|
|
res.add($item)
|
|
|
|
res.add(" netmask ")
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
res.add(try: $(item.netmask().address()) except ValueError as exc: exc.msg)
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add(" brd ")
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
res.add(
|
|
|
|
try: $(item.broadcast().address()) except ValueError as exc: exc.msg)
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc `$`*(route: Route): string =
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
var res = try: $route.dest.address() except ValueError as exc: exc.msg
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add(" via ")
|
2019-04-15 04:27:12 +03:00
|
|
|
if route.gateway.family != AddressFamily.None:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add("gateway ")
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
res.add(try: $route.gateway.address() except ValueError as exc: exc.msg)
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add("link")
|
|
|
|
res.add(" src ")
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
res.add(try: $route.source.address() except ValueError as exc: exc.msg)
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc cmp*(a, b: NetworkInterface): int =
|
2021-02-25 23:04:56 +02:00
|
|
|
cmp(a.ifIndex, b.ifIndex)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
when defined(linux):
|
2023-02-21 12:48:36 +02:00
|
|
|
import ".."/osutils
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
template NLMSG_ALIGN(length: uint): uint =
|
|
|
|
(length + NLMSG_ALIGNTO - 1) and not(NLMSG_ALIGNTO - 1)
|
|
|
|
|
|
|
|
template NLMSG_HDRLEN(): int =
|
|
|
|
int(NLMSG_ALIGN(uint(sizeof(NlMsgHeader))))
|
|
|
|
|
|
|
|
template NLMSG_LENGTH(length: int): uint32 =
|
|
|
|
uint32(NLMSG_HDRLEN() + length)
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc NLMSG_OK(nlh: ptr NlMsgHeader, length: int): bool =
|
|
|
|
(length >= int(sizeof(NlMsgHeader))) and
|
|
|
|
(nlh.nlmsg_len >= uint32(sizeof(NlMsgHeader))) and
|
|
|
|
(nlh.nlmsg_len <= uint32(length))
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc NLMSG_NEXT(nlh: ptr NlMsgHeader,
|
2021-02-25 23:04:56 +02:00
|
|
|
length: var int): ptr NlMsgHeader =
|
2019-04-15 04:27:12 +03:00
|
|
|
length = length - int(NLMSG_ALIGN(uint(nlh.nlmsg_len)))
|
2021-02-25 23:04:56 +02:00
|
|
|
cast[ptr NlMsgHeader](cast[uint](nlh) +
|
|
|
|
cast[uint](NLMSG_ALIGN(uint(nlh.nlmsg_len))))
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc NLMSG_DATA(nlh: ptr NlMsgHeader): ptr byte =
|
|
|
|
cast[ptr byte](cast[uint](nlh) + NLMSG_LENGTH(0))
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc NLMSG_TAIL(nlh: ptr NlMsgHeader): ptr byte {.inline.} =
|
|
|
|
cast[ptr byte](cast[uint](nlh) + NLMSG_ALIGN(uint(nlh.nlmsg_len)))
|
|
|
|
|
|
|
|
template RTA_ALIGN*(length: uint): uint =
|
|
|
|
(length + RTA_ALIGNTO - 1) and not(RTA_ALIGNTO - 1)
|
|
|
|
|
|
|
|
template RTA_HDRLEN(): int =
|
|
|
|
int(RTA_ALIGN(uint(sizeof(RtAttr))))
|
|
|
|
|
|
|
|
template RTA_LENGTH(length: int): uint =
|
|
|
|
uint(RTA_HDRLEN()) + uint(length)
|
|
|
|
|
|
|
|
template RTA_PAYLOAD*(length: uint): uint =
|
|
|
|
length - RTA_LENGTH(0)
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc IFLA_RTA(r: ptr byte): ptr RtAttr =
|
2019-04-15 04:27:12 +03:00
|
|
|
cast[ptr RtAttr](cast[uint](r) + NLMSG_ALIGN(uint(sizeof(IfInfoMessage))))
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc IFA_RTA(r: ptr byte): ptr RtAttr =
|
2019-04-15 04:27:12 +03:00
|
|
|
cast[ptr RtAttr](cast[uint](r) + NLMSG_ALIGN(uint(sizeof(IfAddrMessage))))
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc RT_RTA(r: ptr byte): ptr RtAttr =
|
2019-04-15 04:27:12 +03:00
|
|
|
cast[ptr RtAttr](cast[uint](r) + NLMSG_ALIGN(uint(sizeof(RtMessage))))
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc RTA_OK(rta: ptr RtAttr, length: int): bool =
|
|
|
|
length >= sizeof(RtAttr) and
|
|
|
|
rta.rta_len >= cushort(sizeof(RtAttr)) and
|
|
|
|
rta.rta_len <= cushort(length)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc RTA_NEXT(rta: ptr RtAttr, length: var int): ptr RtAttr =
|
2019-04-15 04:27:12 +03:00
|
|
|
length = length - int(RTA_ALIGN(uint(rta.rta_len)))
|
2021-02-25 23:04:56 +02:00
|
|
|
cast[ptr RtAttr](cast[uint](rta) +
|
|
|
|
cast[uint](RTA_ALIGN(uint(rta.rta_len))))
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc RTA_DATA(rta: ptr RtAttr): ptr byte =
|
|
|
|
cast[ptr byte](cast[uint](rta) + RTA_LENGTH(0))
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc toInterfaceState(it: cint, flags: cuint): InterfaceState =
|
2019-04-15 04:27:12 +03:00
|
|
|
case it
|
|
|
|
of 1:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusNotPresent
|
2019-04-15 04:27:12 +03:00
|
|
|
of 2:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusDown
|
2019-04-15 04:27:12 +03:00
|
|
|
of 3:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusLowerLayerDown
|
2019-04-15 04:27:12 +03:00
|
|
|
of 4:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusTesting
|
2019-04-15 04:27:12 +03:00
|
|
|
of 5:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusDormant
|
2019-04-15 04:27:12 +03:00
|
|
|
of 6:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusUp
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusUnknown
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc toInterfaceType(ft: uint32): InterfaceType =
|
2019-04-15 04:27:12 +03:00
|
|
|
case ft
|
|
|
|
of ARPHRD_ETHER, ARPHRD_EETHER:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfEthernetCsmacd
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_LOOPBACK:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfSoftwareLoopback
|
2019-04-15 04:27:12 +03:00
|
|
|
of 787..799:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfFibreChannel
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_PPP:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfPpp
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_SLIP, ARPHRD_CSLIP, ARPHRD_SLIP6, ARPHRD_CSLIP6:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfSlip
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_IEEE1394:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfIeee1394
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_IEEE80211, ARPHRD_IEEE80211_PRISM, ARPHRD_IEEE80211_RADIOTAP:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfIeee80211
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_ATM:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfAtm
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_HDLC:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfHdlc
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_HIPPI:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfHippiInterface
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_ARCNET:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfArcNet
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_LAPB:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfLapB
|
2019-04-15 04:27:12 +03:00
|
|
|
of ARPHRD_FRAD:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfFrameRelay
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfOther
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc createNetlinkSocket(pid: Pid): SocketHandle =
|
2022-01-04 23:14:30 +01:00
|
|
|
var address: Sockaddr_nl
|
2019-04-15 04:27:12 +03:00
|
|
|
address.family = cushort(AF_NETLINK)
|
|
|
|
address.groups = 0
|
|
|
|
address.pid = cast[uint32](pid)
|
2023-02-21 12:48:36 +02:00
|
|
|
var res = osdefs.socket(AF_NETLINK, osdefs.SOCK_DGRAM, NETLINK_ROUTE)
|
2021-02-25 23:04:56 +02:00
|
|
|
if res != SocketHandle(-1):
|
2023-02-21 12:48:36 +02:00
|
|
|
if osdefs.bindSocket(res, cast[ptr SockAddr](addr address),
|
2022-01-04 23:14:30 +01:00
|
|
|
SockLen(sizeof(Sockaddr_nl))) != 0:
|
2023-02-21 12:48:36 +02:00
|
|
|
discard osdefs.close(res)
|
2021-02-25 23:04:56 +02:00
|
|
|
res = SocketHandle(-1)
|
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc sendLinkMessage(fd: SocketHandle, pid: Pid, seqno: uint32,
|
|
|
|
ntype: uint16, nflags: uint16): bool =
|
|
|
|
var
|
|
|
|
rmsg: Tmsghdr
|
|
|
|
iov: IOVec
|
|
|
|
req: NLReq
|
2022-01-04 23:14:30 +01:00
|
|
|
address: Sockaddr_nl
|
2020-03-26 16:46:31 +07:00
|
|
|
|
|
|
|
type TIovLen = type iov.iov_len
|
|
|
|
|
2019-04-15 04:27:12 +03:00
|
|
|
address.family = cushort(AF_NETLINK)
|
|
|
|
req.hdr.nlmsg_len = NLMSG_LENGTH(sizeof(RtGenMsg))
|
|
|
|
req.hdr.nlmsg_type = ntype
|
|
|
|
req.hdr.nlmsg_flags = nflags
|
|
|
|
req.hdr.nlmsg_seq = seqno
|
|
|
|
req.hdr.nlmsg_pid = cast[uint32](pid)
|
|
|
|
req.msg.rtgen_family = byte(AF_PACKET)
|
|
|
|
iov.iov_base = cast[pointer](addr req)
|
2022-11-02 09:09:15 +02:00
|
|
|
iov.iov_len = TIovLen(req.hdr.nlmsg_len)
|
2019-04-15 04:27:12 +03:00
|
|
|
rmsg.msg_iov = addr iov
|
|
|
|
rmsg.msg_iovlen = 1
|
|
|
|
rmsg.msg_name = cast[pointer](addr address)
|
2022-01-04 23:14:30 +01:00
|
|
|
rmsg.msg_namelen = SockLen(sizeof(Sockaddr_nl))
|
2023-02-21 12:48:36 +02:00
|
|
|
let res = osdefs.sendmsg(fd, addr rmsg, 0).TIovLen
|
2021-02-25 23:04:56 +02:00
|
|
|
(res == iov.iov_len)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc sendRouteMessage(fd: SocketHandle, pid: Pid, seqno: uint32,
|
|
|
|
ntype: uint16, nflags: uint16,
|
|
|
|
dest: TransportAddress): bool =
|
|
|
|
var
|
|
|
|
rmsg: Tmsghdr
|
|
|
|
iov: IOVec
|
2022-01-04 23:14:30 +01:00
|
|
|
address: Sockaddr_nl
|
2019-04-15 04:27:12 +03:00
|
|
|
buffer: array[64, byte]
|
|
|
|
|
2020-03-26 16:46:31 +07:00
|
|
|
type TIovLen = type iov.iov_len
|
|
|
|
|
2021-12-08 16:58:24 +02:00
|
|
|
var req = cast[ptr NLRouteReq](addr buffer[0])
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
address.family = cushort(AF_NETLINK)
|
|
|
|
req.hdr.nlmsg_len = NLMSG_LENGTH(sizeof(RtMessage))
|
|
|
|
req.hdr.nlmsg_type = ntype
|
|
|
|
req.hdr.nlmsg_flags = nflags
|
|
|
|
req.hdr.nlmsg_seq = seqno
|
|
|
|
req.hdr.nlmsg_pid = cast[uint32](pid)
|
|
|
|
|
|
|
|
var attr = cast[ptr RtAttr](NLMSG_TAIL(addr req.hdr))
|
|
|
|
|
|
|
|
req.msg.rtm_flags = RTM_F_LOOKUP_TABLE
|
|
|
|
attr.rta_type = RTA_DST
|
|
|
|
if dest.family == AddressFamily.IPv4:
|
2023-02-21 12:48:36 +02:00
|
|
|
req.msg.rtm_family = byte(osdefs.AF_INET)
|
2019-04-15 04:27:12 +03:00
|
|
|
attr.rta_len = cast[cushort](RTA_LENGTH(4))
|
|
|
|
copyMem(RTA_DATA(attr), cast[ptr byte](unsafeAddr dest.address_v4[0]), 4)
|
|
|
|
req.hdr.nlmsg_len = uint32(NLMSG_ALIGN(uint(req.hdr.nlmsg_len)) +
|
|
|
|
RTA_ALIGN(uint(attr.rta_len)))
|
|
|
|
req.msg.rtm_dst_len = 4 * 8
|
|
|
|
elif dest.family == AddressFamily.IPv6:
|
2023-02-21 12:48:36 +02:00
|
|
|
req.msg.rtm_family = byte(osdefs.AF_INET6)
|
2019-04-15 04:27:12 +03:00
|
|
|
attr.rta_len = cast[cushort](RTA_LENGTH(16))
|
|
|
|
copyMem(RTA_DATA(attr), cast[ptr byte](unsafeAddr dest.address_v6[0]), 16)
|
|
|
|
req.hdr.nlmsg_len = uint32(NLMSG_ALIGN(uint(req.hdr.nlmsg_len)) +
|
|
|
|
RTA_ALIGN(uint(attr.rta_len)))
|
|
|
|
req.msg.rtm_dst_len = 16 * 8
|
|
|
|
|
|
|
|
iov.iov_base = cast[pointer](addr buffer[0])
|
2022-11-02 09:09:15 +02:00
|
|
|
iov.iov_len = TIovLen(req.hdr.nlmsg_len)
|
2019-04-15 04:27:12 +03:00
|
|
|
rmsg.msg_iov = addr iov
|
|
|
|
rmsg.msg_iovlen = 1
|
|
|
|
rmsg.msg_name = cast[pointer](addr address)
|
2022-01-04 23:14:30 +01:00
|
|
|
rmsg.msg_namelen = SockLen(sizeof(Sockaddr_nl))
|
2023-02-21 12:48:36 +02:00
|
|
|
let res = osdefs.sendmsg(fd, addr rmsg, 0).TIovLen
|
2021-02-25 23:04:56 +02:00
|
|
|
(res == iov.iov_len)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc readNetlinkMessage(fd: SocketHandle, data: var seq[byte]): bool =
|
|
|
|
var
|
|
|
|
rmsg: Tmsghdr
|
|
|
|
iov: IOVec
|
2022-01-04 23:14:30 +01:00
|
|
|
address: Sockaddr_nl
|
2019-04-15 04:27:12 +03:00
|
|
|
data.setLen(IFLIST_REPLY_BUFFER)
|
|
|
|
iov.iov_base = cast[pointer](addr data[0])
|
|
|
|
iov.iov_len = IFLIST_REPLY_BUFFER
|
|
|
|
rmsg.msg_iov = addr iov
|
|
|
|
rmsg.msg_iovlen = 1
|
|
|
|
rmsg.msg_name = cast[pointer](addr address)
|
2022-01-04 23:14:30 +01:00
|
|
|
rmsg.msg_namelen = SockLen(sizeof(Sockaddr_nl))
|
2023-02-21 12:48:36 +02:00
|
|
|
var length = osdefs.recvmsg(fd, addr rmsg, 0)
|
2019-04-15 04:27:12 +03:00
|
|
|
if length >= 0:
|
|
|
|
data.setLen(length)
|
2021-02-25 23:04:56 +02:00
|
|
|
true
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
|
|
|
data.setLen(0)
|
2021-02-25 23:04:56 +02:00
|
|
|
false
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc processLink(msg: ptr NlMsgHeader): NetworkInterface =
|
|
|
|
var iface: ptr IfInfoMessage
|
|
|
|
var attr: ptr RtAttr
|
|
|
|
var length: int
|
|
|
|
|
|
|
|
iface = cast[ptr IfInfoMessage](NLMSG_DATA(msg))
|
|
|
|
length = int(msg.nlmsg_len) - int(NLMSG_LENGTH(sizeof(IfInfoMessage)))
|
|
|
|
|
|
|
|
attr = IFLA_RTA(cast[ptr byte](iface))
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = NetworkInterface(
|
|
|
|
ifType: toInterfaceType(iface.ifi_type),
|
|
|
|
ifIndex: iface.ifi_index,
|
2022-11-02 09:09:15 +02:00
|
|
|
flags: uint64(iface.ifi_flags)
|
2021-02-25 23:04:56 +02:00
|
|
|
)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
while RTA_OK(attr, length):
|
|
|
|
if attr.rta_type == IFLA_IFNAME:
|
|
|
|
var p = cast[cstring](RTA_DATA(attr))
|
2021-02-25 23:04:56 +02:00
|
|
|
res.name = $p
|
2019-04-15 04:27:12 +03:00
|
|
|
elif attr.rta_type == IFLA_ADDRESS:
|
|
|
|
var p = cast[ptr byte](RTA_DATA(attr))
|
2021-02-25 23:04:56 +02:00
|
|
|
var plen = min(int(RTA_PAYLOAD(uint(attr.rta_len))), len(res.mac))
|
|
|
|
copyMem(addr res.mac[0], p, plen)
|
|
|
|
res.maclen = plen
|
2019-04-15 04:27:12 +03:00
|
|
|
elif attr.rta_type == IFLA_MTU:
|
|
|
|
var p = cast[ptr uint32](RTA_DATA(attr))
|
2022-11-02 09:09:15 +02:00
|
|
|
res.mtu = int64(p[])
|
2019-04-15 04:27:12 +03:00
|
|
|
elif attr.rta_type == IFLA_OPERSTATE:
|
|
|
|
var p = cast[ptr byte](RTA_DATA(attr))
|
2022-11-02 09:09:15 +02:00
|
|
|
res.state = toInterfaceState(cint(p[]), iface.ifi_flags)
|
2019-04-15 04:27:12 +03:00
|
|
|
attr = RTA_NEXT(attr, length)
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc getAddress(f: int, p: pointer): TransportAddress =
|
2023-02-21 12:48:36 +02:00
|
|
|
if f == osdefs.AF_INET:
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = TransportAddress(family: AddressFamily.IPv4)
|
|
|
|
copyMem(addr res.address_v4[0], p, len(res.address_v4))
|
|
|
|
res
|
2023-02-21 12:48:36 +02:00
|
|
|
elif f == osdefs.AF_INET6:
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = TransportAddress(family: AddressFamily.IPv6)
|
|
|
|
copyMem(addr res.address_v6[0], p, len(res.address_v6))
|
|
|
|
res
|
|
|
|
else:
|
|
|
|
TransportAddress(family: AddressFamily.None)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc processAddress(msg: ptr NlMsgHeader): NetworkInterface =
|
|
|
|
var iaddr: ptr IfAddrMessage
|
|
|
|
var attr: ptr RtAttr
|
|
|
|
var length: int
|
|
|
|
|
|
|
|
iaddr = cast[ptr IfAddrMessage](NLMSG_DATA(msg))
|
|
|
|
length = int(msg.nlmsg_len) - int(NLMSG_LENGTH(sizeof(IfAddrMessage)))
|
|
|
|
|
|
|
|
attr = IFA_RTA(cast[ptr byte](iaddr))
|
|
|
|
|
2022-11-02 09:09:15 +02:00
|
|
|
let family = int(iaddr.ifa_family)
|
|
|
|
var res = NetworkInterface(ifIndex: int(iaddr.ifa_index))
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
var address, local: TransportAddress
|
|
|
|
|
|
|
|
while RTA_OK(attr, length):
|
|
|
|
if attr.rta_type == IFA_LOCAL:
|
|
|
|
local = getAddress(family, cast[pointer](RTA_DATA(attr)))
|
|
|
|
elif attr.rta_type == IFA_ADDRESS:
|
|
|
|
address = getAddress(family, cast[pointer](RTA_DATA(attr)))
|
|
|
|
attr = RTA_NEXT(attr, length)
|
|
|
|
|
|
|
|
if local.family != AddressFamily.None:
|
|
|
|
address = local
|
2021-02-25 23:04:56 +02:00
|
|
|
|
2022-11-02 09:09:15 +02:00
|
|
|
let prefixLength = int(iaddr.ifa_prefixlen)
|
2019-04-15 04:27:12 +03:00
|
|
|
let ifaddr = InterfaceAddress.init(address, prefixLength)
|
2021-02-25 23:04:56 +02:00
|
|
|
res.addresses.add(ifaddr)
|
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc processRoute(msg: ptr NlMsgHeader): Route =
|
|
|
|
var rtmsg = cast[ptr RtMessage](NLMSG_DATA(msg))
|
|
|
|
var length = int(msg.nlmsg_len) - int(NLMSG_LENGTH(sizeof(RtMessage)))
|
|
|
|
var attr = RT_RTA(cast[ptr byte](rtmsg))
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = Route()
|
2019-04-15 04:27:12 +03:00
|
|
|
while RTA_OK(attr, length):
|
|
|
|
if attr.rta_type == RTA_DST:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.dest = getAddress(int(rtmsg.rtm_family),
|
2019-04-15 04:27:12 +03:00
|
|
|
cast[pointer](RTA_DATA(attr)))
|
|
|
|
elif attr.rta_type == RTA_GATEWAY:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.gateway = getAddress(int(rtmsg.rtm_family),
|
2019-04-15 04:27:12 +03:00
|
|
|
cast[pointer](RTA_DATA(attr)))
|
|
|
|
elif attr.rta_type == RTA_OIF:
|
|
|
|
var oid: uint32
|
|
|
|
copyMem(addr oid, RTA_DATA(attr), sizeof(uint32))
|
2021-02-25 23:04:56 +02:00
|
|
|
res.ifIndex = int(oid)
|
2019-04-15 04:27:12 +03:00
|
|
|
elif attr.rta_type == RTA_PREFSRC:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.source = getAddress(int(rtmsg.rtm_family),
|
2019-04-15 04:27:12 +03:00
|
|
|
cast[pointer](RTA_DATA(attr)))
|
|
|
|
attr = RTA_NEXT(attr, length)
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc getRoute(netfd: SocketHandle, pid: Pid,
|
|
|
|
address: TransportAddress): Route =
|
|
|
|
if not sendRouteMessage(netfd, pid, 1, RTM_GETROUTE,
|
|
|
|
NLM_F_REQUEST, address):
|
2021-02-25 23:04:56 +02:00
|
|
|
return Route()
|
2019-04-15 04:27:12 +03:00
|
|
|
var data = newSeq[byte]()
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = Route()
|
2019-04-15 04:27:12 +03:00
|
|
|
while true:
|
|
|
|
if not readNetlinkMessage(netfd, data):
|
|
|
|
break
|
|
|
|
var length = len(data)
|
|
|
|
var msg = cast[ptr NlMsgHeader](addr data[0])
|
|
|
|
var endflag = false
|
|
|
|
while NLMSG_OK(msg, length):
|
|
|
|
if msg.nlmsg_type == NLMSG_ERROR:
|
|
|
|
endflag = true
|
|
|
|
break
|
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
res = processRoute(msg)
|
2019-04-15 04:27:12 +03:00
|
|
|
endflag = true
|
|
|
|
break
|
|
|
|
msg = NLMSG_NEXT(msg, length)
|
|
|
|
if endflag:
|
|
|
|
break
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc getLinks(netfd: SocketHandle, pid: Pid): seq[NetworkInterface] =
|
2021-02-25 23:04:56 +02:00
|
|
|
var res: seq[NetworkInterface]
|
2019-04-15 04:27:12 +03:00
|
|
|
if not sendLinkMessage(netfd, pid, 1, RTM_GETLINK,
|
|
|
|
NLM_F_REQUEST or NLM_F_DUMP):
|
2021-02-25 23:04:56 +02:00
|
|
|
return res
|
2019-04-15 04:27:12 +03:00
|
|
|
var data = newSeq[byte]()
|
2021-02-25 23:04:56 +02:00
|
|
|
res = newSeq[NetworkInterface]()
|
2019-04-15 04:27:12 +03:00
|
|
|
while true:
|
|
|
|
if not readNetlinkMessage(netfd, data):
|
|
|
|
break
|
|
|
|
var length = len(data)
|
|
|
|
if length == 0:
|
|
|
|
break
|
|
|
|
var msg = cast[ptr NlMsgHeader](addr data[0])
|
|
|
|
var endflag = false
|
|
|
|
while NLMSG_OK(msg, length):
|
|
|
|
if msg.nlmsg_type == NLMSG_DONE:
|
|
|
|
endflag = true
|
|
|
|
break
|
|
|
|
elif msg.nlmsg_type == NLMSG_ERROR:
|
|
|
|
endflag = true
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
var iface = processLink(msg)
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add(iface)
|
2019-04-15 04:27:12 +03:00
|
|
|
msg = NLMSG_NEXT(msg, length)
|
|
|
|
if endflag:
|
|
|
|
break
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc getAddresses(netfd: SocketHandle, pid: Pid,
|
|
|
|
ifaces: var seq[NetworkInterface]) =
|
|
|
|
if not sendLinkMessage(netfd, pid, 2, RTM_GETADDR,
|
|
|
|
NLM_F_REQUEST or NLM_F_DUMP):
|
|
|
|
return
|
|
|
|
var data = newSeq[byte]()
|
|
|
|
while true:
|
|
|
|
if not readNetlinkMessage(netfd, data):
|
|
|
|
break
|
|
|
|
var length = len(data)
|
|
|
|
if length == 0:
|
|
|
|
break
|
|
|
|
var msg = cast[ptr NlMsgHeader](addr data[0])
|
|
|
|
var endflag = false
|
|
|
|
while NLMSG_OK(msg, length):
|
|
|
|
if msg.nlmsg_type == NLMSG_DONE:
|
|
|
|
endflag = true
|
|
|
|
break
|
|
|
|
elif msg.nlmsg_type == NLMSG_ERROR:
|
|
|
|
endflag = true
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
var iface = processAddress(msg)
|
|
|
|
for i in 0..<len(ifaces):
|
|
|
|
if ifaces[i].ifIndex == iface.ifIndex:
|
|
|
|
for item in iface.addresses:
|
|
|
|
ifaces[i].addresses.add(item)
|
|
|
|
msg = NLMSG_NEXT(msg, length)
|
|
|
|
if endflag:
|
|
|
|
break
|
|
|
|
|
2023-06-05 22:21:50 +02:00
|
|
|
proc getInterfaces*(): seq[NetworkInterface] {.raises: [].} =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return list of available interfaces.
|
2021-02-25 23:04:56 +02:00
|
|
|
var res: seq[NetworkInterface]
|
2023-02-21 12:48:36 +02:00
|
|
|
var pid = osdefs.getpid()
|
2019-04-15 04:27:12 +03:00
|
|
|
var sock = createNetlinkSocket(pid)
|
|
|
|
if sock == InvalidSocketHandle:
|
2021-02-25 23:04:56 +02:00
|
|
|
return res
|
|
|
|
else:
|
|
|
|
res = getLinks(sock, pid)
|
|
|
|
getAddresses(sock, pid, res)
|
|
|
|
sort(res, cmp)
|
2023-02-21 12:48:36 +02:00
|
|
|
discard osdefs.close(sock)
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2023-06-05 22:21:50 +02:00
|
|
|
proc getBestRoute*(address: TransportAddress): Route {.raises: [].} =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return best applicable OS route, which will be used for connecting to
|
|
|
|
## address ``address``.
|
2023-02-21 12:48:36 +02:00
|
|
|
var pid = osdefs.getpid()
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = Route()
|
2019-04-15 04:27:12 +03:00
|
|
|
var sock = createNetlinkSocket(pid)
|
|
|
|
if sock == InvalidSocketHandle:
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
|
|
|
else:
|
|
|
|
res = getRoute(sock, pid, address)
|
2023-02-21 12:48:36 +02:00
|
|
|
discard osdefs.close(sock)
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2023-02-21 12:48:36 +02:00
|
|
|
elif defined(macosx) or defined(macos) or defined(bsd):
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc toInterfaceType(f: byte): InterfaceType =
|
2022-11-02 09:09:15 +02:00
|
|
|
var ft = int(f)
|
2019-04-15 04:27:12 +03:00
|
|
|
if (ft >= 1 and ft <= 196) or (ft == 237) or (ft == 243) or (ft == 244):
|
2021-02-25 23:04:56 +02:00
|
|
|
cast[InterfaceType](ft)
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfOther
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
proc toInterfaceState(f: cuint): InterfaceState =
|
2019-04-15 04:27:12 +03:00
|
|
|
if (f and IFF_RUNNING) != 0 and (f and IFF_UP) != 0:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusUp
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusDown
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2023-06-05 22:21:50 +02:00
|
|
|
proc getInterfaces*(): seq[NetworkInterface] {.raises: [].} =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return list of available interfaces.
|
2021-02-25 23:04:56 +02:00
|
|
|
var res: seq[NetworkInterface]
|
2019-04-15 04:27:12 +03:00
|
|
|
var ifap: ptr IfAddrs
|
2021-02-25 23:04:56 +02:00
|
|
|
let gres = getIfAddrs(addr ifap)
|
|
|
|
if gres == 0:
|
2019-04-15 04:27:12 +03:00
|
|
|
while not isNil(ifap):
|
|
|
|
var iface: NetworkInterface
|
|
|
|
var ifaddress: InterfaceAddress
|
|
|
|
|
2023-02-21 12:48:36 +02:00
|
|
|
iface.name = $cast[cstring](ifap.ifa_name)
|
2022-11-02 09:09:15 +02:00
|
|
|
iface.flags = uint64(ifap.ifa_flags)
|
2019-04-15 04:27:12 +03:00
|
|
|
var i = 0
|
2021-02-25 23:04:56 +02:00
|
|
|
while i < len(res):
|
|
|
|
if res[i].name == iface.name:
|
2019-04-15 04:27:12 +03:00
|
|
|
break
|
|
|
|
inc(i)
|
2021-02-25 23:04:56 +02:00
|
|
|
if i == len(res):
|
|
|
|
res.add(iface)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
if not isNil(ifap.ifa_addr):
|
2022-11-02 09:09:15 +02:00
|
|
|
let family = int(ifap.ifa_addr.sa_family)
|
2019-04-15 04:27:12 +03:00
|
|
|
if family == AF_LINK:
|
|
|
|
var data = cast[ptr IfData](ifap.ifa_data)
|
2022-01-04 23:14:30 +01:00
|
|
|
var link = cast[ptr Sockaddr_dl](ifap.ifa_addr)
|
2022-11-02 09:09:15 +02:00
|
|
|
res[i].ifIndex = int(link.sdl_index)
|
|
|
|
let nlen = int(link.sdl_nlen)
|
2019-04-15 04:27:12 +03:00
|
|
|
if nlen < len(link.sdl_data):
|
2023-05-01 16:02:59 +03:00
|
|
|
let minsize = min(int(link.sdl_alen), len(res[i].mac))
|
2021-02-25 23:04:56 +02:00
|
|
|
copyMem(addr res[i].mac[0], addr link.sdl_data[nlen], minsize)
|
2022-11-02 09:09:15 +02:00
|
|
|
res[i].maclen = int(link.sdl_alen)
|
2021-02-25 23:04:56 +02:00
|
|
|
res[i].ifType = toInterfaceType(data.ifi_type)
|
|
|
|
res[i].state = toInterfaceState(ifap.ifa_flags)
|
2023-05-01 16:02:59 +03:00
|
|
|
res[i].mtu = int64(data.ifi_mtu)
|
2023-02-21 12:48:36 +02:00
|
|
|
elif family == osdefs.AF_INET:
|
2019-04-15 04:27:12 +03:00
|
|
|
fromSAddr(cast[ptr Sockaddr_storage](ifap.ifa_addr),
|
2022-01-04 23:14:30 +01:00
|
|
|
SockLen(sizeof(Sockaddr_in)), ifaddress.host)
|
2023-02-21 12:48:36 +02:00
|
|
|
elif family == osdefs.AF_INET6:
|
2019-04-15 04:27:12 +03:00
|
|
|
fromSAddr(cast[ptr Sockaddr_storage](ifap.ifa_addr),
|
2022-01-04 23:14:30 +01:00
|
|
|
SockLen(sizeof(Sockaddr_in6)), ifaddress.host)
|
2019-04-15 04:27:12 +03:00
|
|
|
if not isNil(ifap.ifa_netmask):
|
|
|
|
var na: TransportAddress
|
2023-02-23 09:58:58 +02:00
|
|
|
let family = int(ifap.ifa_netmask.sa_family)
|
2023-02-21 12:48:36 +02:00
|
|
|
if family == osdefs.AF_INET:
|
2019-04-15 04:27:12 +03:00
|
|
|
fromSAddr(cast[ptr Sockaddr_storage](ifap.ifa_netmask),
|
2022-01-04 23:14:30 +01:00
|
|
|
SockLen(sizeof(Sockaddr_in)), na)
|
2023-04-30 09:20:08 +03:00
|
|
|
if ifaddress.host.family == AddressFamily.IPv4:
|
2023-02-23 09:58:58 +02:00
|
|
|
ifaddress.net = IpNet.init(ifaddress.host, na)
|
2023-02-21 12:48:36 +02:00
|
|
|
elif family == osdefs.AF_INET6:
|
2019-04-15 04:27:12 +03:00
|
|
|
fromSAddr(cast[ptr Sockaddr_storage](ifap.ifa_netmask),
|
2022-01-04 23:14:30 +01:00
|
|
|
SockLen(sizeof(Sockaddr_in6)), na)
|
2023-04-30 09:20:08 +03:00
|
|
|
if ifaddress.host.family == AddressFamily.IPv6:
|
2023-02-23 09:58:58 +02:00
|
|
|
ifaddress.net = IpNet.init(ifaddress.host, na)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
if ifaddress.host.family != AddressFamily.None:
|
2021-02-25 23:04:56 +02:00
|
|
|
res[i].addresses.add(ifaddress)
|
2019-04-15 04:27:12 +03:00
|
|
|
ifap = ifap.ifa_next
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
sort(res, cmp)
|
2019-04-15 04:27:12 +03:00
|
|
|
freeIfAddrs(ifap)
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2021-12-10 10:55:25 +01:00
|
|
|
proc sasize(data: openArray[byte]): int =
|
2019-04-15 04:27:12 +03:00
|
|
|
# SA_SIZE() template. Taken from FreeBSD net/route.h:1.63
|
|
|
|
if len(data) > 0:
|
|
|
|
if data[0] == 0x00'u8:
|
2021-02-25 23:04:56 +02:00
|
|
|
sizeof(uint32)
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
1 + (int(data[0] - 1) or (sizeof(uint32) - 1))
|
|
|
|
else:
|
|
|
|
0
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2023-06-05 22:21:50 +02:00
|
|
|
proc getBestRoute*(address: TransportAddress): Route {.raises: [].} =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return best applicable OS route, which will be used for connecting to
|
|
|
|
## address ``address``.
|
|
|
|
var sock: cint
|
|
|
|
var msg: RtMessage
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = Route()
|
2023-02-21 12:48:36 +02:00
|
|
|
var pid = osdefs.getpid()
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
if address.family notin {AddressFamily.IPv4, AddressFamily.IPv6}:
|
|
|
|
return
|
|
|
|
|
|
|
|
if address.family == AddressFamily.IPv4:
|
2023-02-21 12:48:36 +02:00
|
|
|
sock = cint(osdefs.socket(PF_ROUTE, osdefs.SOCK_RAW, osdefs.AF_INET))
|
2019-04-15 04:27:12 +03:00
|
|
|
elif address.family == AddressFamily.IPv6:
|
2023-02-21 12:48:36 +02:00
|
|
|
sock = cint(osdefs.socket(PF_ROUTE, osdefs.SOCK_RAW, osdefs.AF_INET6))
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
if sock != -1:
|
|
|
|
var sastore: Sockaddr_storage
|
2021-12-08 16:58:24 +02:00
|
|
|
var salen: SockLen
|
2019-04-15 04:27:12 +03:00
|
|
|
address.toSAddr(sastore, salen)
|
|
|
|
# We doing this trick because Nim's posix declaration of Sockaddr_storage
|
|
|
|
# is not compatible with BSD version. First byte in BSD version is length
|
|
|
|
# of Sockaddr structure, and second byte is family code.
|
|
|
|
copyMem(addr msg.space[0], addr sastore, int(salen))
|
|
|
|
msg.rtm.rtm_type = RTM_GET
|
|
|
|
msg.rtm.rtm_flags = RTF_UP or RTF_GATEWAY
|
|
|
|
msg.rtm.rtm_version = RTM_VERSION
|
|
|
|
msg.rtm.rtm_seq = 0xCAFE
|
|
|
|
msg.rtm.rtm_addrs = RTA_DST
|
|
|
|
msg.space[0] = cast[byte](salen)
|
|
|
|
msg.rtm.rtm_msglen = uint16(sizeof(RtMessage))
|
2023-02-21 12:48:36 +02:00
|
|
|
let wres = osdefs.write(sock, addr msg, sizeof(RtMessage))
|
2021-02-25 23:04:56 +02:00
|
|
|
if wres >= 0:
|
|
|
|
let rres =
|
|
|
|
block:
|
|
|
|
var pres = 0
|
|
|
|
while true:
|
2023-02-21 12:48:36 +02:00
|
|
|
pres = osdefs.read(sock, addr msg, sizeof(RtMessage))
|
2021-02-25 23:04:56 +02:00
|
|
|
if ((pres >= 0) and (msg.rtm.rtm_pid == pid) and
|
|
|
|
(msg.rtm.rtm_seq == 0xCAFE)) or (pres < 0):
|
|
|
|
break
|
|
|
|
pres
|
|
|
|
if (rres >= 0) and (msg.rtm.rtm_version == RTM_VERSION) and
|
|
|
|
(msg.rtm.rtm_errno == 0):
|
|
|
|
res.ifIndex = int(msg.rtm.rtm_index)
|
2019-04-15 04:27:12 +03:00
|
|
|
var so = 0
|
|
|
|
var eo = len(msg.space) - 1
|
|
|
|
for i in 0..<2:
|
|
|
|
let mask = 1 shl i
|
|
|
|
if (msg.rtm.rtm_addrs and mask) != 0:
|
|
|
|
var saddr = cast[ptr Sockaddr_storage](addr msg.space[so])
|
|
|
|
let size = sasize(msg.space.toOpenArray(so, eo))
|
|
|
|
if mask == RTA_DST:
|
2021-12-08 16:58:24 +02:00
|
|
|
fromSAddr(saddr, SockLen(size), res.dest)
|
2019-04-15 04:27:12 +03:00
|
|
|
elif mask == RTA_GATEWAY:
|
2021-12-08 16:58:24 +02:00
|
|
|
fromSAddr(saddr, SockLen(size), res.gateway)
|
2019-04-15 04:27:12 +03:00
|
|
|
so += size
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
if res.dest.isZero():
|
|
|
|
res.dest = address
|
2019-04-15 04:27:12 +03:00
|
|
|
var interfaces = getInterfaces()
|
|
|
|
for item in interfaces:
|
2021-02-25 23:04:56 +02:00
|
|
|
if res.ifIndex == item.ifIndex:
|
2019-04-15 04:27:12 +03:00
|
|
|
for a in item.addresses:
|
|
|
|
if a.host.family == address.family:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.source = a.host
|
2019-04-15 04:27:12 +03:00
|
|
|
break
|
2023-02-21 12:48:36 +02:00
|
|
|
discard osdefs.close(sock)
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
elif defined(windows):
|
2023-02-21 12:48:36 +02:00
|
|
|
import dynlib
|
|
|
|
import ".."/osutils
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
const
|
|
|
|
WorkBufferSize = 16384'u32
|
|
|
|
MaxTries = 3
|
|
|
|
|
|
|
|
proc toInterfaceType(ft: uint32): InterfaceType {.inline.} =
|
|
|
|
if (ft >= 1'u32 and ft <= 196'u32) or
|
|
|
|
(ft == 237) or (ft == 243) or (ft == 244) or (ft == 259) or (ft == 281):
|
2021-02-25 23:04:56 +02:00
|
|
|
cast[InterfaceType](ft)
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
IfOther
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc toInterfaceState(it: cint): InterfaceState {.inline.} =
|
|
|
|
if it >= 1 and it <= 7:
|
2021-02-25 23:04:56 +02:00
|
|
|
cast[InterfaceState](it)
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
StatusUnknown
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc `$`(bstr: ptr WCHAR): string =
|
2023-02-21 12:48:36 +02:00
|
|
|
let res = toString(bstr)
|
|
|
|
if res.isErr(): "" else: res.get()
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc isVista(): bool =
|
|
|
|
var ver: OSVERSIONINFO
|
|
|
|
ver.dwOSVersionInfoSize = DWORD(sizeof(ver))
|
2023-02-21 12:48:36 +02:00
|
|
|
let res = getVersionEx(addr(ver))
|
2019-04-15 04:27:12 +03:00
|
|
|
if res == 0:
|
2021-02-25 23:04:56 +02:00
|
|
|
false
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2021-02-25 23:04:56 +02:00
|
|
|
(ver.dwMajorVersion >= 6)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc toIPv6(a: TransportAddress): TransportAddress =
|
|
|
|
## IPv4-mapped addresses are formed by:
|
|
|
|
## <80 bits of zeros> + <16 bits of ones> + <32-bit IPv4 address>.
|
|
|
|
if a.family == AddressFamily.IPv4:
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = TransportAddress(family: AddressFamily.IPv6)
|
|
|
|
res.address_v6[10] = 0xFF'u8
|
|
|
|
res.address_v6[11] = 0xFF'u8
|
|
|
|
copyMem(addr res.address_v6[12], unsafeAddr a.address_v4[0], 4)
|
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
elif a.family == AddressFamily.IPv6:
|
2021-02-25 23:04:56 +02:00
|
|
|
a
|
|
|
|
else:
|
|
|
|
TransportAddress(family: AddressFamily.IPv6)
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc ipMatchPrefix(number, prefix: TransportAddress, nbits: int): bool =
|
|
|
|
var num6, prefix6: TransportAddress
|
|
|
|
if number.family == AddressFamily.IPv4:
|
|
|
|
num6 = toIPv6(number)
|
|
|
|
else:
|
|
|
|
num6 = number
|
|
|
|
if prefix.family == AddressFamily.IPv4:
|
|
|
|
prefix6 = toIPv6(number)
|
|
|
|
else:
|
|
|
|
prefix6 = prefix
|
|
|
|
var bytesCount = nbits div 8
|
|
|
|
var bitsCount = nbits mod 8
|
|
|
|
for i in 0..<bytesCount:
|
|
|
|
if num6.address_v6[i] != prefix6.address_v6[i]:
|
|
|
|
return false
|
|
|
|
if bitsCount != 0:
|
|
|
|
var mask = cast[byte](0xFF'u8 shl (8 - bitsCount))
|
|
|
|
let i = bytesCount
|
|
|
|
if (num6.address_v6[i] and mask) != (prefix6.address_v6[i] and mask):
|
|
|
|
return false
|
2021-02-25 23:04:56 +02:00
|
|
|
true
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
proc processAddress(ifitem: ptr IpAdapterAddressesXp,
|
|
|
|
ifunic: ptr IpAdapterUnicastAddressXpLh,
|
|
|
|
vista: bool): InterfaceAddress =
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = InterfaceAddress()
|
2019-04-15 04:27:12 +03:00
|
|
|
var netfamily = ifunic.address.lpSockaddr.sa_family
|
|
|
|
fromSAddr(cast[ptr Sockaddr_storage](ifunic.address.lpSockaddr),
|
2021-02-25 23:04:56 +02:00
|
|
|
SockLen(ifunic.address.iSockaddrLength), res.host)
|
2019-04-15 04:27:12 +03:00
|
|
|
if not vista:
|
|
|
|
var prefix = ifitem.firstPrefix
|
|
|
|
var prefixLength = -1
|
|
|
|
while not isNil(prefix):
|
|
|
|
var pa: TransportAddress
|
|
|
|
var prefamily = prefix.address.lpSockaddr.sa_family
|
|
|
|
fromSAddr(cast[ptr Sockaddr_storage](prefix.address.lpSockaddr),
|
|
|
|
SockLen(prefix.address.iSockaddrLength), pa)
|
|
|
|
if netfamily == prefamily:
|
2022-11-02 09:09:15 +02:00
|
|
|
if ipMatchPrefix(res.host, pa, int(prefix.prefixLength)):
|
|
|
|
prefixLength = max(prefixLength, int(prefix.prefixLength))
|
2019-04-15 04:27:12 +03:00
|
|
|
prefix = prefix.next
|
|
|
|
if prefixLength >= 0:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.net = IpNet.init(res.host, prefixLength)
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
2022-11-02 09:09:15 +02:00
|
|
|
let prefixLength = int(ifunic.onLinkPrefixLength)
|
2019-04-15 04:27:12 +03:00
|
|
|
if prefixLength >= 0:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.net = IpNet.init(res.host, prefixLength)
|
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2023-06-05 22:21:50 +02:00
|
|
|
proc getInterfaces*(): seq[NetworkInterface] {.raises: [].} =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return list of network interfaces.
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = newSeq[NetworkInterface]()
|
2019-04-15 04:27:12 +03:00
|
|
|
var size = WorkBufferSize
|
|
|
|
var tries = 0
|
|
|
|
var buffer: seq[byte]
|
2021-02-25 23:04:56 +02:00
|
|
|
var gres: uint32
|
2019-04-15 04:27:12 +03:00
|
|
|
var vista = isVista()
|
|
|
|
|
|
|
|
while true:
|
|
|
|
buffer = newSeq[byte](size)
|
|
|
|
var addresses = cast[ptr IpAdapterAddressesXp](addr buffer[0])
|
2023-02-21 12:48:36 +02:00
|
|
|
gres = getAdaptersAddresses(osdefs.AF_UNSPEC, GAA_FLAG_INCLUDE_PREFIX,
|
|
|
|
nil, addresses, addr size)
|
2023-04-30 09:20:08 +03:00
|
|
|
case OSErrorCode(gres)
|
|
|
|
of ERROR_SUCCESS:
|
2019-04-15 04:27:12 +03:00
|
|
|
buffer.setLen(size)
|
|
|
|
break
|
2023-04-30 09:20:08 +03:00
|
|
|
of ERROR_BUFFER_OVERFLOW:
|
2019-04-15 04:27:12 +03:00
|
|
|
discard
|
|
|
|
else:
|
|
|
|
break
|
|
|
|
inc(tries)
|
|
|
|
if tries >= MaxTries:
|
|
|
|
break
|
|
|
|
|
2023-04-30 09:20:08 +03:00
|
|
|
if OSErrorCode(gres) == ERROR_SUCCESS:
|
2019-04-15 04:27:12 +03:00
|
|
|
var slider = cast[ptr IpAdapterAddressesXp](addr buffer[0])
|
|
|
|
while not isNil(slider):
|
|
|
|
var iface = NetworkInterface(
|
2022-11-02 09:09:15 +02:00
|
|
|
ifIndex: int(slider.ifIndex),
|
2019-04-15 04:27:12 +03:00
|
|
|
ifType: toInterfaceType(slider.ifType),
|
|
|
|
state: toInterfaceState(slider.operStatus),
|
|
|
|
name: $slider.adapterName,
|
|
|
|
desc: $slider.description,
|
2022-11-02 09:09:15 +02:00
|
|
|
mtu: int(slider.mtu),
|
|
|
|
maclen: int(slider.physicalAddressLength),
|
|
|
|
flags: uint64(slider.flags)
|
2019-04-15 04:27:12 +03:00
|
|
|
)
|
|
|
|
copyMem(addr iface.mac[0], addr slider.physicalAddress[0],
|
|
|
|
len(iface.mac))
|
|
|
|
var unicast = slider.unicastAddress
|
|
|
|
while not isNil(unicast):
|
|
|
|
var ifaddr = processAddress(slider, unicast, vista)
|
|
|
|
iface.addresses.add(ifaddr)
|
|
|
|
unicast = unicast.next
|
2021-02-25 23:04:56 +02:00
|
|
|
res.add(iface)
|
2019-04-15 04:27:12 +03:00
|
|
|
slider = slider.next
|
|
|
|
|
2021-02-25 23:04:56 +02:00
|
|
|
sort(res, cmp)
|
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
2023-06-05 22:21:50 +02:00
|
|
|
proc getBestRoute*(address: TransportAddress): Route {.raises: [].} =
|
2019-04-15 04:27:12 +03:00
|
|
|
## Return best applicable OS route, which will be used for connecting to
|
|
|
|
## address ``address``.
|
2021-02-25 23:04:56 +02:00
|
|
|
var res = Route()
|
2019-04-15 04:27:12 +03:00
|
|
|
if isVista():
|
|
|
|
let iph = loadLib("iphlpapi.dll")
|
|
|
|
if iph != nil:
|
|
|
|
var bestRoute: MibIpForwardRow2
|
|
|
|
var empty: TransportAddress
|
|
|
|
var dest, src: Sockaddr_storage
|
|
|
|
var luid: uint64
|
2021-12-08 16:58:24 +02:00
|
|
|
var destlen: SockLen
|
2019-04-15 04:27:12 +03:00
|
|
|
address.toSAddr(dest, destlen)
|
|
|
|
var getBestRoute2 = cast[GETBESTROUTE2](symAddr(iph, "GetBestRoute2"))
|
2021-02-25 23:04:56 +02:00
|
|
|
var gres = getBestRoute2(addr luid, 0'u32, nil,
|
2019-04-15 04:27:12 +03:00
|
|
|
cast[ptr SOCKADDR_INET](addr dest),
|
|
|
|
0'u32,
|
|
|
|
addr bestRoute,
|
|
|
|
cast[ptr SOCKADDR_INET](addr src))
|
2021-02-25 23:04:56 +02:00
|
|
|
if gres == 0:
|
2023-02-21 12:48:36 +02:00
|
|
|
if src.ss_family == osdefs.AF_INET:
|
2021-12-08 16:58:24 +02:00
|
|
|
fromSAddr(addr src, SockLen(sizeof(Sockaddr_in)), res.source)
|
2023-02-21 12:48:36 +02:00
|
|
|
elif src.ss_family == osdefs.AF_INET6:
|
2021-12-08 16:58:24 +02:00
|
|
|
fromSAddr(addr src, SockLen(sizeof(Sockaddr_in6)), res.source)
|
2023-02-21 12:48:36 +02:00
|
|
|
if bestRoute.nextHop.si_family == osdefs.AF_INET:
|
2019-04-15 04:27:12 +03:00
|
|
|
fromSAddr(cast[ptr Sockaddr_storage](addr bestRoute.nextHop),
|
2021-12-08 16:58:24 +02:00
|
|
|
SockLen(sizeof(Sockaddr_in)), res.gateway)
|
2023-02-21 12:48:36 +02:00
|
|
|
elif bestRoute.nextHop.si_family == osdefs.AF_INET6:
|
2019-04-15 04:27:12 +03:00
|
|
|
fromSAddr(cast[ptr Sockaddr_storage](addr bestRoute.nextHop),
|
2021-12-08 16:58:24 +02:00
|
|
|
SockLen(sizeof(Sockaddr_in6)), res.gateway)
|
2021-02-25 23:04:56 +02:00
|
|
|
if res.gateway.isZero():
|
|
|
|
res.gateway = empty
|
|
|
|
res.dest = address
|
|
|
|
res.ifIndex = int(bestRoute.interfaceIndex)
|
|
|
|
res.metric = int(bestRoute.metric)
|
2019-04-15 04:27:12 +03:00
|
|
|
else:
|
|
|
|
if address.family == AddressFamily.IPv4:
|
|
|
|
var bestRoute: MibIpForwardRow
|
|
|
|
var dest: uint32
|
|
|
|
copyMem(addr dest, unsafeAddr address.address_v4[0], 4)
|
2021-02-25 23:04:56 +02:00
|
|
|
let gres = getBestRouteXp(dest, 0'u32, addr bestRoute)
|
|
|
|
if gres == 0:
|
2019-04-15 04:27:12 +03:00
|
|
|
var interfaces = getInterfaces()
|
2021-02-25 23:04:56 +02:00
|
|
|
res.dest = address
|
2019-04-15 04:27:12 +03:00
|
|
|
if bestRoute.dwForwardNextHop != 0'u32:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.gateway = TransportAddress(family: AddressFamily.IPv4)
|
|
|
|
copyMem(addr res.gateway.address_v4[0],
|
2019-04-15 04:27:12 +03:00
|
|
|
addr bestRoute.dwForwardNextHop, 4)
|
2021-02-25 23:04:56 +02:00
|
|
|
res.metric = int(bestRoute.dwForwardMetric1)
|
|
|
|
res.ifIndex = int(bestRoute.dwForwardIfIndex)
|
2019-04-15 04:27:12 +03:00
|
|
|
for item in interfaces:
|
|
|
|
if item.ifIndex == int(bestRoute.dwForwardIfIndex):
|
|
|
|
for a in item.addresses:
|
|
|
|
if a.host.family == AddressFamily.IPv4:
|
2021-02-25 23:04:56 +02:00
|
|
|
res.source = a.host
|
2019-04-15 04:27:12 +03:00
|
|
|
break
|
2021-02-25 23:04:56 +02:00
|
|
|
res
|
2019-04-15 04:27:12 +03:00
|
|
|
|
|
|
|
else:
|
|
|
|
{.fatal: "Sorry, your OS is currently not supported!".}
|