-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathspot_sweep_hover_in_bounds.py
225 lines (186 loc) · 8.35 KB
/
spot_sweep_hover_in_bounds.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
"""
Small Demo for Spot.
"""
import argparse
import asyncio
import time
from typing import Generator, Literal
from bosdyn.client.frame_helpers import ODOM_FRAME_NAME
from bosdyn.client.math_helpers import Quat, SE3Pose, math
from spot_api import Spot
Num = float | int
def parse_args():
"""Parses arguments from the command line."""
# Define parser
parser = argparse.ArgumentParser(
prog='SpotSweeHoverInBounds',
description='Makes Spot sweep the designated area.',
)
# Add arguments
parser.add_argument('--height', type=float, default=0.2, help="How far off the ground to keep Spot's gripper.")
parser.add_argument('--f1', type=int, default=10, help="The number of the top-left fiducial.")
parser.add_argument('--f2', type=int, default=9 , help="The number of the bottom-right fiducial.")
parser.add_argument('-s', '--scanner_length', type=float, default=0.2, help="How long the scan area is. Determines row spacing of S-shaped scan.")
parser.add_argument('-v', '--velocity', type=float, default=0.1 , help="The velocity of the scan in m/s.")
parser.add_argument('-a', '--align', choices=["f1", "f2", "body"], default="f1",
help=
"""
How to align the Scan Box.
The two fiducials define the top-left and bottom-right corner of the area we
are scanning, but there are multiple squares that go through the two points. As such,
we have a decision as to what angle the scan box should have.
If this is "f1", then the box will be aligned with the rotation of the `F1` fiducial.
If this is "f2", then the box will be aligned with the rotation of the `F2` fiducial.
If this is "body", then the box will be aligned with the rotation of the body
when the program is begun.
""".strip()
)
# parse args and return the result
return parser.parse_args()
async def main():
"""
Assumes that there are two fiducials currently in Spot's view, assumes that
they designate a rectangular area that should be swept, and makes Spot sweep
the designated area in an S pattern. The rectangular area uses one of the
points as its top-left point and the other as its bottom-right point.
The gripper hovers over the ground for the duration of the scan. The gripper
is also rotated 45 degrees for the duration of the scan so a scanner (or a
scanner prop) can be easily held by its gripper to perform the actual scan.
"""
args = parse_args()
scan_height: Num = args.height
scan_area_len: Num = args.scanner_length
f1_num: int = args.f1
f2_num: int = args.f2
scan_velocity: Num = args.velocity
scan_area_align: Literal["f1", "f2", "body"] = args.align
# create the Spot robot
spot = Spot()
# Ensure that Spot has an arm.
assert spot.has_arm(), "Spot must have an arm for this demo."
#print(await spot.robot_state())
#print(await spot.robot_hardware())
#print(await spot.robot_metrics())
# Lease the robot
with spot.leased(take=True) as spot:
# Power the motors of the Spot on
await spot.power_motors_on()
# Make Spot stand up
await spot.stand()
time.sleep(1)
# get the fiducials
fiducials = await spot.fiducials_in_view()
# if we have too many or too few we ask the user to add more or disambiguate before trying again
if len(fiducials) != 2:
print(f"Exactly 2 fiducials must be in view. Instead, {len(fiducials)} were in view.")
return
# get the fiducials
f1 = fiducials[0]
f2 = fiducials[1]
if (f2.id == f1_num) or (f1.id == f2_num):
(f1, f2) = (f2, f1)
if f1.id != f1_num:
print(f"fiducial {f1_num} was not seen, fiducial {f1.id} seen instead")
if f2.id != f2_num:
print(f"fiducial {f2_num} was not seen, fiducial {f2.id} seen instead")
print(f"TL: {f1.id}, BR: {f2.id}")
## get the fiducial in the odom frame
odom_t_f1 = f1.odom_t_fiducial()
odom_t_f2 = f2.odom_t_fiducial()
# check that the fiducials are on the same horizontal plane (within a given threshold)
THRESH = 1
if not (-THRESH <= (odom_t_f1.z - odom_t_f2.z) <= THRESH):
print("Detected a large vertical difference between the two fiducials. They must be on the floor.")
return
# Get Points in Accordance to their rotation frame so that the box has the correct alignment
body_t_f1 = f1.body_t_fiducial()
body_t_f2 = f2.body_t_fiducial()
if scan_area_align == "f1":
# Put everything in `F1` frame
f1_t_body = body_t_f1.inverse()
f1_pose = f1_t_body * body_t_f1
f2_pose = f1_t_body * body_t_f2
elif scan_area_align == "f2":
# Put everything in `F2` frame
f2_t_body = body_t_f2.inverse()
f1_pose = f2_t_body * body_t_f1
f2_pose = f2_t_body * body_t_f2
else: # ALIGN == "body"
# Put everything in "body" frame
f1_pose = body_t_f1
f2_pose = body_t_f2
# Generate points
# The points are generated under the assumption that the scan area was
# right in front of spot when it booted up and therefore positive y is
# forward (relative to the bot) and positive x is to the right (relative
# to the bot).
# get the "box" bounds for the rectangle we want to scan within
sort_args = [f1_pose, f2_pose]
min_x = min(*sort_args, key=lambda v: v.x).x # bottom
min_y = min(*sort_args, key=lambda v: v.y).y # right
max_x = max(*sort_args, key=lambda v: v.x).x # top
max_y = max(*sort_args, key=lambda v: v.y).y # left
def frange(start: float, end: float, step: float) -> Generator[float, None, None]:
"""Float version of `range(..)`."""
assert step != 0, "step must not be 0"
if end < start:
step = -abs(step)
start += step
while end < start:
yield start
start += step
else:
step = abs(step)
while start < end:
yield start
start += step
# generate the points to scan
points = []
for (i, x) in enumerate(frange(min_x - scan_area_len/2, max_x - scan_area_len/2, scan_area_len/2)):
if i % 2 == 0:
points.append((x, min_y))
points.append((x, max_y))
else:
points.append((x, max_y))
points.append((x, min_y))
# print((max_x, max_y), (min_x, min_y))
# import pprint
# pprint.pprint(points)
# return
# translate points to odom frame
if scan_area_align == "f1":
odom_t_align = odom_t_f1
elif scan_area_align == "f2":
odom_t_align = odom_t_f2
else: # ALIGN == "body"
f1_t_body = body_t_f1.inverse()
odom_t_align = odom_t_f1 * f1_t_body
npoints: list[tuple[float, float, float]] = []
for (x, y) in points:
align_t_point = SE3Pose(x, y, f1_pose.z, f1_pose.rot)
odom_t_point = odom_t_align * align_t_point
if len(npoints) != 0:
move_time = math.sqrt(
(npoints[-1][0] - odom_t_point.x)**2 +
(npoints[-1][1] - odom_t_point.y)**2
)
else:
move_time = 1 # number of seconds to move to first position
npoints.append((odom_t_point.x, odom_t_point.y, move_time / scan_velocity))
points = npoints
# get the rotation so that it is aligned with the scan area (which is aligned with the starting body rotation)
align_t_rot = Quat.from_pitch(math.pi / 12)
odom_t_rot = odom_t_align.rot * align_t_rot
# send the points
for (x, y, move_time) in points:
await spot.set_ee_pose(
x, y, odom_t_f1.z + scan_height, # assume that the z is the z of the ground (because the fiducial should be on the ground)
rot=odom_t_rot,
frame_name=ODOM_FRAME_NAME,
move_time=move_time,
body_may_move=True,
)
# stow the arm
await spot.stow_arm(extra_safe=True)
if __name__ == "__main__":
asyncio.run(main())