685
|
1 #!camtest/bin/python3
|
1740
|
2 import asyncio
|
685
|
3 import binascii
|
1740
|
4 import io
|
|
5 import json
|
685
|
6 import logging
|
|
7 import os
|
1740
|
8 import time
|
773
|
9
|
1740
|
10 import apriltag
|
|
11 import cv2
|
|
12 import numpy
|
|
13 from aioesphomeapi.client import APIClient
|
|
14 from aioesphomeapi.model import CameraState
|
685
|
15 from aiohttp import web
|
|
16 from aiohttp.web import Response
|
|
17 from aiohttp_sse import sse_response
|
1740
|
18 from docopt import docopt
|
685
|
19
|
1740
|
20 logging.basicConfig(level=logging.INFO)
|
|
21 log = logging.getLogger(__name__)
|
|
22
|
685
|
23
|
|
24 class CameraReceiver:
|
1740
|
25
|
|
26 def __init__(self, host):
|
685
|
27 self.lastFrameTime = None
|
773
|
28 self.host = host
|
685
|
29 self.lastFrame = b"", ''
|
|
30 self.recent = []
|
|
31
|
|
32 async def start(self):
|
773
|
33 try:
|
1740
|
34 self.c = c = APIClient(self.host, 6053, 'MyPassword')
|
773
|
35 await c.connect(login=True)
|
|
36 await c.subscribe_states(on_state=self.on_state)
|
|
37 except OSError:
|
|
38 return
|
1740
|
39 self.t = asyncio.create_task(
|
|
40 self.start_requesting_image_stream_forever())
|
773
|
41
|
|
42 async def start_requesting_image_stream_forever(self):
|
|
43 while True:
|
|
44 try:
|
|
45 await self.c.request_image_stream()
|
|
46 except AttributeError:
|
|
47 return
|
|
48 # https://github.com/esphome/esphome/blob/dev/esphome/components/esp32_camera/esp32_camera.cpp#L265 says a 'stream' is 5 sec long
|
|
49 await asyncio.sleep(4)
|
685
|
50
|
|
51 def on_state(self, s):
|
|
52 if isinstance(s, CameraState):
|
1740
|
53 jpg = s.data
|
685
|
54 if len(self.recent) > 10:
|
|
55 self.recent = self.recent[-10:]
|
|
56
|
|
57 self.recent.append(jpg)
|
773
|
58 #print('recent lens: %s' % (','.join(str(len(x))
|
|
59 # for x in self.recent)))
|
685
|
60 else:
|
|
61 print('other on_state', s)
|
|
62
|
|
63 def analyze(self, jpg):
|
1740
|
64 img = cv2.imdecode(numpy.asarray(bytearray(jpg)), cv2.IMREAD_GRAYSCALE)
|
685
|
65 result = detector.detect(img)
|
|
66 msg = {}
|
|
67 if result:
|
|
68 center = result[0].center
|
|
69 msg['center'] = [round(center[0], 2), round(center[1], 2)]
|
|
70 return msg
|
|
71
|
|
72 async def frames(self):
|
|
73 while True:
|
|
74 if self.recent:
|
|
75 if self.lastFrameTime and time.time() - self.lastFrameTime > 15:
|
|
76 print('no recent frames')
|
|
77 os.abort()
|
|
78
|
|
79 jpg = self.recent.pop(0)
|
|
80 msg = self.analyze(jpg)
|
|
81 yield jpg, msg
|
|
82 self.lastFrame = jpg, msg
|
|
83 self.lastFrameTime = time.time()
|
|
84 else:
|
773
|
85 await asyncio.sleep(.05)
|
685
|
86
|
|
87
|
|
88 def imageUri(jpg):
|
|
89 return 'data:image/jpeg;base64,' + binascii.b2a_base64(jpg).decode('ascii')
|
|
90
|
1740
|
91
|
685
|
92 async def stream(request):
|
|
93 async with sse_response(request) as resp:
|
|
94 await resp.send(imageUri(recv.lastFrame[0]))
|
|
95 await resp.send(json.dumps(recv.lastFrame[1]))
|
|
96 async for frame, msg in recv.frames():
|
|
97 await resp.send(json.dumps(msg))
|
|
98 await resp.send(imageUri(frame))
|
|
99 return resp
|
|
100
|
1740
|
101
|
685
|
102 async def index(request):
|
|
103 d = r"""
|
|
104 <html>
|
|
105 <body>
|
|
106 <style>
|
|
107 #center {
|
|
108 position: absolute;
|
|
109 font-size: 35px;
|
|
110 color: orange;
|
|
111 text-shadow: black 0 1px 1px;
|
|
112 margin-left: -14px;
|
|
113 margin-top: -23px;
|
|
114 }
|
|
115 </style>
|
|
116 <script>
|
|
117 var evtSource = new EventSource("/stream");
|
|
118 evtSource.onmessage = function(e) {
|
|
119 if (e.data[0] == '{') {
|
|
120 const msg = JSON.parse(e.data);
|
|
121 const st = document.querySelector('#center').style;
|
|
122 if (msg.center) {
|
|
123 st.left = msg.center[0];
|
|
124 st.top = msg.center[1];
|
|
125 } else {
|
|
126 st.left = -999;
|
|
127 }
|
|
128 } else {
|
|
129 document.getElementById('response').src = e.data;
|
|
130 }
|
|
131 }
|
|
132 </script>
|
|
133 <h1>Response from server:</h1>
|
|
134 <div style="position: relative">
|
|
135 <img id="response"></img>
|
|
136 <span id="center" style="position: absolute">◎</span>
|
|
137 </div>
|
|
138 </body>
|
|
139 </html>
|
|
140 """
|
|
141 return Response(text=d, content_type='text/html')
|
|
142
|
773
|
143 arguments = docopt('''
|
|
144 this
|
685
|
145
|
773
|
146 Usage:
|
1740
|
147 this [-v] [--cam host] [--port to_serve]
|
773
|
148
|
|
149 Options:
|
1740
|
150 --port n http server [default: 10020]
|
|
151 --cam host hostname of esphome server
|
773
|
152 ''')
|
|
153
|
|
154 logging.getLogger('aioesphomeapi.connection').setLevel(logging.INFO)
|
|
155
|
1740
|
156 recv = CameraReceiver(arguments['--cam'])
|
773
|
157
|
|
158 detector = apriltag.Detector()
|
|
159
|
|
160 f = recv.start()
|
1740
|
161 async def starter(app):
|
|
162 asyncio.create_task(f)
|
773
|
163
|
|
164 start_time = time.time()
|
685
|
165 app = web.Application()
|
1740
|
166 app.on_startup.append(starter)
|
685
|
167 app.router.add_route('GET', '/stream', stream)
|
|
168 app.router.add_route('GET', '/', index)
|
1740
|
169 web.run_app(app, port=int(arguments['--port'])) |