Skip to content
This repository was archived by the owner on Oct 1, 2021. It is now read-only.

Commit 47d451d

Browse files
scopballoob
authored andcommitted
Start adding type hints, test with mypy (#228)
* Start adding type hints, test with mypy * Turn on and address mypy check_untyped_defs
1 parent 440d535 commit 47d451d

17 files changed

+79
-30
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -10,3 +10,4 @@ lib64
1010
pip-selfcheck.json
1111
.tox
1212
.pytest_cache/
13+
.mypy_cache/

.travis.yml

+2
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,8 @@ matrix:
66
env: TOXENV=py34
77
- python: "3.4.2"
88
env: TOXENV=lint
9+
- python: "3.4.2"
10+
env: TOXENV=typing
911
- python: "3.5"
1012
env: TOXENV=py35
1113
- python: "3.6"

netdisco/daikin.py

+5-4
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
import socket
33

44
from datetime import timedelta
5+
from typing import Dict, List # noqa: F401
56
from urllib.parse import unquote
67

78
DISCOVERY_MSG = b"DAIKIN_UDP/common/basic_info"
@@ -18,7 +19,7 @@ class Daikin:
1819

1920
def __init__(self):
2021
"""Initialize the Daikin discovery."""
21-
self.entries = []
22+
self.entries = [] # type: List[Dict[str, str]]
2223

2324
def scan(self):
2425
"""Scan the network."""
@@ -47,9 +48,9 @@ def update(self):
4748
try:
4849
data, (address, _) = sock.recvfrom(1024)
4950

50-
# pylint: disable=consider-using-dict-comprehension
51-
entry = dict([e.split('=')
52-
for e in data.decode("UTF-8").split(',')])
51+
entry = {x[0]: x[1] for x in (
52+
e.split('=', 1)
53+
for e in data.decode("UTF-8").split(','))}
5354

5455
# expecting product, mac, activation code, version
5556
if 'ret' not in entry or entry['ret'] != 'OK':

netdisco/discoverables/__init__.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,17 @@
11
"""Provides helpful stuff for discoverables."""
22
# pylint: disable=abstract-method
33
import ipaddress
4+
from typing import Dict, TYPE_CHECKING # noqa: F401
45
from urllib.parse import urlparse
56

67
from ..const import (
78
ATTR_NAME, ATTR_MODEL_NAME, ATTR_HOST, ATTR_PORT, ATTR_SSDP_DESCRIPTION,
89
ATTR_SERIAL, ATTR_MODEL_NUMBER, ATTR_HOSTNAME, ATTR_MAC_ADDRESS,
910
ATTR_PROPERTIES, ATTR_MANUFACTURER, ATTR_UDN, ATTR_UPNP_DEVICE_TYPE)
1011

12+
if TYPE_CHECKING:
13+
from zeroconf import ServiceInfo # noqa: F401
14+
1115

1216
class BaseDiscoverable:
1317
"""Base class for discoverable services or device types."""
@@ -80,7 +84,7 @@ def __init__(self, netdis, typ):
8084
"""Initialize MDNSDiscoverable."""
8185
self.netdis = netdis
8286
self.typ = typ
83-
self.services = {}
87+
self.services = {} # type: Dict[str, ServiceInfo]
8488

8589
netdis.mdns.register_service(self)
8690

netdisco/discoverables/wink.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,16 @@
11
"""Discover Wink hub devices."""
2+
from typing import List # noqa: F401
3+
24
from . import SSDPDiscoverable
5+
from ..ssdp import UPNPEntry # noqa: F401
36

47

58
class Discoverable(SSDPDiscoverable):
69
"""Add support for discovering Wink hub devices."""
710

811
def get_entries(self):
912
"""Return all Wink entries."""
10-
results = []
13+
results = [] # type: List[UPNPEntry]
1114
results.extend(self.find_by_st('urn:wink-com:device:hub2:2'))
1215
results.extend(self.find_by_st('urn:wink-com:device:hub:2'))
1316
results.extend(self.find_by_st('urn:wink-com:device:relay:2'))

netdisco/discovery.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -123,7 +123,8 @@ def _load_device_support(self):
123123
module = importlib.import_module(
124124
discoverables_format.format(module_name))
125125

126-
self.discoverables[module_name] = module.Discoverable(self)
126+
self.discoverables[module_name] = \
127+
getattr(module, 'Discoverable')(self)
127128

128129
def print_raw_data(self):
129130
"""Helper method to show what is discovered in your network."""

netdisco/gdm.py

+6-5
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,14 @@
88
"""
99
import socket
1010
import struct
11+
from typing import Any, Dict, List # noqa: F401
1112

1213

1314
class GDM:
1415
"""Base class to discover GDM services."""
1516

1617
def __init__(self):
17-
self.entries = []
18+
self.entries = [] # type: List[Dict[str, Any]]
1819
self.last_scan = None
1920

2021
def scan(self):
@@ -81,13 +82,13 @@ def update(self):
8182
# Look for responses from all recipients
8283
while True:
8384
try:
84-
data, server = sock.recvfrom(1024)
85-
data = data.decode('utf-8')
85+
bdata, server = sock.recvfrom(1024)
86+
data = bdata.decode('utf-8')
8687
if '200 OK' in data.splitlines()[0]:
87-
data = {k: v.strip() for (k, v) in (
88+
ddata = {k: v.strip() for (k, v) in (
8889
line.split(':') for line in
8990
data.splitlines() if ':' in line)}
90-
self.entries.append({'data': data,
91+
self.entries.append({'data': ddata,
9192
'from': server})
9293
except socket.timeout:
9394
break

netdisco/lms.py

+4-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
"""Squeezebox/Logitech Media server discovery."""
22
import socket
3+
from typing import Dict, List, Union # noqa: F401
34

45
from .const import ATTR_HOST, ATTR_PORT
56

@@ -12,7 +13,7 @@ class LMS:
1213

1314
def __init__(self):
1415
"""Initialize the Logitech discovery."""
15-
self.entries = []
16+
self.entries = [] # type: List[Dict[str, Union[str, int]]]
1617
self.last_scan = None
1718

1819
def scan(self):
@@ -49,7 +50,8 @@ def update(self):
4950
# Where YY is length of port string (ie 4)
5051
# And XXXX is the web interface port (ie 9000)
5152
port = None
52-
if data.startswith(b'JSON', 1):
53+
# https://github.com/python/typeshed/pull/2696
54+
if data.startswith(b'JSON', 1): # type: ignore
5355
length = data[5:6][0]
5456
port = int(data[0-length:])
5557
entries.append({

netdisco/mdns.py

+4-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
"""Add support for discovering mDNS services."""
2+
from typing import List # noqa: F401
3+
24
import zeroconf
35

46

@@ -8,8 +10,8 @@ class MDNS:
810
def __init__(self):
911
"""Initialize the discovery."""
1012
self.zeroconf = None
11-
self.services = []
12-
self._browsers = []
13+
self.services = [] # type: List[zeroconf.ServiceInfo]
14+
self._browsers = [] # type: List[zeroconf.ServiceBrowser]
1315

1416
def register_service(self, service):
1517
"""Register a mDNS service."""

netdisco/service.py

+3-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
import threading
44
import time
55
from collections import defaultdict
6+
from typing import Any, Callable, Dict, List # noqa: F401
67

78
from .discovery import NetworkDiscovery
89

@@ -25,7 +26,7 @@ def __init__(self, interval=DEFAULT_INTERVAL):
2526
self.interval = interval
2627

2728
# Listeners for new services
28-
self.listeners = []
29+
self.listeners = [] # type: List[Callable[[str, Any], None]]
2930

3031
# To track when we have to stop
3132
self._stop = threading.Event()
@@ -38,7 +39,7 @@ def __init__(self, interval=DEFAULT_INTERVAL):
3839

3940
# Dict to keep track of found services. We do not want to
4041
# broadcast the same found service twice.
41-
self._found = defaultdict(list)
42+
self._found = defaultdict(list) # type: Dict[str, List]
4243

4344
def add_listener(self, listener):
4445
"""Add a listener for new services."""

netdisco/smartglass.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
import struct
44
import binascii
55
from datetime import timedelta
6+
from typing import Any, Dict, List, Optional, Tuple # noqa: F401
67

78

89
DISCOVERY_PORT = 5050
@@ -25,13 +26,15 @@
2526
"""
2627
DISCOVERY_CLIENT_TYPE = 4
2728

29+
_Response = Dict[str, Any]
30+
2831

2932
class XboxSmartGlass:
3033
"""Base class to discover Xbox SmartGlass devices."""
3134

3235
def __init__(self):
3336
"""Initialize the Xbox SmartGlass discovery."""
34-
self.entries = []
37+
self.entries = [] # type: List[Tuple[str, Optional[_Response]]]
3538
self._discovery_payload = self.discovery_packet()
3639

3740
@staticmethod

netdisco/ssdp.py

+4-3
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
import socket
55
import logging
66
from datetime import datetime, timedelta
7+
from typing import Dict, List, Optional, Set # noqa: F401
78
from xml.etree import ElementTree
89

910
import requests
@@ -33,7 +34,7 @@ class SSDP:
3334

3435
def __init__(self):
3536
"""Initialize the discovery."""
36-
self.entries = []
37+
self.entries = [] # type: List[UPNPEntry]
3738
self.last_scan = None
3839

3940
def scan(self):
@@ -65,7 +66,7 @@ def find_by_device_description(self, values):
6566
"""
6667
self.update()
6768

68-
seen = set()
69+
seen = set() # type: Set[Optional[str]]
6970
results = []
7071

7172
# Make unique based on the location since we don't care about ST here
@@ -100,7 +101,7 @@ def remove_expired(self):
100101
class UPNPEntry:
101102
"""Found uPnP entry."""
102103

103-
DESCRIPTION_CACHE = {'_NO_LOCATION': {}}
104+
DESCRIPTION_CACHE = {'_NO_LOCATION': {}} # type: Dict[str, Dict]
104105

105106
def __init__(self, values):
106107
"""Initialize the discovery."""

netdisco/tellstick.py

+4-3
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
import socket
33
from datetime import timedelta
44
import logging
5+
from typing import List, Tuple # noqa: F401
56

67

78
DISCOVERY_PORT = 30303
@@ -15,7 +16,7 @@ class Tellstick:
1516

1617
def __init__(self):
1718
"""Initialize the Tellstick discovery."""
18-
self.entries = []
19+
self.entries = [] # type: List[Tuple[str]]
1920

2021
def scan(self):
2122
"""Scan the network."""
@@ -43,8 +44,8 @@ def update(self):
4344
# expecting product, mac, activation code, version
4445
if len(entry) != 4:
4546
continue
46-
entry = (address,) + tuple(entry)
47-
entries.append(entry)
47+
entry.insert(0, address)
48+
entries.append(tuple(entry))
4849

4950
except socket.timeout:
5051
break

netdisco/util.py

+10-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
"""Util functions used by Netdisco."""
22
from collections import defaultdict
3+
from typing import Any, Dict, List, Optional # noqa: F401
34

45

56
# Taken from http://stackoverflow.com/a/10077069
@@ -9,21 +10,26 @@ def etree_to_dict(t):
910
# strip namespace
1011
tag_name = t.tag[t.tag.find("}")+1:]
1112

12-
d = {tag_name: {} if t.attrib else None}
13+
d = {
14+
tag_name: {} if t.attrib else None
15+
} # type: Dict[str, Optional[Dict[str, Any]]]
1316
children = list(t)
1417
if children:
15-
dd = defaultdict(list)
18+
dd = defaultdict(list) # type: Dict[str, List]
1619
for dc in map(etree_to_dict, children):
1720
for k, v in dc.items():
1821
dd[k].append(v)
1922
d = {tag_name: {k: v[0] if len(v) == 1 else v for k, v in dd.items()}}
23+
dt = d[tag_name]
2024
if t.attrib:
21-
d[tag_name].update(('@' + k, v) for k, v in t.attrib.items())
25+
assert dt is not None
26+
dt.update(('@' + k, v) for k, v in t.attrib.items())
2227
if t.text:
2328
text = t.text.strip()
2429
if children or t.attrib:
2530
if text:
26-
d[tag_name]['#text'] = text
31+
assert dt is not None
32+
dt['#text'] = text
2733
else:
2834
d[tag_name] = text
2935
return d

requirements.txt

+1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,3 @@
11
zeroconf>=0.21.0
22
requests>=2.0
3+
typing; python_version<'3.5'

setup.cfg

+13
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,16 @@
11
[tool:pytest]
22
testpaths = tests
33
norecursedirs = .git
4+
5+
[mypy]
6+
check_untyped_defs = true
7+
# TODO disallow_untyped_calls = true
8+
# TODO disallow_untyped_defs = true
9+
follow_imports = silent
10+
ignore_missing_imports = true
11+
no_implicit_optional = true
12+
warn_incomplete_stub = true
13+
warn_redundant_casts = true
14+
warn_return_any = true
15+
warn_unused_configs = true
16+
warn_unused_ignores = true

tox.ini

+7-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
[tox]
2-
envlist = py34, py35, py36, py37, lint
2+
envlist = py34, py35, py36, py37, lint, typing
33
skip_missing_interpreters = True
44

55
[testenv]
@@ -18,3 +18,9 @@ deps =
1818
commands =
1919
flake8 netdisco tests setup.py example_service.py
2020
pylint netdisco tests
21+
22+
[testenv:typing]
23+
deps =
24+
mypy>=0.650
25+
commands =
26+
mypy netdisco tests setup.py example_service.py

0 commit comments

Comments
 (0)