-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcollect_data.py
executable file
·234 lines (207 loc) · 9.84 KB
/
collect_data.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
#!/usr/bin/env python3
import concurrent.futures
import cozmo
import csv
import os
import pygame
import time
from argparse import ArgumentParser
from cozmo.util import degrees
from datetime import datetime
# globals (imported in drive.py)
# wheel speeds in millimiters per second
# (left wheel speed, right wheel speed)
go_forward = (100, 100)
go_backward = (-100, -100)
turn_left = (-40, 40)
turn_right = (40, -40)
# head speeds in radians per second
tilt_up = 0.3
tilt_down = -0.3
def add_speed(wheel_speeds: tuple, addition: tuple):
return (wheel_speeds[0] + addition[0], wheel_speeds[1] + addition[1])
def collect_data(robot: cozmo.robot.Robot, frequency = 10.0, append=False, no_save=False, verbose=False):
def drive(robot: cozmo.robot.Robot, wheels, head, stop):
event = pygame.event.poll()
if event.type == pygame.KEYDOWN or event.type == pygame.KEYUP:
keys = pygame.key.get_pressed()
# get wheel keys
wheels["up"] = keys[pygame.K_UP]
wheels["down"] = keys[pygame.K_DOWN]
wheels["left"] = keys[pygame.K_LEFT]
wheels["right"] = keys[pygame.K_RIGHT]
# get head keys
head["up"] = keys[pygame.K_w]
head["down"] = keys[pygame.K_r] # colemak
# head["down"] = keys[pygame.K_s] # QWERTY
# get "stop" key
stop = keys[pygame.K_SPACE]
# calculate wheel speeds
wheel_speeds = (0.0, 0.0)
# up arrow key
if wheels["up"]:
wheel_speeds = add_speed(wheel_speeds, go_forward)
# down arrow key
if wheels["down"]:
wheel_speeds = add_speed(wheel_speeds, go_backward)
# left arrow key
if wheels["left"]:
wheel_speeds = add_speed(wheel_speeds, turn_left)
# right arrow key
if wheels["right"]:
wheel_speeds = add_speed(wheel_speeds, turn_right)
# calculate head speed
head_speed = 0
if head["up"]:
# this is fine in a nested function since nested functions can access
# outer function variables (tilt_up/down) but not modify them
head_speed += tilt_up
if head["down"]:
head_speed += tilt_down
# send wheel speeds to cozmo
robot.drive_wheel_motors(wheel_speeds[0], wheel_speeds[1], l_wheel_acc=500, r_wheel_acc=500)
# send head speed
robot.move_head(head_speed)
return wheels, head, stop
pygame.init()
screen = pygame.display.set_mode((100,100))
# other variables we need later
period = 1/frequency # default: 0.1 sec
wheels = {"up": 0, "down": 0, "left": 0, "right": 0}
head = {"up": 0, "down": 0}
stop = 0
if not no_save:
header = ["left wheel speed", "right wheel speed", "gyro x", "gyro y", "gyro z", "pose x", "pose y", "pose z",
"q0", "q1", "q2", "q3", "angle_z", "origin_id", "is_accurate", "wheels: up key", "wheels: down key",
"wheels: left key", "wheels: right key", "head: up key", "head: down key", "stop", "timestamp"]
# where jpg filenames should begin. start at 1 so filenames match rows in csv (accounting for header row)
start_from = 1
if append: # if we're appending, need to update starting point
img_list = [int(filename[:-4]) for filename in os.listdir("img") if filename.endswith(".jpg")]
if len(img_list) > 0: # it's possible we're appending to a folder without any images
start_from = max(img_list) + 1
begin, end = start_from, start_from # we use these to calculate how many new data points are added
print("Press [space] to stop")
try:
if not no_save:
with open("data.csv", "a" if append else "x") as file:
writer = csv.writer(file)
if not append:
writer.writerow(header)
while True:
# start timer
start_time = time.time()
# drive cozmo
wheels, head, stop = drive(robot, wheels, head, stop)
# save data
img = robot.world.latest_image
if img is not None:
# save image
if verbose:
print("Saving image {}".format(end))
img.raw_image.save("img/{}.jpg".format(end))
# save time/speed/pose data
present_time = datetime.now().time()
lws = robot.left_wheel_speed.speed_mmps
rws = robot.right_wheel_speed.speed_mmps
gyro = robot.gyro
pose = robot.pose
row = [robot.left_wheel_speed.speed_mmps, robot.right_wheel_speed.speed_mmps, gyro.x, gyro.y, gyro.z, datetime.now().time()]
row = [robot.pose.position, robot.pose.rotation]
row = [lws, rws, gyro.x, gyro.y, gyro.z, pose.position.x, pose.position.y, pose.position.z, pose.rotation.q0, pose.rotation.q1, pose.rotation.q2,
pose.rotation.q3, pose.rotation.angle_z.radians, pose.origin_id, pose.is_accurate, wheels["up"], wheels["down"], wheels["left"],
wheels["right"], head["up"], head["down"], stop, present_time]
if verbose:
print("Saving data to row {}".format(end))
writer.writerow(row)
# update count
end += 1
if stop:
raise KeyboardInterrupt()
# end timer
end_time = time.time()
# calculate how long to sleep to mantain desired frequency (default 10 Hz)
sample_time = start_time - end_time
if sample_time < period:
time.sleep(period - sample_time)
else:
while True:
# start timer
start_time = time.time()
# drive cozmo
wheels, head, stop = drive(robot, wheels, head, stop)
if stop:
raise KeyboardInterrupt()
# end timer
end_time = time.time()
# calculate how long to sleep to mantain desired frequency (default 10 Hz)
sample_time = start_time - end_time
if sample_time < period:
time.sleep(period - sample_time)
except (KeyboardInterrupt, SystemExit):
print("Stop command received")
if not no_save:
print("Saved {} new data points".format(end - begin + 1))
finally:
robot.stop_all_motors()
pygame.quit()
def init(robot: cozmo.robot.Robot, frequency = 10.0):
"""Collect LfD data.
Arguments:
robot (cozmo.robot.Robot): cozmo robot object
"""
parser = ArgumentParser(description="Collect LfD data.")
parser.add_argument("-a", "--append_to", default=None, help="folder to which new data should be appended (default: None)")
parser.add_argument("-v", "--verbose", action="store_true", help="print extra messages (default: False)")
parser.add_argument("-c", "--color", action="store_true", help="use color images (default: False)")
parser.add_argument('-n', "--no_save", action="store_true", help="don't save data (default: False)")
args = parser.parse_args()
if args.color:
robot.camera.color_image_enabled = args.color
time.sleep(0.1) # color takes a bit to kick in
robot.camera.image_stream_enabled = True
append_to = args.append_to
verbose = args.verbose
no_save = args.no_save
absolute_path = os.path.dirname(os.path.realpath(__file__))
if append_to:
folder_name = append_to
mode = "append"
else:
if not no_save:
folder_name = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
os.makedirs("{}/data/{}/img".format(absolute_path, folder_name)) # root folder contains csv, img contains images
mode = "write"
try:
if not no_save:
os.chdir("{}/data/{}".format(absolute_path, folder_name))
if verbose:
print("Saving data in {} mode".format(mode))
collect_data(robot, frequency, append=True if append_to else False, no_save=no_save, verbose=verbose)
if not no_save:
os.chdir("..")
custom_name = input("What would you like to call this data folder? (leave empty to use {}){}".format(
"original name" if append_to else "timestamp", os.linesep))
while True:
if len(custom_name) > 0:
try:
os.rename(folder_name, custom_name)
break
except OSError:
custom_name = input("That folder already exists. Try another name (leave empty to use {}):{}".format(
"original name" if append_to else "timestamp", os.linesep))
else:
break
print("Data saved to {}".format(custom_name if len(custom_name) > 0 else folder_name))
else:
print("Data not saved per command-line arguments.")
except NotADirectoryError:
print("Folder does not exist. Was it mistyped?")
finally:
print("Exiting")
def main(robot: cozmo.robot.Robot):
robot.set_head_angle(degrees(0)).wait_for_completed()
robot.set_lift_height(0).wait_for_completed()
init(robot, 10)
if __name__ == "__main__":
cozmo.run_program(main)