Skip to content

Commit 8b80757

Browse files
authored
don't include link-local addresses (169.254) in public_ips (#1040)
* don't include link-local (169.254) addresses in public_ips public_ips is meant to return addresses that might be connectable, which link-local are not * consolidate netifaces/psutil into _populate_from_dict where dict is like `{"en0": ["1.2.3.4"]}`
1 parent 3b1a9fe commit 8b80757

File tree

1 file changed

+33
-47
lines changed

1 file changed

+33
-47
lines changed

jupyter_client/localinterfaces.py

+33-47
Original file line numberDiff line numberDiff line change
@@ -8,11 +8,11 @@
88
import socket
99
import subprocess
1010
from subprocess import PIPE, Popen
11-
from typing import Any, Callable, Iterable, Sequence
11+
from typing import Any, Callable, Iterable, Mapping, Sequence
1212
from warnings import warn
1313

14-
LOCAL_IPS: list = []
15-
PUBLIC_IPS: list = []
14+
LOCAL_IPS: list[str] = []
15+
PUBLIC_IPS: list[str] = []
1616

1717
LOCALHOST: str = ""
1818

@@ -75,27 +75,34 @@ class NoIPAddresses(Exception): # noqa
7575
pass
7676

7777

78-
def _populate_from_list(addrs: Sequence[str] | None) -> None:
78+
def _populate_from_list(addrs: Sequence[str]) -> None:
7979
"""populate local and public IPs from flat list of all IPs"""
80+
_populate_from_dict({"all": addrs})
81+
82+
83+
def _populate_from_dict(addrs: Mapping[str, Sequence[str]]) -> None:
84+
"""populate local and public IPs from dict of {'en0': 'ip'}"""
8085
if not addrs:
81-
raise NoIPAddresses
86+
raise NoIPAddresses()
8287

8388
global LOCALHOST
8489
public_ips = []
8590
local_ips = []
8691

87-
for ip in addrs:
88-
local_ips.append(ip)
89-
if not ip.startswith("127."):
90-
public_ips.append(ip)
91-
elif not LOCALHOST:
92-
LOCALHOST = ip
92+
for iface, ip_list in addrs.items():
93+
for ip in ip_list:
94+
local_ips.append(ip)
95+
if not LOCALHOST and (iface.startswith("lo") or ip.startswith("127.")):
96+
LOCALHOST = ip
97+
if not iface.startswith("lo") and not ip.startswith(("127.", "169.254.")):
98+
# don't include link-local address in public_ips
99+
public_ips.append(ip)
93100

94101
if not LOCALHOST or LOCALHOST == "127.0.0.1":
95102
LOCALHOST = "127.0.0.1"
96103
local_ips.insert(0, LOCALHOST)
97104

98-
local_ips.extend(["0.0.0.0", ""]) # noqa
105+
local_ips.extend(["0.0.0.0", ""]) # noqa: S104
99106

100107
LOCAL_IPS[:] = _uniq_stable(local_ips)
101108
PUBLIC_IPS[:] = _uniq_stable(public_ips)
@@ -155,59 +162,38 @@ def _load_ips_psutil() -> None:
155162
"""load ip addresses with netifaces"""
156163
import psutil
157164

158-
global LOCALHOST
159-
local_ips = []
160-
public_ips = []
165+
addr_dict: dict[str, list[str]] = {}
161166

162167
# dict of iface_name: address_list, eg
163168
# {"lo": [snicaddr(family=<AddressFamily.AF_INET>, address="127.0.0.1",
164169
# ...), snicaddr(family=<AddressFamily.AF_INET6>, ...)]}
165170
for iface, ifaddresses in psutil.net_if_addrs().items():
166-
for address_data in ifaddresses:
167-
if address_data.family == socket.AF_INET:
168-
addr = address_data.address
169-
if not (iface.startswith("lo") or addr.startswith("127.")):
170-
public_ips.append(addr)
171-
elif not LOCALHOST:
172-
LOCALHOST = addr
173-
local_ips.append(addr)
174-
if not LOCALHOST:
175-
# we never found a loopback interface (can this ever happen?), assume common default
176-
LOCALHOST = "127.0.0.1"
177-
local_ips.insert(0, LOCALHOST)
178-
local_ips.extend(["0.0.0.0", ""]) # noqa
179-
LOCAL_IPS[:] = _uniq_stable(local_ips)
180-
PUBLIC_IPS[:] = _uniq_stable(public_ips)
171+
addr_dict[iface] = [
172+
address_data.address
173+
for address_data in ifaddresses
174+
if address_data.family == socket.AF_INET
175+
]
176+
177+
_populate_from_dict(addr_dict)
181178

182179

183180
def _load_ips_netifaces() -> None:
184181
"""load ip addresses with netifaces"""
185182
import netifaces # type: ignore[import-not-found]
186183

187-
global LOCALHOST
188-
local_ips = []
189-
public_ips = []
184+
addr_dict: dict[str, list[str]] = {}
190185

191186
# list of iface names, 'lo0', 'eth0', etc.
192187
for iface in netifaces.interfaces():
193188
# list of ipv4 addrinfo dicts
189+
addr_dict[iface] = []
190+
194191
ipv4s = netifaces.ifaddresses(iface).get(netifaces.AF_INET, [])
195192
for entry in ipv4s:
196193
addr = entry.get("addr")
197-
if not addr:
198-
continue
199-
if not (iface.startswith("lo") or addr.startswith("127.")):
200-
public_ips.append(addr)
201-
elif not LOCALHOST:
202-
LOCALHOST = addr
203-
local_ips.append(addr)
204-
if not LOCALHOST:
205-
# we never found a loopback interface (can this ever happen?), assume common default
206-
LOCALHOST = "127.0.0.1"
207-
local_ips.insert(0, LOCALHOST)
208-
local_ips.extend(["0.0.0.0", ""]) # noqa
209-
LOCAL_IPS[:] = _uniq_stable(local_ips)
210-
PUBLIC_IPS[:] = _uniq_stable(public_ips)
194+
if addr:
195+
addr_dict[iface].append(addr)
196+
_populate_from_dict(addr_dict)
211197

212198

213199
def _load_ips_gethostbyname() -> None:

0 commit comments

Comments
 (0)