# rrt connect algorithm """ This is rrt connect implementation for 3D @author: yue qi """ import numpy as np from numpy.matlib import repmat from collections import defaultdict import time import matplotlib.pyplot as plt import os import sys sys.path.append(os.path.dirname(os.path.abspath(__file__)) + "/../../Sampling-based Planning/") from rrt_3D.env3D import env from rrt_3D.utils3D import getDist, sampleFree, nearest, steer, isCollide, near, visualization, cost, path, edgeset class rrt_connect(): def __init__(self): self.env = env() self.Parent = {} self.V = [] self.E = set() self.i = 0 self.maxiter = 10000 self.stepsize = 0.5 self.Path = [] self.done = False self.qinit = tuple(self.env.start) self.qgoal = tuple(self.env.goal) self.x0, self.xt = tuple(self.env.start), tuple(self.env.goal) self.qnew = None self.ind = 0 self.fig = plt.figure(figsize=(10, 8)) #----------Normal RRT algorithm def BUILD_RRT(self, qinit): tree = Tree(qinit) for k in range(self.maxiter): qrand = self.RANDOM_CONFIG() self.EXTEND(tree, qrand) return tree def EXTEND(self, tree, q): qnear = tuple(self.NEAREST_NEIGHBOR(q, tree)) qnew, dist = steer(self, qnear, q) self.qnew = qnew # store qnew outside if self.NEW_CONFIG(q, qnear, qnew, dist=dist): tree.add_vertex(qnew) tree.add_edge(qnear, qnew) if qnew == q: return 'Reached' else: return 'Advanced' return 'Trapped' def NEAREST_NEIGHBOR(self, q, tree): # find the nearest neighbor in the tree V = np.array(tree.V) if len(V) == 1: return V[0] xr = repmat(q, len(V), 1) dists = np.linalg.norm(xr - V, axis=1) return tuple(tree.V[np.argmin(dists)]) def RANDOM_CONFIG(self): return tuple(sampleFree(self)) def NEW_CONFIG(self, q, qnear, qnew, dist = None): # to check if the new configuration is valid or not by # making a move is used for steer # check in bound collide, _ = isCollide(self, qnear, qnew, dist = dist) return not collide #----------RRT connect algorithm def CONNECT(self, Tree, q): print('in connect') while True: S = self.EXTEND(Tree, q) if S != 'Advanced': break return S def RRT_CONNECT_PLANNER(self, qinit, qgoal): Tree_A = Tree(qinit) Tree_B = Tree(qgoal) for k in range(self.maxiter): print(k) qrand = self.RANDOM_CONFIG() if self.EXTEND(Tree_A, qrand) != 'Trapped': print('trapped') qnew = self.qnew # get qnew from outside if self.CONNECT(Tree_B, qnew) == 'Reached': print('reached') # return self.PATH(Tree_A, Tree_B) return Tree_A, Tree_B = self.SWAP(Tree_A, Tree_B) print('Failure') return 'Failure' # def PATH(self, tree_a, tree_b): def SWAP(self, tree_a, tree_b): tree_a, tree_b = tree_b, tree_a return tree_a, tree_b class Tree(): def __init__(self, node): self.V = [] self.Parent = {} self.V.append(node) self.Parent[node] = None def add_vertex(self, node): if node not in self.V: self.V.append(node) def add_edge(self, parent, child): # here edge is defined a tuple of (parent, child) (qnear, qnew) self.Parent[child] = parent if __name__ == '__main__': p = rrt_connect() starttime = time.time() p.RRT_CONNECT_PLANNER(p.qinit, p.qgoal) print('time used = ' + str(time.time() - starttime))