Skip to content

Commit 65dd547

Browse files
authored
Merge pull request #65 from fronzbot/battery-percent
Added new request url for camera config
2 parents 66a6701 + 91a877d commit 65dd547

File tree

4 files changed

+86
-17
lines changed

4 files changed

+86
-17
lines changed

CHANGES.rst

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,9 @@ A list of changes between each release
55

66
0.8.0.dev (Development version)
77
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
8+
- Added support for battery voltage level
9+
- Added motion detection per camera
10+
- Added fully accessible camera configuration dict
811

912
0.7.0 (2018-02-08)
1013
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

blinkpy/blinkpy.py

Lines changed: 43 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -107,13 +107,17 @@ def __init__(self, config, blink):
107107
config['thumbnail'])
108108
self.clip = "{}{}".format(self.urls.base_url, config['video'])
109109
self.temperature = config['temp']
110-
self.battery = config['battery']
110+
self._battery_string = config['battery']
111111
self.notifications = config['notifications']
112-
self.motion = {}
112+
self.motion = dict()
113113
self.header = None
114114
self.image_link = None
115115
self.arm_link = None
116116
self.region_id = config['region_id']
117+
self.battery_voltage = -180
118+
self.motion_detected = None
119+
self.wifi_strength = None
120+
self.camera_config = dict()
117121

118122
@property
119123
def attributes(self):
@@ -128,6 +132,8 @@ def attributes(self):
128132
'thumbnail': self.thumbnail,
129133
'video': self.clip,
130134
'notifications': self.notifications,
135+
'motion_detected': self.motion_detected,
136+
'wifi_strength': self.wifi_strength,
131137
'network_id': self.blink.network_id
132138
}
133139
return attributes
@@ -142,13 +148,18 @@ def armed(self):
142148
"""Return camera arm status."""
143149
return True if self._status == 'armed' else False
144150

151+
@property
152+
def battery(self):
153+
"""Return battery level as percentage."""
154+
return round(self.battery_voltage / 180 * 100)
155+
145156
@property
146157
def battery_string(self):
147158
"""Return string indicating battery status."""
148159
status = "Unknown"
149-
if self.battery > 1 and self.battery <= 3:
160+
if self._battery_string > 1 and self._battery_string <= 3:
150161
status = "OK"
151-
elif self.battery >= 0:
162+
elif self._battery_string >= 0:
152163
status = "Low"
153164
return status
154165

@@ -175,10 +186,26 @@ def update(self, values):
175186
self.urls.base_url, values['thumbnail'])
176187
self.clip = "{}{}".format(
177188
self.urls.base_url, values['video'])
178-
self.temperature = values['temp']
179-
self.battery = values['battery']
189+
self._battery_string = values['battery']
180190
self.notifications = values['notifications']
181191

192+
try:
193+
cfg = self.blink.camera_config_request(self.id)
194+
self.camera_config = cfg
195+
except requests.exceptions.RequestException as err:
196+
_LOGGER.warning("Could not get config for %s with id %s",
197+
self.name, self.id)
198+
_LOGGER.warning("Exception raised: %s", err)
199+
200+
try:
201+
self.battery_voltage = cfg['camera'][0]['battery_voltage']
202+
self.motion_detected = cfg['camera'][0]['motion_alert']
203+
self.wifi_strength = cfg['camera'][0]['wifi_strength']
204+
self.temperature = cfg['camera'][0]['temperature']
205+
except KeyError:
206+
_LOGGER.warning("Problem extracting config for camera %s",
207+
self.name)
208+
182209
def image_refresh(self):
183210
"""Refresh current thumbnail."""
184211
url = self.urls.home_url
@@ -327,6 +354,7 @@ def get_videos(self, start_page=0, end_page=1):
327354
videos.append(this_page)
328355

329356
for page in videos:
357+
_LOGGER.debug("Retrieved video page %s", page)
330358
for entry in page:
331359
camera_name = entry['camera_name']
332360
clip_addr = entry['address']
@@ -365,6 +393,7 @@ def get_cameras(self):
365393
device = BlinkCamera(element, self)
366394
self.cameras[device.name] = device
367395
self._idlookup[device.id] = device.name
396+
self.refresh()
368397

369398
def set_links(self):
370399
"""Set access links and required headers for each camera in system."""
@@ -497,3 +526,11 @@ def _status_request(self):
497526
self.network_id)
498527
headers = self._auth_header
499528
return _request(self, url=url, headers=headers, reqtype='get')
529+
530+
def camera_config_request(self, camera_id):
531+
"""Retrieve more info about Blink config."""
532+
url = "{}/network/{}/camera/{}/config".format(self.urls.base_url,
533+
self.network_id,
534+
str(camera_id))
535+
headers = self._auth_header
536+
return _request(self, url=url, headers=headers, reqtype='get')

tests/test_blink_cameras.py

Lines changed: 31 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,17 @@
1515
USERNAME = 'foobar'
1616
PASSWORD = 'deadbeef'
1717

18+
CAMERA_CFG = {
19+
'camera': [
20+
{
21+
'battery_voltage': 90,
22+
'motion_alert': True,
23+
'wifi_strength': -30,
24+
'temperature': 68
25+
}
26+
]
27+
}
28+
1829

1930
class TestBlinkCameraSetup(unittest.TestCase):
2031
"""Test the Blink class in blinkpy."""
@@ -35,18 +46,21 @@ def setUp(self):
3546
'notifications': 2,
3647
'region_id': 'test'
3748
}
49+
3850
self.blink.urls = blinkpy.BlinkURLHandler('test')
3951
self.blink.network_id = '0000'
4052

