-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathapp.py
460 lines (359 loc) · 15.3 KB
/
app.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
from opendomain_responder import basic_qa
import qi
import argparse
import os
import sys
import time
from PIL import Image
import requests
import json
import base64
import random
import numpy as np
import cv2
import datetime
from face_trainer import FaceTrainer
from face_detector import FaceDetector
from basic_awareness import HumanTrackedEventWatcher
####################################
# DEVELOPER : VIKRAM SINGH #
# TECHNOLOGY STACK : PYTHON #
####################################
# ==============================================================================
# --- CAMERA INFORMATION ---
# AL_resolution
AL_kQQQQVGA = 8 # Image of 40*30px
AL_kQQQVGA = 7 # Image of 80*60px
AL_kQQVGA = 0 # Image of 160*120px
AL_kQVGA = 1 # Image of 320*240px
AL_kVGA = 2 # Image of 640*480px
AL_k4VGA = 3 # Image of 1280*960px
AL_k16VGA = 4 # Image of 2560*1920px
# Camera IDs
AL_kTopCamera = 0
AL_kBottomCamera = 1
AL_kDepthCamera = 2
# Need to add All color space variables
AL_kRGBColorSpace = 13
# ==============================================================================
# ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
# =============================== CONFIG =======================================
NODE_IP = "10.9.43.90" # IP Address of the node in which you are running this program
# Name of topic file situated at robots dir => `/home/nao/chat/`
TOPIC_NAME = "face_detect_greeter.top"
GREET_NAMES_LOGS_PATH = "recorded_data/greet_logs/" # saved names list of people recognised
TIME_GAP_TO_WISH_GUEST = 5 # In minutes
TIME_GAP_TO_WISH_KNOWN_PERSON = 5 # In minutes
# ==============================================================================
# =============== ORIENTATION BASED ON INITIAL BOT POSITION ====================
PHI = 0 # AMOUNT TO MOVE TO FACE TOWARDS EAST (0`)
# 0
# math.pi/2 -math.pi/2
# math.pi
PENDING_PHI = 0
# ==============================================================================
class FaceRecognitionGreeter(object):
def __init__(self, app, human_tracked_event_watcher, face_detector, face_trainer):
super(FaceRecognitionGreeter, self).__init__()
try:
app.start()
except RuntimeError:
print ("Can't connect to Naoqi at ip \"" + args.ip + "\" on port " +
str(args.port) + ".\n")
sys.exit(1)
session = app.session
self.subscribers_list = []
# SUBSCRIBING SERVICES
self.memory_service = session.service("ALMemory")
self.motion_service = session.service("ALMotion")
self.posture_service = session.service("ALRobotPosture")
self.video_service = session.service("ALVideoDevice")
self.tts = session.service("ALTextToSpeech")
self.speaking_movement = session.service("ALSpeakingMovement")
self.dialog_service = session.service("ALDialog")
self.tablet_service = session.service("ALTabletService")
self.human_watcher = human_tracked_event_watcher
self.face_trainer = face_trainer
self.face_detector = face_detector
# INITIALISING Variables
self.imageNo2d = 1
self.lastGuestTimestamp = 0
self.startFaceTraining = False
self.userName = '-'
def create_callbacks(self):
self.connect_callback("developer_name_event",
self.developer_name_event)
# self.connect_callback("Dialog/LastInput",
# self.last_dialog_event)
self.connect_callback("unknown_input_event",
self.who_question_event)
self.connect_callback("specific_question_event",
self.specific_question_event)
self.connect_callback("register_human_event",
self.register_human_event)
self.tablet_service.onInputText.connect(self.got_button_event)
return
def connect_callback(self, event_name, callback_func):
print "Callback connection"
subscriber = self.memory_service.subscriber(event_name)
subscriber.signal.connect(callback_func)
self.subscribers_list.append(subscriber)
return
def got_button_event(self, buttonId, value):
if buttonId == 1:
self.userName = value
if value == '':
self.userName = "NULL"
if buttonId == 0: # Cancel is pressed
self.userName = "NULL"
self._clearTablet()
return
def register_human_event(self, value):
print "Registering face training event"
self.startFaceTraining = True
self._makePepperSpeak("Please start looking into my eyes untill I say done")
time.sleep(1)
return
def developer_name_event(self, value):
# setting Memory Variable
self.memory_service.insertData("developerName", "Vikram Singh")
return
# def last_dialog_event(self, value):
# print "Last Dialog Event Called"
# print "Value : ", value
# return
def who_question_event(self, value):
print "Question : ", value
ans = basic_qa.answerQues(value)
if ans:
print "Answer : ", ans
self._makePepperSpeak(ans)
return
def specific_question_event(self, value):
print "Value : ", value
msg = ''
if value == 'TIME':
msg = "Present time is " + time.ctime().split()[3]
if value == 'DAY':
msg = "The day today is " + datetime.datetime.today().strftime('%A')
if value == 'DATE':
msg = "The date today is " + datetime.datetime.today().strftime('%D')
if value == 'MONTH':
msg = "The present month is " + datetime.datetime.today().strftime('%B')
if value == 'YEAR':
msg = "The present year is " + time.ctime().split()[4]
if msg:
self._makePepperSpeak(msg)
return
def _makePepperSpeak(self, userMsg):
# MAKING PEPPER SPEAK
# future = self.animation_player_service.run("animations/Stand/Gestures/Give_3", _async=True)
sentence = "\RSPD=" + str(100) + "\ " # Speed
sentence += "\VCT=" + str(100) + "\ " # Voice Shaping
sentence += userMsg
sentence += "\RST\ "
self.tts.say(str(sentence))
# future.value()
return
def _startFaceTrainProcedure(self, image):
print
print
print
name = ""
# Capture name from tablet
self._makePepperSpeak("Please enter your name")
self.tablet_service.showInputTextDialog("Please Enter Your Name", "Okay", "Cancel")
while True:
if self.userName != '-':
if self.userName == "NULL":
self.userName = '-'
print "Cancel button clicked or name not given, hence cannot training image"
self.startFaceTraining = False
return
else:
name = self.userName
self.userName = '-'
break
res = self.face_trainer.trainForBot(name, image)
if res[0]:
self._makePepperSpeak("Your Face Training is done")
# Saving file temporarily for displaying
base_url = "/var/www/html/pepper_hack/images/" # your localhost server addr
view_url = base_url + str(name) + '.png'
res[1].save(view_url, 'PNG')
self._showOnTablet(str(name) + '.png', False)
time.sleep(5)
# Deleting File
os.remove(view_url)
self._clearTablet()
else:
self._makePepperSpeak("Sorry I couldn't register you at the moment please try after some time")
# Initializing the Detection Model
self.face_detector.scanKnownFaces()
self.startFaceTraining = False
self.lastGuestTimestamp = 0
return
def _startFrameDetect(self, image):
print
print
print
data = self.face_detector.detectFrameFromModel(image)
if data["faceFound"]:
if data["name"] == "unknown":
if (time.time() - self.lastGuestTimestamp) / 60 > TIME_GAP_TO_WISH_GUEST:
self.lastGuestTimestamp = time.time()
self._makePepperSpeak("Hello Guest")
else:
# Log the greeting
LOG_FILE_PATH = GREET_NAMES_LOGS_PATH + \
'/' + str(datetime.date.today()) + '.log'
with open(LOG_FILE_PATH, 'a+') as f:
userFound = False
lastTimestamp = 0
lines = f.readlines()
for line in lines:
username = line.split(',')[0].strip()
tStmp = float(line.split(',')[2].strip())
if str(username).lower() == str(data["name"]).lower():
userFound = True
lastTimestamp = tStmp
if userFound:
print "User already greeted"
if not userFound or (time.time() - lastTimestamp) / 60 > TIME_GAP_TO_WISH_KNOWN_PERSON:
f.write('{}, {}, {}\n'.format(
data["name"], time.ctime(), time.time()))
self._makePepperSpeak(
"Hello {} How are you".format(data["name"]))
return
def _retrieveVideoFeed(self):
# Initializing the Detection Model
self.face_detector.scanKnownFaces()
# Top Camera
cameraId = 0
# WARNING : The same Name could be used only six time.
strName = "capture2DImage_{}".format(random.randint(1, 10000000000))
clientRGB = self.video_service.subscribeCamera(
strName, cameraId, AL_kQVGA, AL_kRGBColorSpace, 10)
# create image
width = 320
height = 240
image = np.zeros((height, width, 3), np.uint8)
while True:
print "Starting Video Feed to cancel press ESC"
# get image
result = self.video_service.getImageRemote(clientRGB)
if result == None:
print 'cannot capture.'
elif result[6] == None:
print 'no image data string.'
else:
# translate value to mat
values = map(ord, list(str(bytearray(result[6]))))
i = 0
for y in range(0, height):
for x in range(0, width):
image.itemset((y, x, 0), values[i + 0])
image.itemset((y, x, 1), values[i + 1])
image.itemset((y, x, 2), values[i + 2])
i += 3
# show image
cv2.imshow(
"pepper-top-camera-{}*{}".format(width, height), image)
if self.startFaceTraining:
self._startFaceTrainProcedure(image)
else:
self._startFrameDetect(image)
# exit by [ESC]
if cv2.waitKey(33) == 27:
break
def _clearTablet(self):
# Hide the web view
self.tablet_service.hideImage()
self.tablet_service.hideDialog()
return
def _showOnTablet(self, mediaName, videoPresent):
# Display Image On Tablet
base_url = "http://{}/pepper_hack/images/".format(NODE_IP)
view_url = base_url + str(mediaName)
try:
if videoPresent:
self.tablet_service.playVideo(view_url)
else:
# self.tablet_service.showImageNoCache(view_url) # wont use this because we have images which are constant here
self.tablet_service.showImage(view_url)
print "Displaying on tablet : " + view_url
except Exception, err:
print "Error Showing On Tablet : " + str(err)
return
def _addTopic(self):
print "Starting topic adding process"
# Controlling hand gestures and movement while speaking
self.speaking_movement.setEnabled(True)
self.dialog_service.setLanguage("English")
# Loading the topic given by the user (absolute path is required)
topic_path = "/home/nao/chat/{}".format(TOPIC_NAME)
topf_path = topic_path.decode('utf-8')
self.topic_name = self.dialog_service.loadTopic(
topf_path.encode('utf-8'))
# Activating the loaded topic
self.dialog_service.activateTopic(self.topic_name)
# Starting the dialog engine - we need to type an arbitrary string as the identifier
# We subscribe only ONCE, regardless of the number of topics we have activated
self.dialog_service.subscribe('face_detector_example')
print "\nSpeak to the robot using rules. Robot is ready"
return
def _cleanUp(self):
print "Starting Clean Up process"
self.human_watcher.stop_basic_awareness()
# Stopping any movement if there
self.motion_service.stopMove()
# stopping the dialog engine
self.dialog_service.unsubscribe('face_detector_example')
# Deactivating the topic
self.dialog_service.deactivateTopic(self.topic_name)
# now that the dialog engine is stopped and there are no more activated topics,
# we can unload our topic and free the associated memory
self.dialog_service.unloadTopic(self.topic_name)
self._clearTablet()
self.posture_service.goToPosture("StandInit", 0.1)
return
def startGreeterProgram(self):
print "Starting Greeter"
self._retrieveVideoFeed()
return
def run(self):
# start
print "Waiting for the robot to be in wake up position"
self.motion_service.wakeUp()
self.posture_service.goToPosture("StandInit", 0.1)
time.sleep(1)
self.create_callbacks()
self._addTopic()
self.human_watcher.start_basic_awareness()
try:
self.startGreeterProgram()
except KeyboardInterrupt:
print "Interrupted by user, shutting down"
self._cleanUp()
print "Waiting for the robot to be in rest position"
self.motion_service.rest()
sys.exit(0)
return
if __name__ == "__main__":
parser = argparse.ArgumentParser()
parser.add_argument("--ip", type=str, default="10.9.45.11",
help="Robot IP address. On robot or Local Naoqi: use \
'127.0.0.1'.")
parser.add_argument("--port", type=int, default=9559,
help="Naoqi port number")
args = parser.parse_args()
# Initialize qi framework.
connection_url = "tcp://" + args.ip + ":" + str(args.port)
app = qi.Application(["FaceRecognitionGreeter",
"--qi-url=" + connection_url])
human_tracked_event_watcher = HumanTrackedEventWatcher(app)
face_trainer = FaceTrainer()
face_detector = FaceDetector()
event_watcher = FaceRecognitionGreeter(
app, human_tracked_event_watcher, face_detector, face_trainer)
event_watcher.run()