4153
def tearDown(self):
4254
"""Clean up after test."""
4355
self.blink = None
4456

57+
@mock.patch('blinkpy.blinkpy.Blink.camera_config_request',
58+
return_value=CAMERA_CFG)
4559
@mock.patch('blinkpy.blinkpy.requests.post',
4660
side_effect=mresp.mocked_requests_post)
4761
@mock.patch('blinkpy.blinkpy.requests.get',
4862
side_effect=mresp.mocked_requests_get)
49-
def test_camera_properties(self, mock_get, mock_post):
63+
def test_camera_properties(self, mock_get, mock_post, mock_cfg):
5064
"""Tests all property set/recall."""
5165
self.blink.urls = blinkpy.BlinkURLHandler('test')
5266

@@ -56,7 +70,7 @@ def test_camera_properties(self, mock_get, mock_post):
5670

5771
for name in self.blink.cameras:
5872
camera = self.blink.cameras[name]
59-
73+
camera.update(self.camera_config)
6074
self.assertEqual(camera.id, '1111')
6175
self.assertEqual(camera.name, 'foobar')
6276
self.assertEqual(camera.armed, False)
@@ -68,19 +82,21 @@ def test_camera_properties(self, mock_get, mock_post):
6882
camera.clip,
6983
"https://rest.test.{}/test/clip/clip.mp4".format(BLINK_URL)
7084
)
71-
self.assertEqual(camera.temperature, 70)
72-
self.assertEqual(camera.battery, 3)
85+
self.assertEqual(camera.temperature, 68)
86+
self.assertEqual(camera.battery, 50)
7387
self.assertEqual(camera.battery_string, "OK")
7488
self.assertEqual(camera.notifications, 2)
7589
self.assertEqual(camera.region_id, 'test')
90+
self.assertEqual(camera.motion_detected, True)
91+
self.assertEqual(camera.wifi_strength, -30)
92+
7693
camera_config = self.camera_config
7794
camera_config['active'] = 'armed'
7895
camera_config['thumbnail'] = '/test2/image'
7996
camera_config['video'] = '/test2/clip.mp4'
8097
camera_config['temp'] = 60
8198
camera_config['battery'] = 0
8299
camera_config['notifications'] = 4
83-
84100
for name in self.blink.cameras:
85101
camera = self.blink.cameras[name]
86102
camera.update(camera_config)
@@ -93,8 +109,8 @@ def test_camera_properties(self, mock_get, mock_post):
93109
camera.clip,
94110
"https://rest.test.{}/test2/clip.mp4".format(BLINK_URL)
95111
)
96-
self.assertEqual(camera.temperature, 60)
97-
self.assertEqual(camera.battery, 0)
112+
self.assertEqual(camera.temperature, 68)
113+
self.assertEqual(camera.battery, 50)
98114
self.assertEqual(camera.battery_string, "Low")
99115
self.assertEqual(camera.notifications, 4)
100116
camera_config['battery'] = -10
@@ -107,7 +123,9 @@ def test_camera_case(self):
107123
self.blink.cameras['foobar'] = camera_object
108124
self.assertEqual(camera_object, self.blink.cameras['fOoBaR'])
109125

110-
def test_camera_attributes(self):
126+
@mock.patch('blinkpy.blinkpy.Blink.camera_config_request',
127+
return_value=CAMERA_CFG)
128+
def test_camera_attributes(self, mock_cfg):
111129
"""Tests camera attributes."""
112130
self.blink.urls = blinkpy.BlinkURLHandler('test')
113131

@@ -117,6 +135,7 @@ def test_camera_attributes(self):
117135

118136
for name in self.blink.cameras:
119137
camera = self.blink.cameras[name]
138+
camera.update(self.camera_config)
120139
camera_attr = camera.attributes
121140
self.assertEqual(camera_attr['device_id'], '1111')
122141
self.assertEqual(camera_attr['name'], 'foobar')
@@ -129,7 +148,9 @@ def test_camera_attributes(self):
129148
camera_attr['video'],
130149
"https://rest.test.{}/test/clip/clip.mp4".format(BLINK_URL)
131150
)
132-
self.assertEqual(camera_attr['temperature'], 70)
133-
self.assertEqual(camera_attr['battery'], 3)
151+
self.assertEqual(camera_attr['temperature'], 68)
152+
self.assertEqual(camera_attr['battery'], 50)
134153
self.assertEqual(camera_attr['notifications'], 2)
135154
self.assertEqual(camera_attr['network_id'], '0000')
155+
self.assertEqual(camera_attr['motion_detected'], True)
156+
self.assertEqual(camera_attr['wifi_strength'], -30)

tests/test_blink_functions.py

Lines changed: 9 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -61,9 +61,17 @@ def test_get_videos(self, req):
6161
'/test/thumb')
6262

6363
@mock.patch('blinkpy.blinkpy._request')
64-
def test_get_cameras(self, req):
64+
@mock.patch('blinkpy.blinkpy.Blink._video_request')
65+
def test_get_cameras(self, vid_req, req):
6566
"""Test camera extraction."""
6667
req.return_value = {'devices': [self.config]}
68+
vid_req.return_value = [
69+
{
70+
'camera_name': 'foobar',
71+
'address': '/new.mp4',
72+
'thumbnail': '/new'
73+
}
74+
]
6775
self.blink.get_cameras()
6876
self.assertTrue('foobar' in self.blink.cameras)
6977

0 commit comments

Comments
 (0)