From 39455da4f08c7e4263cc061c818126ba7680c975 Mon Sep 17 00:00:00 2001 From: cd6 Date: Thu, 14 Feb 2019 15:03:47 +0000 Subject: [PATCH 01/18] Possibly ready to merge --- robot_software/followLineServer.py | 120 +++++++++++++++-------------- 1 file changed, 61 insertions(+), 59 deletions(-) diff --git a/robot_software/followLineServer.py b/robot_software/followLineServer.py index e8a0f03..38ff41f 100644 --- a/robot_software/followLineServer.py +++ b/robot_software/followLineServer.py @@ -3,6 +3,8 @@ import logging from time import sleep, time from threading import Thread +import detectMarking +import control class FollowLine: @@ -13,7 +15,8 @@ class FollowLine: KI = 0 # integral gain lowest DT = 50 # milliseconds - represents change in time since last sensor reading/ - MARKING_NUMBER = 1 # number of consecutive colour readings to detect marking + MARKING_NUMBER = 2 # number of consecutive colour readings to detect marking + MARKING_INTERVAL = 1 # time between marking checks in seconds # Constructor def __init__(self): @@ -23,30 +26,34 @@ def __init__(self): # colour sensors self.csfl = ev3.ColorSensor('in1') # colour sensor front left self.csfr = ev3.ColorSensor('in2') # colour sensor front right - self.csb = ev3.ColorSensor('in3') # colour sensor back + self.csbl = ev3.ColorSensor('in3') # colour sensor back left + self.csbr = ev3.ColorSensor('in4') # colour sensor back right assert self.csfl.connected assert self.csfr.connected - assert self.csb.connected + assert self.csbl.connected + assert self.csbr.connected self.csfl.mode = 'COL-REFLECT' # measure light intensity self.csfr.mode = 'COL-REFLECT' # measure light intensity - self.csb.mode = 'COL-COLOR' # measure colour - + self.csbl.mode = 'COL-COLOR' # measure colour + self.csbr.mode = 'COL-COLOR' # motors self.lm = ev3.LargeMotor('outA') # left motor self.rm = ev3.LargeMotor('outC') # right motor self.cm = ev3.LargeMotor('outD') # centre motor assert self.lm.connected assert self.rm.connected + assert self.cm.connected self.consecutive_colours = 0 # counter for consecutive colour readings + self.number_of_markers = 0 # at which marker it should stop self.runner = None + self.reverse = False def detect_marking(self): - colour = self.csb.value() - print(colour) + colour = self.csbl.value() if colour == 3 or colour == 2: # 3 = green 2 = blue self.consecutive_colours += 1 - print("CONSECUTIVE COLOURS: ", self.consecutive_colours) + #print("CONSECUTIVE COLOURS: ", self.consecutive_colours) if self.consecutive_colours > self.MARKING_NUMBER: return colour else: @@ -63,48 +70,49 @@ def on_line(sensor_value, position): logging.error("onLine: wrong position value for sensor") return False - def correct_trajectory(self, csfl, csfr, lm, rm, number_of_markers): + # limit motor speed to safe values: [-1000, 1000] deg/sec + def limit_speed(self, speed): + if speed > 1000: + return 1000 + if speed < -1000: + return -1000 + return speed + + def correct_trajectory(self, lm, rm, number_of_markers): integral = 0 previous_error = 0 marker_counter = 0 start_time = time() - interval_between_colors = 1 # time between marker checks in seconds while not self.shut_down: - lval = csfl.value() - rval = csfr.value() - error = lval - rval - 10 - logging.info("PID error: ", error) - integral += (error * self.DT) - derivative = (error - previous_error) / self.DT - - # u zero: on target, drive forward - # u positive: too bright, turn right - # u negative: too dark, turn left - # u is torque (See IVR lecture on Control) - u = (self.KP * error) + (self.KI * integral) + (self.KD * derivative) - - # limit u to safe values: [-1000, 1000] deg/sec - # limit u to safe values: [-1000, 1000] deg/sec - if self.MOTOR_SPEED + abs(u) > 1000: # reduce u if speed and torque are too high - if u >= 0: - u = 1000 - self.MOTOR_SPEED - else: - u = self.MOTOR_SPEED - 1000 + if self.reverse: + lval = self.csbr.value() # back right becomes front left + rval = self.csbl.value() + else: + lval = self.csfl.value() + rval = self.csfr.value() + + u, integral, previous_error = control.calculate_torque\ + (self.csfl.value(), self.csfr.value(), self.DT, integral, previous_error) + speed_left = self.limit_speed(self.MOTOR_SPEED + u) + speed_right = self.limit_speed(self.MOTOR_SPEED - u) + # run motors - lm.run_timed(time_sp=self.DT, speed_sp=-(self.MOTOR_SPEED + u)) - rm.run_timed(time_sp=self.DT, speed_sp=-(self.MOTOR_SPEED - u)) + if self.reverse: + lm.run_timed(time_sp=self.DT, speed_sp=self.MOTOR_SPEED - u) + rm.run_timed(time_sp=self.DT, speed_sp=self.MOTOR_SPEED + u) + else: + lm.run_timed(time_sp=self.DT, speed_sp=-(self.MOTOR_SPEED + u)) + rm.run_timed(time_sp=self.DT, speed_sp=-(self.MOTOR_SPEED - u)) sleep(self.DT / 1000) - #print("u {}".format(u)) - #print("lm {}\n".format(lm.speed_sp)) - #print("rm {}".format(rm.speed_sp)) - #print("PID:", lval, rval) - - previous_error = error + # print("u {}".format(u)) + # print("lm {}\n".format(lm.speed_sp)) + # print("rm {}".format(rm.speed_sp)) + # print("PID:", lval, rval) # Check markers - if time() - start_time > interval_between_colors: + if time() - start_time > self.MARKING_INTERVAL: # returns 3 if green, 2 if blue marker_colour = self.detect_marking() if marker_colour == 3: @@ -117,36 +125,30 @@ def correct_trajectory(self, csfl, csfr, lm, rm, number_of_markers): elif marker_colour == 2: # stop on blue marker self.stop() - + def move_sideways(self, cm): while not self.shut_down: cm.run_timed(time_sp=self.DT, speed_sp=300) - def run(self, number_of_markers): - self.correct_trajectory(self.csfl, self.csfr, self.lm, self.rm, number_of_markers) - # while not self.shut_down: - # print(self.csb.value()) + def start(self, number_of_markers): + self.shut_down = False + self.number_of_markers = number_of_markers + if self.number_of_markers == 0: + ev3.Sound.speak("0 number of markers specified").wait() + self.stop() + else: + self.runner = Thread(target=self.run, name='move') + self.runner.start() + + def run(self): + self.correct_trajectory() self.stop() def stop(self): self.shut_down = True self.rm.stop() self.lm.stop() - ev3.Sound.speak("yeet").wait() - - def start(self, number_of_markers): - self.consecutive_colours = 0 - if (self.shut_down): - self.shut_down = False - self.runner = Thread(target=self.run, name='move', args=(number_of_markers,)) - self.runner.start() - else: - self.shut_down = True - self.runner = Thread(target=self.run, name='move', args=(number_of_markers,)) - self.shut_down = False - self.runner.start() - - + ev3.Sound.speak("bruh").wait() # Main function From 110df638915852d1eb53350ecc8ba4168ca614f3 Mon Sep 17 00:00:00 2001 From: cd6 Date: Mon, 25 Mar 2019 21:22:35 +0000 Subject: [PATCH 02/18] Half speed for back wheels --- robot_software/followLine.py | 249 +++++++++++++++++++++++++++++++++++ robot_software/followPath.py | 61 +++++++++ 2 files changed, 310 insertions(+) create mode 100644 robot_software/followLine.py create mode 100644 robot_software/followPath.py diff --git a/robot_software/followLine.py b/robot_software/followLine.py new file mode 100644 index 0000000..13dba56 --- /dev/null +++ b/robot_software/followLine.py @@ -0,0 +1,249 @@ +#! /usr/bin/env python3 +import ev3dev.ev3 as ev3 +from time import sleep, time +import control + + +class FollowLine: + # From https://gist.github.com/CS2098/ecb3a078ed502c6a7d6e8d17dc095b48 + MOTOR_SPEED = 700 + DT = 50 # milliseconds - represents change in time since last sensor reading/ + + + MARKING_NUMBER = 1 # number of consecutive colour readings to detect marking + MARKING_INTERVAL = 1 # time between marking checks in seconds + reverse = False + + BLACK = 1 # black reading from colour sensor in COL-COLOR mode + BLUE = 2 # blue reading from colour sensor in COL-COLOR mode + GREEN = 3 # green reading from colour sensor in COL-COLOR mode + + CORRECTION_TIME = 100 # time in millisecond Bob moves away from blue line to correct sideways movement + SIDEWAYS_SPEED = 700 # how fast Bob moves when moving sideways + + # Constructor + def __init__(self): + self.btn = ev3.Button() + self.shut_down = False + + # colour sensors + self.csfl = ev3.ColorSensor('in1') # colour sensor front left + self.csfr = ev3.ColorSensor('in2') # colour sensor front right + self.csbl = ev3.ColorSensor('in3') # colour sensor back left + self.csbr = ev3.ColorSensor('in4') # colour sensor back right + assert self.csfl.connected + assert self.csfr.connected + assert self.csbl.connected + assert self.csbr.connected + + # motors + self.lm = ev3.LargeMotor('outA') # left motor + self.rm = ev3.LargeMotor('outC') # right motor + self.cm = ev3.LargeMotor('outD') # centre motor + assert self.lm.connected + assert self.rm.connected + assert self.cm.connected + + self.consecutive_colours = 0 # counter for consecutive colour readings + self.ignore_blue = False # when switching from sideways to forwards + self.ignore_green = False # when switching from forwards to sideways + # self.number_of_markers = 0 # at which marker it should stop + + self.start_time = 0 # when robot starts doing a command + self.marker_counter = 0 # how many markers have been passed in current command + + self.reverse = 1 # 1 if Bob is reversing, -1 if not + + def detect_marking(self, colour_left, colour_right, desired_colour): + # print(colour_left, colour_right) + if colour_right == desired_colour and colour_left == desired_colour: + self.consecutive_colours += 1 + print("CONSECUTIVE COLOURS: ", self.consecutive_colours) + if self.consecutive_colours >= self.MARKING_NUMBER: + self.consecutive_colours = 0 + return True + else: + self.consecutive_colours = 0 + return False + + # limit motor speed to safe values: [-1000, 1000] deg/sec + @staticmethod + def limit_speed(speed): + if speed > 1000: + return 1000 + if speed < -1000: + return -1000 + return speed + + # adjust modes of colour sensors depending on the direction of Bob + # COL-REFLECT: measure light intensity + # COL-COLOR: measure colour + def set_cs_modes(self, direction): + if direction == 'forward': + self.csfl.mode = 'COL-REFLECT' + self.csfr.mode = 'COL-REFLECT' + self.csbl.mode = 'COL-COLOR' + self.csbr.mode = 'COL-COLOR' + elif direction == 'backward': + self.csfl.mode = 'COL-COLOR' + self.csfr.mode = 'COL-COLOR' + self.csbl.mode = 'COL-REFLECT' + self.csbr.mode = 'COL-REFLECT' + elif direction == 'left' or direction == 'right': + self.csfl.mode = 'COL-COLOR' + self.csfr.mode = 'COL-COLOR' + self.csbl.mode = 'COL-COLOR' + self.csbr.mode = 'COL-COLOR' + else: + return False # wrong direction command sent + return True + + # increase marker counter when seeing desired colour + def count_markings(self, cs_left, cs_right, desired_colour): + # Wait before checking for colour again + if time() - self.start_time > self.MARKING_INTERVAL: + # marking of desired colour detected + if self.detect_marking(cs_left.value(), cs_right.value(), desired_colour): + self.marker_counter += 1 + ev3.Sound.beep() + self.start_time = time() + return + + # follows a line and corrects trajectory continually + # uses light sensors to follow line and colour sensors to detect markings + def correct_trajectory(self, light_left, light_right, motor_left, motor_right, pid_controller): + # most likely off line, may need to recalibrate numbers later + # time_off_line = self.get_back_on_line(light_left, light_right, time_off_line) + + # Calculate torque using PID control + torque = pid_controller.calculate_torque(light_left.value(), light_right.value()) + # Set the speed of the motors + speed_left = self.limit_speed(self.MOTOR_SPEED + torque) + speed_right = self.limit_speed(self.MOTOR_SPEED - torque) + print('Speed left:', speed_left) + print('Speed right:', speed_right) + + # run motors + motor_left.run_timed(time_sp=self.DT, speed_sp=speed_left * self.reverse) + motor_right.run_timed(time_sp=self.DT, speed_sp=speed_right * self.reverse) + sleep(self.DT / 1000) + + return + + # move forward + def run_forward(self, distance, desired_colour): + self.reverse = -1 + self.start_time = time() + self.marker_counter = 0 + pid_controller = control.Control(self.DT) + + while not self.shut_down: + self.correct_trajectory(self.csfl, self.csfr, self.lm, self.rm, pid_controller) + self.count_markings(self.csbl, self.csbr, desired_colour) + if self.marker_counter >= distance: + return + + # move backward + def run_backward(self, distance, desired_colour): + self.reverse = 1 + self.start_time = time() + self.marker_counter = 0 + pid_controller = control.Control(self.DT) + + while not self.shut_down: + self.correct_trajectory(self.csbr, self.csbl, self.rm, self.lm, pid_controller) + self.count_markings(self.csfr, self.csfl, desired_colour) + if self.marker_counter >= distance: + return + + # move sideways between two blue lines + def run_sideways(self, distance, direction, last_direction): + self.start_time = time() + self.marker_counter = 0 + + while not self.shut_down: + # if a colour sensor is on a blue line, correct position to be between them again + if direction == 'left': + right_speed = self.SIDEWAYS_SPEED/2.0 + left_speed = self.SIDEWAYS_SPEED + else: + right_speed = self.SIDEWAYS_SPEED + left_speed = self.SIDEWAYS_SPEED / 2.0 + if self.detect_marking(self.csbl.value(), self.csbr.value(), self.BLUE): + # back sensors on blue line, so move forward for some time + self.lm.run_timed(time_sp=self.CORRECTION_TIME, speed_sp=-left_speed) + self.rm.run_timed(time_sp=self.CORRECTION_TIME, speed_sp=-right_speed) + sleep(self.CORRECTION_TIME / 1000) + if self.detect_marking(self.csfl.value(), self.csfr.value(), self.BLUE): + # front sensors on blue line, so move backward for some time + + self.lm.run_timed(time_sp=self.CORRECTION_TIME, speed_sp=left_speed) + self.rm.run_timed(time_sp=self.CORRECTION_TIME, speed_sp=right_speed) + sleep(self.CORRECTION_TIME / 1000) + + # colour sensor for marking detection needs to be at front or back dependng on the last direction + if last_direction == 'forward': + cs_left = self.csbr + cs_right = self.csbl + else: + # if last direction is backward, left, or right + cs_left = self.csfr + cs_right = self.csfl + + # move sideways for a bit while counting black markings + if direction == 'left': + self.cm.run_timed(time_sp=self.DT, speed_sp=self.SIDEWAYS_SPEED) + sleep(self.DT / 1000) + self.count_markings(cs_left, cs_left, self.BLACK) + if direction == 'right': + self.cm.run_timed(time_sp=self.DT, speed_sp=-self.SIDEWAYS_SPEED) + sleep(self.DT / 1000) + self.count_markings(cs_right, cs_right, self.BLACK) + + if self.marker_counter >= distance: + return + + # when line is lost oscillate side to side until it is found + def get_back_on_line(self, lval, rval, time_off_line): + if lval > 90 and rval > 70: + if time_off_line == 0: + time_off_line = time() + # if off line for more than a second move side-to-side until line is found + # print(time() - time_off_line) + if time() - time_off_line > 0.5: + correction_speed = 200 + correction_time = 100 + # can change thresholds + while lval > 70 and rval > 50: + self.cm.run_timed(time_sp=correction_time, speed_sp=correction_speed) + correction_speed *= -1 + # increase the time to move in one direction to increased the search radius + correction_time += 100 + sleep(correction_time / 1000) # milliseconds to seconds + lval = self.csfl.value() + rval = self.csfr.value() + time_off_line = 0 + else: + time_off_line = 0 + return time_off_line + + def move_toward_shelf(self): + self.cm.run_timed(time_sp=self.DT*50, speed_sp=-self.SIDEWAYS_SPEED) + sleep(self.DT*50.0/1000.0) + return + + def move_away_from_shelf(self): + + self.cm.run_timed(time_sp=1000, speed_sp=self.SIDEWAYS_SPEED) + sleep(1) + # if self.detect_marking(self.csbl.value(), self.csbl.value(), self.BLACK): + return + + def stop_shelf_movement(self): + self.cm.stop(stop_action='hold') + + def stop(self): + self.shut_down = True + self.rm.stop() + self.lm.stop() + ev3.Sound.speak("whack").wait() diff --git a/robot_software/followPath.py b/robot_software/followPath.py new file mode 100644 index 0000000..a87f9d5 --- /dev/null +++ b/robot_software/followPath.py @@ -0,0 +1,61 @@ +#! /usr/bin/env python3 +import ev3dev.ev3 as ev3 +from followLine import FollowLine +from threading import Thread + + +class FollowPath: + # Follow a path given by the server + BLACK = 1 # black reading from colour sensor in COL-COLOR mode + BLUE = 2 # blue reading from colour sensor in COL-COLOR mode + GREEN = 3 # green reading from colour sensor in COL-COLOR mode + + # Constructor + def __init__(self): + self.shut_down = False + self.runner = None + self.last_direction = '' # saves previous direction Bob moved in + + def go(self, path): + print(path) + line_follower = FollowLine() + for direction, distance in path: + if line_follower.set_cs_modes(direction): + # modes set successfully + # direction move in forwards axis or side axis + if direction == 'forward': + line_follower.run_forward(distance, self.GREEN) + elif direction == 'backward': + line_follower.run_backward(distance, self.GREEN) + elif direction == 'left' or direction == 'right': + if self.last_direction == 'forward': # Bob has to move forward to blue line + line_follower.set_cs_modes('forward') + line_follower.run_forward(1, self.BLUE) + if self.last_direction == 'backward': # Bob has to move backward to blue line + line_follower.set_cs_modes('backward') + line_follower.run_backward(1, self.BLUE) + line_follower.set_cs_modes(direction) + line_follower.run_sideways(distance, direction, self.last_direction) + self.last_direction = direction + else: + # not a valid direction for colour sensors + if direction == 'G': + ev3.Sound.speak("Scoopdidoop").wait() + else: + ev3.Sound.speak("Wrong command given. What does", direction, "mean?").wait() + line_follower.stop() + + + # TODO: possibly move start and stop to FollowPath or move correct trajectory to a separate file instead + def start(self, path): + self.shut_down = False + print(path) + if len(path) == 0: + ev3.Sound.speak("No instructions given").wait() + else: + self.go(path) + +if __name__ == "__main__": + path_follower = FollowPath() + current_path = [('right', 1), ('forward', 1)] + path_follower.start(current_path) From 7d3f4aa1b6f6f4457f1fa5167cd455e568861add Mon Sep 17 00:00:00 2001 From: cd6 Date: Wed, 27 Mar 2019 15:02:53 +0000 Subject: [PATCH 03/18] Code for moving motors on pi but not working --- robot_software/lift.py | 41 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 41 insertions(+) create mode 100644 robot_software/lift.py diff --git a/robot_software/lift.py b/robot_software/lift.py new file mode 100644 index 0000000..9bb8369 --- /dev/null +++ b/robot_software/lift.py @@ -0,0 +1,41 @@ +#! /usr/bin/env python3 +import time +from iotools import IOTools +import sys + +class Lift: + MC_SPEED = 100 + LIFT_TIME = 5 + # use ports 2 and 3 + mc_id_a = 2 + mc_id_b = 3 + + def __init__(self, onRobot): + IO = IOTools(True) + self.mc = IO.motor_control + self.mc.stopMotors() + + def lift(self, direction): + print('lifting {}'.format(direction)) + if direction == 'up': + self.mc.setMotor(self.mc_id_a, -self.MC_SPEED) + self.mc.setMotor(self.mc_id_b, -self.MC_SPEED) + time.sleep(self.LIFT_TIME) + self.mc.stopMotors() + time.sleep(1) + elif direction == 'down': + self.mc.setMotor(self.mc_id_a, self.MC_SPEED) + self.mc.setMotor(self.mc_id_b, self.MC_SPEED) + time.sleep(self.LIFT_TIME/2.0) + self.mc.stopMotors() + time.sleep(1) + + +if __name__ == "__main__": + onRobot = bool(sys.argv.count('-rss')) + lifter = Lift(onRobot) + lifter.lift('up') + time.sleep(lifter.LIFT_TIME+1) + lifter.lift('down') + + From af49bc40b6d9c4f5a8eda5029bb2e5b5dd917d7f Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Thu, 28 Mar 2019 12:59:14 +0000 Subject: [PATCH 04/18] Added lifting to rasp pi scripts --- robot_software/bobTranslation.pyc | Bin 0 -> 564 bytes robot_software/ev3_listener.py | 30 +++++++++++++++------------ robot_software/followPath.pyc | Bin 0 -> 2065 bytes robot_software/rasppi_coordinator.py | 13 +++++++----- robot_software/rasppi_listener.py | 13 ++++++++---- 5 files changed, 34 insertions(+), 22 deletions(-) create mode 100644 robot_software/bobTranslation.pyc create mode 100644 robot_software/followPath.pyc diff --git a/robot_software/bobTranslation.pyc b/robot_software/bobTranslation.pyc new file mode 100644 index 0000000000000000000000000000000000000000..ef4239b34718642ccbb7b3469cc99b76cc783cbf GIT binary patch literal 564 zcmb_YOHRWu5S^r@KUx-SP&h$Wr0fww0tpr7M3&65Mxn9}hm=Zfl|XG;>`}&RBPthusC9|xceoD9Lh1PegEF^_2jx7W?fE{@C>lqH6#Eo;>UWs!5~N<-!v9V(pu0~R NBcE$^8b 1023) s= socket.socket(socket.AF_INET, socket.SOCK_STREAM) @@ -33,16 +35,18 @@ #MoveIn() if str_instruction == 'move_in': - - line_follower = FollowLine() - line_follower.move_toward_shelf() + if sys.argv[0] == 1: + line_follower = FollowLine() + line_follower.move_toward_shelf() elif str_instruction == 'move_out': print('inb') - line_follower = FollowLine() - line_follower.move_away_from_shelf() + if sys.argv[0] == 1: + line_follower = FollowLine() + line_follower.move_away_from_shelf() elif str_instruction == 'stop_shelf': - line_follower = FollowLine() - line_follower.stop() + if sys.argv[0] == 1: + line_follower = FollowLine() + line_follower.stop() #stop moving in diff --git a/robot_software/followPath.pyc b/robot_software/followPath.pyc new file mode 100644 index 0000000000000000000000000000000000000000..6b8fcc8cb1ad001b538ca625982c20b0f861a371 GIT binary patch literal 2065 zcmcIl+m0JW5Url^c;n438ztl>Bw~2-16q?H9*_{1y&_myXqK?T3L4GMwC%RXJU7nqp78JO&4+*d{vxB*FU0#RG}8;G!3aAlH%!*iUv(1>PP4|C~s08(XT{TL_hV>QFPkbfGP72e!Y%@ z>Ng>d%AzP2C&ub}tckH7MwL$U*QUQu{b-Ghx9*|2Pa!m|0QCd*0W^6)efSUAIlur5 zE7x%P8x)^**ok2~CSbv5&A7{rNA}U&69`YVBueDStiZgI5qKb+h!^)fpKDN7i?qr+ z5$|Gglon3=7Bn}S`y?+H);G}Wl~#KOP|dB?m463nkvgm}mCn2=Z5;?+>vS>b;7uaj z0krOOrz_VTR65s{$tLINBG#WvS@+8wBHu=rW*RepDDZ__?QE1aukFWrLOh zb7L9Om7?lH1qFec1XRT`70?oEz#l9KNWIKtMIbA@tibgDtO$99`;84Fk!%k;VFr1v z&HuUbj4nJYh5;f(;;SjXtfY{4)b9Se-Vv;KAesN&@!E03=!U=x#co7bAUz;4BO=0K zUfvgkO+k395Z`f&#+#IHu#fR=jvFt!nYbkzT5RcN`i$Xk$zov_*GDf7AthtyX*OYw z35^Op@WQjo3`btb?3upD{x;-kR+h8ef3p# zo#R|jb!xxtY(tCD$ZtV_z)9wkX_;#$cEQzw$^usgu<-4HD&qNd z0ZFuS<9uo}{Wj`{Yh+m7gLEH3kcv<__tZ|fr5-45kJOIZ#=DlHVeWCzuIr2UbK6IA z-$JbKnYz$WF2d&($v-26Y}^pq3v0)qHoV$x;Ul#F7e literal 0 HcmV?d00001 diff --git a/robot_software/rasppi_coordinator.py b/robot_software/rasppi_coordinator.py index 074ae5a..092f27b 100644 --- a/robot_software/rasppi_coordinator.py +++ b/robot_software/rasppi_coordinator.py @@ -51,8 +51,11 @@ def listen_to_server(self,username): headers={'username':'merchant_01'} url = 'http://{}:{}/api/warehouse/5c755f58bfcf4c592bfd00a6/orders/{}'.format(self.server_info['ip'],self.server_info['port'],path['job']['id']) update = requests.post(url,headers=headers,json={'status':'READY_TO_COLLECT'}) - print(update.text) - print('done - notified') + while not(json.loads(update.text)['success']): + + time.sleep(5) + update = requests.post(url,headers=headers,json={'status':'READY_TO_COLLECT'}) + print('notified') self.retry_timeout = 1 sleep(5) except KeyboardInterrupt: @@ -68,7 +71,7 @@ def job_handler(self,instruction_set): for instruction in (instruction_set): command = instruction['command'] res = None - if command == "lift" or command == "drop": + if command == "lift": res = self.reliable_send_data(self.rasp_target,str(instruction)) elif command == "grab": res = self.reliable_grab() @@ -145,5 +148,5 @@ def open_and_send(self, target,payload): -#rjr = RobotJobListener(('192.168.105.38',9000),('192.168.105.38',65432),('192.168.105.38',65433)) -#rjr.start_reliable_listener('robot') +rjr = RobotJobListener(('192.168.105.38',9000),('192.168.105.38',65432),('192.168.105.38',65433)) +rjr.start_reliable_listener('robot') diff --git a/robot_software/rasppi_listener.py b/robot_software/rasppi_listener.py index 2671fd5..4d037cc 100644 --- a/robot_software/rasppi_listener.py +++ b/robot_software/rasppi_listener.py @@ -15,15 +15,20 @@ def listen(): print('Connected by', addr) data = conn.recv(1024) - if data == b'grab': + data = data.decode("utf-8") + print(data) + + if data[0] == 'grab': print('grab') - elif data == b'prepare': + elif data[0] == 'prepare': print('prepare_grabber') - elif data == b'wait_for_bump': + elif data[0] == 'wait_for_bump': print('wait for bump') raw_input() print("BUMP!") - + elif data[0] == 'lift': + #trigger lift + print('lifting {}'.format(data[1])) conn.sendall(b'done') conn.close() listen() \ No newline at end of file From bd366bd37e32b7e3005114369312cca640a3adb1 Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Thu, 28 Mar 2019 13:03:56 +0000 Subject: [PATCH 05/18] Updated Fake_DB with new shelves --- server/fake_db.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/server/fake_db.json b/server/fake_db.json index 23a4ebb..f7a5257 100755 --- a/server/fake_db.json +++ b/server/fake_db.json @@ -15,7 +15,7 @@ "name": "Appleton Tower Superstore", "image": "", "location": { "latitude": 55.944592, "longitude": -3.18706 }, - "dimensions": { "x": 5, "y": 6, "z": [0, 0.2, 0.4] }, + "dimensions": { "x": 5, "y": 6, "z": [0, 1] }, "items": [], "merchantId": "5c755f2cbfcf4c592bfd00a5" } From 52a0bcf9ef1e51900e16883e7b958e8687e33b97 Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Thu, 28 Mar 2019 14:05:57 +0000 Subject: [PATCH 06/18] Lifting added to path finding and rasp pi script --- .vscode/settings.json | 2 +- robot_software/ev3_listener.py | 12 +----------- robot_software/rasppi_coordinator.py | 16 +++++++++------- robot_software/rasppi_listener.py | 11 +++++------ server/fake_db.json | 2 +- server/robot-pathfinding.js | 13 +++++++------ 6 files changed, 24 insertions(+), 32 deletions(-) diff --git a/.vscode/settings.json b/.vscode/settings.json index 615aafb..bd53818 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -1,3 +1,3 @@ { - "python.pythonPath": "/usr/bin/python3" + "python.pythonPath": "/Library/Frameworks/Python.framework/Versions/3.6/bin/python3" } \ No newline at end of file diff --git a/robot_software/ev3_listener.py b/robot_software/ev3_listener.py index 5f42e76..ff171f0 100644 --- a/robot_software/ev3_listener.py +++ b/robot_software/ev3_listener.py @@ -31,8 +31,7 @@ str_instruction = str_instruction.replace('\'', '\"') str_instruction = str_instruction.replace('u\"', '\"') print(str_instruction) - #TODO: fix follow path so that it can move in or make a new function to handle it. - #MoveIn() + if str_instruction == 'move_in': if sys.argv[0] == 1: @@ -47,15 +46,6 @@ if sys.argv[0] == 1: line_follower = FollowLine() line_follower.stop() - #stop moving in - - - #else: - # robot = FollowPath() - # robot.start([extract(json.loads(str_instruction))]) - - - print('done') conn.sendall(b'done') conn.close() diff --git a/robot_software/rasppi_coordinator.py b/robot_software/rasppi_coordinator.py index 092f27b..f2ba1be 100644 --- a/robot_software/rasppi_coordinator.py +++ b/robot_software/rasppi_coordinator.py @@ -45,14 +45,13 @@ def listen_to_server(self,username): path = json.loads(r.text) print(path) if path['success'] != False: - print(path) + if path['job'] != []: self.job_handler(path['job']['instruction_set']) headers={'username':'merchant_01'} url = 'http://{}:{}/api/warehouse/5c755f58bfcf4c592bfd00a6/orders/{}'.format(self.server_info['ip'],self.server_info['port'],path['job']['id']) update = requests.post(url,headers=headers,json={'status':'READY_TO_COLLECT'}) while not(json.loads(update.text)['success']): - time.sleep(5) update = requests.post(url,headers=headers,json={'status':'READY_TO_COLLECT'}) print('notified') @@ -68,17 +67,18 @@ def listen_to_server(self,username): def job_handler(self,instruction_set): # TODO, open this on a new thread i = 0 + for instruction in (instruction_set): + print(instruction) command = instruction['command'] res = None - if command == "lift": - res = self.reliable_send_data(self.rasp_target,str(instruction)) - elif command == "grab": - res = self.reliable_grab() + + if command == "grab": + res = self.reliable_grab(instruction["parameters"]['height']) else: res = self.reliable_send_data(self.ev3_target,str(instruction)) - def reliable_grab(self): + def reliable_grab(self,height): try: global thread_manager thread_manager['bumped'] = False @@ -91,7 +91,9 @@ def reliable_grab(self): print('bump!') thread_manager['bumped'] = True #self.reliable_send_data(self.ev3_target,"stop_shelf") + self.reliable_send_data(self.rasp_target,"lift {}".format(height)) self.reliable_send_data(self.rasp_target,"grab") + self.reliable_send_data(self.rasp_target,"lift 0") self.reliable_send_data(self.ev3_target,"move_out") return except KeyboardInterrupt: diff --git a/robot_software/rasppi_listener.py b/robot_software/rasppi_listener.py index 4d037cc..fd46b39 100644 --- a/robot_software/rasppi_listener.py +++ b/robot_software/rasppi_listener.py @@ -1,5 +1,5 @@ import socket - +import json def listen(): PORT = 65432 # Port to listen on (non-privileged ports are > 1023) @@ -15,9 +15,8 @@ def listen(): print('Connected by', addr) data = conn.recv(1024) - data = data.decode("utf-8") - print(data) - + data = data.decode('utf-8') + data = data.split(' ') if data[0] == 'grab': print('grab') elif data[0] == 'prepare': @@ -27,8 +26,8 @@ def listen(): raw_input() print("BUMP!") elif data[0] == 'lift': - #trigger lift - print('lifting {}'.format(data[1])) + print('lift to {}'.format(data[1])) + conn.sendall(b'done') conn.close() listen() \ No newline at end of file diff --git a/server/fake_db.json b/server/fake_db.json index f7a5257..59c508c 100755 --- a/server/fake_db.json +++ b/server/fake_db.json @@ -79,7 +79,7 @@ "_id": "5c7565eea29bee5b5b8a5426", "name": "Irn Bru (330ml can)", "image": "", - "position": { "x": 3, "y": 3, "z": 0 }, + "position": { "x": 3, "y": 3, "z": 1 }, "quantity": 1, "unit": null, "price": 0.65, diff --git a/server/robot-pathfinding.js b/server/robot-pathfinding.js index a2efb62..a7e2f8e 100755 --- a/server/robot-pathfinding.js +++ b/server/robot-pathfinding.js @@ -40,11 +40,12 @@ generate_drop_instruction = () => { command: 'drop' } } -generate_grab_instruction = () => { +generate_grab_instruction = (height) => { return { - command: 'grab', - parameters: {} - } + command: 'grab', + parameters: {'height':height} + } + } generate_lift_instruction = (start, end) => { @@ -98,7 +99,7 @@ convert_order_to_job = (order, robot, warehouse_grid) => { var robot_xy = [robot_pos['x'], robot_pos['y'], robot_pos['z']] for (var i = 0; i < item_list.length; i++) { var current_item = item_list[i]['position'] - var item_xy = [current_item['x'], current_item['y'], robot_pos['z']] + var item_xy = [current_item['x'], current_item['y'], current_item['z']] var path = pathfind_to_point(robot_xy, item_xy, warehouse_grid) if (path == [] || path == undefined) { @@ -111,7 +112,7 @@ convert_order_to_job = (order, robot, warehouse_grid) => { if (robot_xy[2] != item_xy[2]) { job['instruction_set'].push(generate_lift_instruction(robot_xy[2], item_xy[2])) } - job['instruction_set'].push(generate_grab_instruction()) + job['instruction_set'].push(generate_grab_instruction(item_xy[2])) // return home for (var p = path.length - 1; p > 0; p--) { job['instruction_set'].push(generate_movement_instruction(path[p], path[p - 1])) From c6ae349b2ea9b835101d4ea81121240294525e4a Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Thu, 28 Mar 2019 15:08:18 +0000 Subject: [PATCH 07/18] Changes to EV3 and Pi once onboard --- robot_software/ev3_listener.py | 84 ++++++++++++++-------------- robot_software/followLine.py | 2 +- robot_software/followPath.py | 7 ++- robot_software/rasppi_coordinator.py | 8 +-- robot_software/rasppi_listener.py | 1 - robot_software/toddler.py | 11 ++-- 6 files changed, 59 insertions(+), 54 deletions(-) diff --git a/robot_software/ev3_listener.py b/robot_software/ev3_listener.py index ff171f0..f3a0e36 100644 --- a/robot_software/ev3_listener.py +++ b/robot_software/ev3_listener.py @@ -1,51 +1,53 @@ #! /usr/bin/env python3 import sys import socket -if sys.argv[0] == 1: - from followPath import FollowPath - from bobTranslation import extract - from followLine import FollowLine - import ev3dev.ev3 as ev3 +from followPath import FollowPath +from bobTranslation import extract +from followLine import FollowLine +import ev3dev.ev3 as ev3 import json # Get local machine name +class EV3Listener: + def __init__(self): + self.path_follower = FollowPath() + def get_instructions(self): + PORT = 65433 # Port to listen on (non-privileged ports are > 1023) -PORT = 65433 # Port to listen on (non-privileged ports are > 1023) + s= socket.socket(socket.AF_INET, socket.SOCK_STREAM) + HOST = socket.gethostbyname(socket.gethostname()) + s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) + s.bind((HOST, PORT)) + print("Listening on {}:{}".format(HOST,PORT)) + #ev3.Sound.tone([(1000, 250, 0),(1500, 250, 0),(2000, 250, 0)]).wait() + while True: + s.listen(2) + conn, addr = s.accept() + print('Connected by', addr) -s= socket.socket(socket.AF_INET, socket.SOCK_STREAM) -HOST = socket.gethostbyname(socket.gethostname()) -s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) -s.bind((HOST, PORT)) -print("Listening on {}:{}".format(HOST,PORT)) -#ev3.Sound.tone([(1000, 250, 0),(1500, 250, 0),(2000, 250, 0)]).wait() -while True: - s.listen(2) - conn, addr = s.accept() - print('Connected by', addr) + data = conn.recv(1024) + if data: - data = conn.recv(1024) - if data: - - str_instruction = data.decode('utf-8') - str_instruction = str_instruction.replace('\'', '\"') - str_instruction = str_instruction.replace('u\"', '\"') - print(str_instruction) - - - if str_instruction == 'move_in': - if sys.argv[0] == 1: - line_follower = FollowLine() - line_follower.move_toward_shelf() - elif str_instruction == 'move_out': - print('inb') - if sys.argv[0] == 1: - line_follower = FollowLine() - line_follower.move_away_from_shelf() - elif str_instruction == 'stop_shelf': - if sys.argv[0] == 1: - line_follower = FollowLine() - line_follower.stop() - print('done') - conn.sendall(b'done') - conn.close() + str_instruction = data.decode('utf-8') + str_instruction = str_instruction.replace('\'', '\"') + str_instruction = str_instruction.replace('u\"', '\"') + print(str_instruction) + + if str_instruction == 'move_in': + self.path_follower.go('in') + elif str_instruction == 'move_out': + self.path_follower.go('out') + elif str_instruction == 'stop_shelf': + self.path_follower.go('stop') + else: + try: + movement = json.loads(str_instruction) + self.path_follower.go(extract(movement)) + except: + continue + print('done') + conn.sendall(b'done') + conn.close() +ev3 = EV3Listener() +ev3.get_instructions() \ No newline at end of file diff --git a/robot_software/followLine.py b/robot_software/followLine.py index 9adf6a1..84106cd 100644 --- a/robot_software/followLine.py +++ b/robot_software/followLine.py @@ -19,7 +19,7 @@ class FollowLine: GREEN = 3 # green reading from colour sensor in COL-COLOR mode CORRECTION_TIME = 100 # time in millisecond Bob moves away from blue line to correct sideways movement - SIDEWAYS_SPEED = 800 # how fast Bob moves when moving sideways + SIDEWAYS_SPEED = 400 # how fast Bob moves when moving sideways # Constructor def __init__(self): diff --git a/robot_software/followPath.py b/robot_software/followPath.py index d76ca85..d98747d 100644 --- a/robot_software/followPath.py +++ b/robot_software/followPath.py @@ -39,8 +39,13 @@ def go(self, path): self.last_direction = direction else: # not a valid direction for colour sensors - if direction == 'G': + if direction == 'in': ev3.Sound.speak("Scoopdidoop").wait() + line_follower.move_toward_shelf() + elif direction == 'out': + line_follower.move_toward_shelf() + elif direction == 'stop': + line_follower.stop_shelf_movement() else: ev3.Sound.speak("Wrong command given. What does", direction, "mean?").wait() line_follower.stop() diff --git a/robot_software/rasppi_coordinator.py b/robot_software/rasppi_coordinator.py index f2ba1be..e01df4e 100644 --- a/robot_software/rasppi_coordinator.py +++ b/robot_software/rasppi_coordinator.py @@ -8,7 +8,7 @@ from threading import Thread from time import sleep from bobTranslation import extract - +from rasppi_listener import listen thread_manager = {'bumped':False} class RobotJobListener(): @@ -90,7 +90,7 @@ def reliable_grab(self,height): self.reliable_send_data(self.rasp_target,"wait_for_bump") print('bump!') thread_manager['bumped'] = True - #self.reliable_send_data(self.ev3_target,"stop_shelf") + self.reliable_send_data(self.ev3_target,"stop_shelf") self.reliable_send_data(self.rasp_target,"lift {}".format(height)) self.reliable_send_data(self.rasp_target,"grab") self.reliable_send_data(self.rasp_target,"lift 0") @@ -148,7 +148,3 @@ def open_and_send(self, target,payload): print('error') return -1 - - -rjr = RobotJobListener(('192.168.105.38',9000),('192.168.105.38',65432),('192.168.105.38',65433)) -rjr.start_reliable_listener('robot') diff --git a/robot_software/rasppi_listener.py b/robot_software/rasppi_listener.py index fd46b39..375ce11 100644 --- a/robot_software/rasppi_listener.py +++ b/robot_software/rasppi_listener.py @@ -30,4 +30,3 @@ def listen(): conn.sendall(b'done') conn.close() -listen() \ No newline at end of file diff --git a/robot_software/toddler.py b/robot_software/toddler.py index 4ab9791..9d6b692 100644 --- a/robot_software/toddler.py +++ b/robot_software/toddler.py @@ -9,7 +9,6 @@ from grabber import Grabber from rasppi_coordinator import RobotJobListener - class Logger(object): def __init__(self, onRobot): self.useTerminal = not onRobot @@ -65,14 +64,18 @@ def listen(self): print('Connected by', addr) data = conn.recv(1024) - if data == b'grab': + data = data.decode('utf-8') + data = data.split(' ') + if data == 'grab': self.grabber.grab() - elif data == b'prepare': + elif data == 'prepare': self.grabber.prepare_grabber() - elif data == b'wait_for_bump': + elif data == 'wait_for_bump': while(self.getInputs()[0] == 0 or self.getInputs()[1] == 0): print("Wait for bump") print("bump") + elif data[0] == 'lift': + print('lift to {}'.format(data[1])) conn.sendall(b'done') conn.close() except KeyboardInterrupt: From e78aca684cd644ce4dcdce31003c75d3db479f98 Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Fri, 29 Mar 2019 11:35:34 +0000 Subject: [PATCH 08/18] Change Ev3 to have single FollowLine object --- robot_software/ev3_listener.py | 2 +- robot_software/followLine.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/robot_software/ev3_listener.py b/robot_software/ev3_listener.py index f3a0e36..34a0eb1 100644 --- a/robot_software/ev3_listener.py +++ b/robot_software/ev3_listener.py @@ -43,7 +43,7 @@ def get_instructions(self): else: try: movement = json.loads(str_instruction) - self.path_follower.go(extract(movement)) + self.path_follower.go(extract(movement)) except: continue print('done') diff --git a/robot_software/followLine.py b/robot_software/followLine.py index c37ac55..06a97d8 100644 --- a/robot_software/followLine.py +++ b/robot_software/followLine.py @@ -19,7 +19,7 @@ class FollowLine: GREEN = 3 # green reading from colour sensor in COL-COLOR mode CORRECTION_TIME = 100 # time in millisecond Bob moves away from blue line to correct sideways movement - SIDEWAYS_SPEED = 400 # how fast Bob moves when moving sideways + SIDEWAYS_SPEED = 600 # how fast Bob moves when moving sideways # Constructor def __init__(self): From f210a6830e2932e14c3b2fb3e62ca65d85ed1059 Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Fri, 29 Mar 2019 15:10:24 +0000 Subject: [PATCH 09/18] Updated Schema --- SCHEMA.md | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/SCHEMA.md b/SCHEMA.md index ba9d491..b03603c 100644 --- a/SCHEMA.md +++ b/SCHEMA.md @@ -31,7 +31,8 @@ Item { "position": Position, "quantity": Double, "unit": String (or null if there is no unit), - "price": Double // In GBP + "price": Double, // In GBP + "size": "tiny", "small", "large" // Tiny - 3 of these can fit in bob, Small - 2 can fit, Large - 1 can fit } Position { From dea5b27ae4f023042c50b352cbcc75259beea246 Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Fri, 29 Mar 2019 15:11:33 +0000 Subject: [PATCH 10/18] Updated Fake DB --- server/fake_db.json | 36 +++++++++++++++++++++++------------- 1 file changed, 23 insertions(+), 13 deletions(-) diff --git a/server/fake_db.json b/server/fake_db.json index 59c508c..eedb947 100755 --- a/server/fake_db.json +++ b/server/fake_db.json @@ -33,7 +33,8 @@ "quantity": 1, "unit": null, "price": 0.25, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"small" }, { "_id": "5c7565e1a29bee5b5b8a5422", @@ -43,7 +44,8 @@ "quantity": 1, "unit": null, "price": 0.7, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"small" }, { "_id": "5c7565e4a29bee5b5b8a5423", @@ -53,7 +55,8 @@ "quantity": 1, "unit": null, "price": 2.5, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"large" }, { "_id": "5c7565e6a29bee5b5b8a5424", @@ -63,7 +66,8 @@ "quantity": 1, "unit": "kg", "price": 0.8, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"small" }, { "_id": "5c7565eca29bee5b5b8a5425", @@ -73,27 +77,30 @@ "quantity": 1, "unit": null, "price": 2.3, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"large" }, { "_id": "5c7565eea29bee5b5b8a5426", "name": "Irn Bru (330ml can)", "image": "", "position": { "x": 3, "y": 3, "z": 1 }, - "quantity": 1, + "quantity": 5, "unit": null, "price": 0.65, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"small" }, { "_id": "5c7565f1a29bee5b5b8a5427", "name": "Apple", "image": "", - "position": { "x": 1, "y": 5, "z": 0 }, - "quantity": 1, + "position": { "x": 1, "y": 5, "z": 1 }, + "quantity": 5, "unit": "kg", "price": 1.35, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"small" }, { "_id": "5c7565f7a29bee5b5b8a5428", @@ -103,7 +110,8 @@ "quantity": 1, "unit": null, "price": 1.59, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"large" }, { "_id": "5c7567e034f58d5beeb55559", @@ -117,7 +125,8 @@ "quantity": 1, "unit": "kg", "price": 1.1, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"small" }, { "_id": "5c88f9626b55930017ca2ffa", @@ -131,7 +140,8 @@ "quantity": 1, "unit": null, "price": 0.6, - "warehouseId": "5c755f58bfcf4c592bfd00a6" + "warehouseId": "5c755f58bfcf4c592bfd00a6", + "size":"small" } ], "bob_movement": [{ "_id": "5c73f048ffb612103e1288a8", "moving": false, "markers": 1 }], From 36702c51193ac1cb86f5b22d0ecba5bc9dc06e3a Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Fri, 29 Mar 2019 15:11:39 +0000 Subject: [PATCH 11/18] Updated Pathfinding --- server/robot-pathfinding.js | 90 ++++++++++++++++++++++++++++--------- 1 file changed, 68 insertions(+), 22 deletions(-) diff --git a/server/robot-pathfinding.js b/server/robot-pathfinding.js index a7e2f8e..b115ca0 100755 --- a/server/robot-pathfinding.js +++ b/server/robot-pathfinding.js @@ -45,14 +45,14 @@ generate_grab_instruction = (height) => { command: 'grab', parameters: {'height':height} } - + x } generate_lift_instruction = (start, end) => { return { command: 'lift', parameters: { - height: Math.abs(start - end) + height: end - start } } } @@ -85,41 +85,87 @@ generate_movement_instruction = (start, end) => { } } } -convert_order_to_job = (order, robot, warehouse_grid) => { +package_items_into_groups = (item_list) => { + var sorted_items = {"tiny":[],"small":[],"large":[]} + for(var i = 0; i < item_list.length; i++) { + var item = item_list[i] + for (var j = 0; j < item['quantity']; j++) { + sorted_items[item['size']].push(item) + } + } + var item_groups = [] + item_groups.push(sorted_items.large) + for (var i = 0; i < sorted_items.small.length; i+=2) { + if (i + 1 < sorted_items.small.length) { + console.log("here") + item_groups.push([sorted_items.small[i],sorted_items.small[i+1]]) + } else { + if (sorted_items.tiny.length >= 1) { + item_groups.push([sorted_items.tiny.pop(),sorted_items.small[i]]) + } else { + + item_groups.push([sorted_items.small[i]]) + } + } + } + for (var i = 0; i < sorted_items.tiny.length; i+=3) { + to_push = [sorted_items.tiny[i]] + if (i + 1 < sorted_items.tiny.length) { + to_push.push(sorted_items.tiny[i+1]) + } + if (i + 2 < sorted_items.tiny.length) { + to_push.push(sorted_items.tiny[i+2]) + } + item_groups.push(to_push) + } + console.log(item_groups) + return item_groups +} + convert_order_to_job = (order, robot, warehouse_grid) => { if (order == undefined) { return {} } var robot_pos = robot['location'] - var item_list = order['items'] - + + const item_list = order['items'] + const item_groups = package_items_into_groups(item_list) var job = { id: order['_id'], instruction_set: [] } var robot_xy = [robot_pos['x'], robot_pos['y'], robot_pos['z']] - for (var i = 0; i < item_list.length; i++) { - var current_item = item_list[i]['position'] - var item_xy = [current_item['x'], current_item['y'], current_item['z']] - - var path = pathfind_to_point(robot_xy, item_xy, warehouse_grid) - if (path == [] || path == undefined) { - return {} + const robot_home_xy = [robot['home_x'],robot['home_y']] + // collect item groups + for (var i = 0; i < item_groups.length; i++) { + var current_group = item_groups[i] + var path = [] + for (var j = 0; j < current_group.length; j++) { + var current_item = current_group[j]['position'] + var item_xy = [current_item['x'], current_item['y'], current_item['z']] + + path = pathfind_to_point(robot_xy, item_xy, warehouse_grid) + if (path == [] || path == undefined) { + return {} + } + //go to position and grab + for (var p = 1; p < path.length; p++) { + job['instruction_set'].push(generate_movement_instruction(path[p - 1], path[p])) + } + + + job['instruction_set'].push(generate_grab_instruction(item_xy[2])) + + robot_xy = [item_xy[0],item_xy[1],0] } - //go to position and grab + // return home + path = pathfind_to_point(robot_xy, robot_home_xy, warehouse_grid) for (var p = 1; p < path.length; p++) { job['instruction_set'].push(generate_movement_instruction(path[p - 1], path[p])) } - if (robot_xy[2] != item_xy[2]) { - job['instruction_set'].push(generate_lift_instruction(robot_xy[2], item_xy[2])) - } - job['instruction_set'].push(generate_grab_instruction(item_xy[2])) - // return home - for (var p = path.length - 1; p > 0; p--) { - job['instruction_set'].push(generate_movement_instruction(path[p], path[p - 1])) - } job['instruction_set'].push(generate_drop_instruction()) + robot_xy = [robot_home_xy[0],robot_home_xy[1],0] } - + console.log(job) return job } module.exports.get_robot_path = (order, robot, warehouse) => { From d9d3fd4906c9d9e169ae1d46ab3e2d69ed05a558 Mon Sep 17 00:00:00 2001 From: Oktay Sen Date: Fri, 29 Mar 2019 15:29:45 +0000 Subject: [PATCH 12/18] Added "size" to Item in Android. --- .../main/java/io/github/assis10t/bobandroid/AddToCartDialog.kt | 3 ++- .../src/main/java/io/github/assis10t/bobandroid/pojo/Item.kt | 3 ++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/android/app/src/main/java/io/github/assis10t/bobandroid/AddToCartDialog.kt b/android/app/src/main/java/io/github/assis10t/bobandroid/AddToCartDialog.kt index 4bb4083..2daa0d3 100644 --- a/android/app/src/main/java/io/github/assis10t/bobandroid/AddToCartDialog.kt +++ b/android/app/src/main/java/io/github/assis10t/bobandroid/AddToCartDialog.kt @@ -56,7 +56,8 @@ class AddToCartDialog(val activity: WarehouseActivity, val item: Item): Dialog(a item.position, Integer.parseInt(quantity.text.toString()).toDouble(), item.unit, - item.price + item.price, + item.size ) addToCart(context, cartItem) activity.refreshItems() diff --git a/android/app/src/main/java/io/github/assis10t/bobandroid/pojo/Item.kt b/android/app/src/main/java/io/github/assis10t/bobandroid/pojo/Item.kt index 565e6cf..cf52959 100644 --- a/android/app/src/main/java/io/github/assis10t/bobandroid/pojo/Item.kt +++ b/android/app/src/main/java/io/github/assis10t/bobandroid/pojo/Item.kt @@ -10,7 +10,8 @@ class Item ( val position: Position? = null, val quantity: Double? = null, val unit: String? = null, - val price: Double = 0.0 + val price: Double = 0.0, + val size: String? = null ) { companion object { fun fromString(str: String) = Gson().fromJson(str, Item::class.java) From 65b1eba7aba8855201d81f64cb1f4e3fd041b331 Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Fri, 29 Mar 2019 16:21:55 +0000 Subject: [PATCH 13/18] Further Changes --- server/fake_db.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/server/fake_db.json b/server/fake_db.json index eedb947..90adb04 100755 --- a/server/fake_db.json +++ b/server/fake_db.json @@ -100,7 +100,7 @@ "unit": "kg", "price": 1.35, "warehouseId": "5c755f58bfcf4c592bfd00a6", - "size":"small" + "size":"tiny" }, { "_id": "5c7565f7a29bee5b5b8a5428", From d656df40da3f028f885e5fef78710a919b086233 Mon Sep 17 00:00:00 2001 From: Frederick Bawden Date: Fri, 29 Mar 2019 16:28:17 +0000 Subject: [PATCH 14/18] Bug Fixing and clearing log statements --- server/fake_db.json | 4 ++-- server/robot-pathfinding.js | 13 +++++-------- 2 files changed, 7 insertions(+), 10 deletions(-) diff --git a/server/fake_db.json b/server/fake_db.json index 90adb04..c126750 100755 --- a/server/fake_db.json +++ b/server/fake_db.json @@ -63,7 +63,7 @@ "name": "Carrot", "image": "", "position": { "x": 1, "y": 3, "z": 0 }, - "quantity": 1, + "quantity": 5, "unit": "kg", "price": 0.8, "warehouseId": "5c755f58bfcf4c592bfd00a6", @@ -100,7 +100,7 @@ "unit": "kg", "price": 1.35, "warehouseId": "5c755f58bfcf4c592bfd00a6", - "size":"tiny" + "size":"small" }, { "_id": "5c7565f7a29bee5b5b8a5428", diff --git a/server/robot-pathfinding.js b/server/robot-pathfinding.js index b115ca0..4a4ca38 100755 --- a/server/robot-pathfinding.js +++ b/server/robot-pathfinding.js @@ -20,7 +20,6 @@ makeWarehouse = (width, height) => { walkable_grid[s * 2][i] = 1 } } - console.log(walkable_grid) return walkable_grid } @@ -30,9 +29,7 @@ pathfind_to_point = (current_pos, end_pos, warehouse_grid) => { var path = finder.findPath(current_pos[0], current_pos[1], end_pos[0], end_pos[1], pf_grid) path = PF.Util.compressPath(path) - console.log(current_pos) - console.log(end_pos) - console.log(path) + return path } generate_drop_instruction = () => { @@ -94,10 +91,12 @@ package_items_into_groups = (item_list) => { } } var item_groups = [] - item_groups.push(sorted_items.large) + if (item_groups.length > 0) { + item_groups.push(sorted_items.large) + } + for (var i = 0; i < sorted_items.small.length; i+=2) { if (i + 1 < sorted_items.small.length) { - console.log("here") item_groups.push([sorted_items.small[i],sorted_items.small[i+1]]) } else { if (sorted_items.tiny.length >= 1) { @@ -118,7 +117,6 @@ package_items_into_groups = (item_list) => { } item_groups.push(to_push) } - console.log(item_groups) return item_groups } convert_order_to_job = (order, robot, warehouse_grid) => { @@ -165,7 +163,6 @@ package_items_into_groups = (item_list) => { job['instruction_set'].push(generate_drop_instruction()) robot_xy = [robot_home_xy[0],robot_home_xy[1],0] } - console.log(job) return job } module.exports.get_robot_path = (order, robot, warehouse) => { From 6059f489f3ae9263120e80f1cd7a8de29746becf Mon Sep 17 00:00:00 2001 From: Hristiyan Yaprakov Date: Tue, 2 Apr 2019 11:54:53 +0100 Subject: [PATCH 15/18] Add and edit size of item. --- website/assets/sass/style.sass | 5 +++- website/pages/merchant/items/create.vue | 25 +++++++++++++++++++- website/pages/merchant/items/edit/_id.vue | 18 ++++++++++++++ website/pages/merchant/items/index.vue | 6 ++++- website/pages/merchant/warehouses/_id.vue | 6 ++++- website/pages/merchant/warehouses/create.vue | 2 +- 6 files changed, 57 insertions(+), 5 deletions(-) diff --git a/website/assets/sass/style.sass b/website/assets/sass/style.sass index b415b05..25efd1a 100644 --- a/website/assets/sass/style.sass +++ b/website/assets/sass/style.sass @@ -1148,7 +1148,7 @@ footer height: auto display: block padding: 1.2rem 1.5rem - line-height: 1 + line-height: 1.3 font-size: $size-7 &.is-checkbox @@ -1160,6 +1160,9 @@ footer font-size: $size-7 font-weight: normal + i + font-size: inherit + .oh overflow: hidden diff --git a/website/pages/merchant/items/create.vue b/website/pages/merchant/items/create.vue index 830456d..74736ee 100644 --- a/website/pages/merchant/items/create.vue +++ b/website/pages/merchant/items/create.vue @@ -30,6 +30,9 @@
+

+ X and Y specify the row and column respectively and Z specifies which shelf the product will be on. +

+ + + + + +
+
+
@@ -134,6 +155,7 @@ export default { }, quantity: null, unit: null, + size: 'tiny', price: null } }, @@ -177,6 +199,7 @@ export default { position: this.position, quantity: this.quantity, unit: this.unit, + size: this.size, price: this.price, }, { headers: { diff --git a/website/pages/merchant/items/edit/_id.vue b/website/pages/merchant/items/edit/_id.vue index b20ba7c..5696fd7 100644 --- a/website/pages/merchant/items/edit/_id.vue +++ b/website/pages/merchant/items/edit/_id.vue @@ -84,6 +84,24 @@
+
+
+ +

+ Tiny - 3 of these can fit in the robots basket, Small - 2 can fit, Large - 1 can fit. +

+ +
+
diff --git a/website/pages/merchant/items/index.vue b/website/pages/merchant/items/index.vue index ce36ecd..966ab6b 100644 --- a/website/pages/merchant/items/index.vue +++ b/website/pages/merchant/items/index.vue @@ -43,6 +43,7 @@ Quantity Unit Price + Size Edit items Delete items @@ -64,7 +65,10 @@ {{ item.quantity }} - {{ item.unit }} + {{ item.unit ? item.unit : '-' }} + + + {{ item.size }} {{ item.price }} GBP diff --git a/website/pages/merchant/warehouses/_id.vue b/website/pages/merchant/warehouses/_id.vue index bd91d3f..7610897 100644 --- a/website/pages/merchant/warehouses/_id.vue +++ b/website/pages/merchant/warehouses/_id.vue @@ -117,6 +117,7 @@ Position Quantity Unit + Size Price Edit items Delete items @@ -139,7 +140,10 @@ {{ item.quantity }} - {{ item.unit }} + {{ item.unit ? item.unit : '-' }} + + + {{ item.size }} {{ item.price }} GBP diff --git a/website/pages/merchant/warehouses/create.vue b/website/pages/merchant/warehouses/create.vue index eb004e7..f47a324 100644 --- a/website/pages/merchant/warehouses/create.vue +++ b/website/pages/merchant/warehouses/create.vue @@ -38,7 +38,7 @@

- Enter the dimensions of the warehouse. X and Y values are the amount of rows and columns that the warehouse phisically has for the robot. + Enter the dimensions of the warehouse. X and Y values are the amount of rows and columns that the warehouse physically has for the robot.

From 6eec09037247d0415257a599dab4b7993e3c04f7 Mon Sep 17 00:00:00 2001 From: Hristiyan Yaprakov Date: Tue, 2 Apr 2019 14:57:59 +0100 Subject: [PATCH 16/18] Localise MDI --- website/nuxt.config.js | 7 ++++--- website/package-lock.json | 18 ++++++++++++++++++ website/package.json | 2 ++ 3 files changed, 24 insertions(+), 3 deletions(-) diff --git a/website/nuxt.config.js b/website/nuxt.config.js index 3704d6a..89d1c34 100644 --- a/website/nuxt.config.js +++ b/website/nuxt.config.js @@ -16,8 +16,8 @@ module.exports = { ], link: [ { rel: 'icon', type: 'image/x-icon', href: '/favicon.ico' }, - { rel: 'stylesheet', type: 'text/css', href: '//cdn.materialdesignicons.com/2.0.46/css/materialdesignicons.min.css' }, - { rel: 'stylesheet', type: 'text/css', href: 'https://fonts.googleapis.com/css?family=Raleway&Open+Sans' } + // { rel: 'stylesheet', type: 'text/css', href: '//cdn.materialdesignicons.com/2.0.46/css/materialdesignicons.min.css' }, + // { rel: 'stylesheet', type: 'text/css', href: 'https://fonts.googleapis.com/css?family=Raleway&Open+Sans' } ] }, @@ -34,7 +34,8 @@ module.exports = { ** Global CSS */ css: [ - '@/assets/sass/style.sass' + '@/assets/sass/style.sass', + '@/node_modules/@mdi/font/css/materialdesignicons.min.css', ], /* diff --git a/website/package-lock.json b/website/package-lock.json index da0c44d..f72e2c0 100644 --- a/website/package-lock.json +++ b/website/package-lock.json @@ -974,6 +974,11 @@ "resolved": "https://registry.npmjs.org/@firebase/webchannel-wrapper/-/webchannel-wrapper-0.2.12.tgz", "integrity": "sha512-1QWhGGu03NjPYk45w5muLq6a43Fhvd5+LGZrPydXLs7/FgQHqW6aSKmJClTwBHP1rynLSLLA7+9E/35cur6g/g==" }, + "@mdi/font": { + "version": "3.5.95", + "resolved": "https://registry.npmjs.org/@mdi/font/-/font-3.5.95.tgz", + "integrity": "sha512-WHSJ0TJ70qkn+EPsW9w22pQU+kjEnRZlfN4N7xsFFmKa6VhpdQcwTWqj9PDH3oq6Be2p0IW/VDURJvPWDnBAUw==" + }, "@nuxt/babel-preset-app": { "version": "2.4.2", "resolved": "https://registry.npmjs.org/@nuxt/babel-preset-app/-/babel-preset-app-2.4.2.tgz", @@ -7402,6 +7407,11 @@ "object-visit": "^1.0.0" } }, + "material-design-icons": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/material-design-icons/-/material-design-icons-3.0.1.tgz", + "integrity": "sha1-mnHEh0chjrylHlGmbaaCA4zct78=" + }, "md5.js": { "version": "1.3.5", "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", @@ -8024,6 +8034,14 @@ "@nuxt/webpack": "2.4.2" } }, + "nuxt-material-design-icons": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/nuxt-material-design-icons/-/nuxt-material-design-icons-1.0.4.tgz", + "integrity": "sha512-1DwI3n7zLIMO+dV87JUSohhF6yFtIXyzBZ15YLQYSDEQWFBpo/3k1excO1dooaaGVJFS4ObnOCreiy4Xov4t/A==", + "requires": { + "material-design-icons": "^3.0.1" + } + }, "nwsapi": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.1.0.tgz", diff --git a/website/package.json b/website/package.json index 670c670..2a266fc 100644 --- a/website/package.json +++ b/website/package.json @@ -12,6 +12,7 @@ "test": "jest" }, "dependencies": { + "@mdi/font": "^3.5.95", "@nuxtjs/axios": "^5.3.6", "@nuxtjs/bulma": "^1.2.1", "axios": "^0.18.0", @@ -20,6 +21,7 @@ "firebase": "^5.8.2", "node-sass": "^4.11.0", "nuxt": "^2.3.4", + "nuxt-material-design-icons": "^1.0.4", "sass-loader": "^7.1.0", "vue-cookies": "^1.5.12", "vue-headroom": "^0.9.0", From 9c342368073227bb14f511c34dbd0bab91c5483f Mon Sep 17 00:00:00 2001 From: Hristiyan Yaprakov Date: Thu, 4 Apr 2019 13:21:27 +0100 Subject: [PATCH 17/18] User testing fixes --- website/pages/merchant/items/create.vue | 23 ++++++----- website/pages/merchant/items/edit/_id.vue | 2 +- website/pages/merchant/warehouses/create.vue | 42 +++++++++++++------- 3 files changed, 42 insertions(+), 25 deletions(-) diff --git a/website/pages/merchant/items/create.vue b/website/pages/merchant/items/create.vue index 74736ee..1f494f9 100644 --- a/website/pages/merchant/items/create.vue +++ b/website/pages/merchant/items/create.vue @@ -16,7 +16,7 @@
- +
@@ -43,7 +43,7 @@
@@ -56,7 +56,7 @@
@@ -69,7 +69,7 @@
@@ -78,12 +78,15 @@
- +
+

+ kg, lbs, litres, etc... or blank. +

@@ -107,7 +110,7 @@
- +
@@ -146,17 +149,17 @@ export default { warehouseId: this.$nuxt._route.params.id, warehouse: {}, - name: null, + name: "", image: null, position: { x: 0, y: 0, z: 0, }, - quantity: null, + quantity: "", unit: null, size: 'tiny', - price: null + price: "" } }, methods: { @@ -222,7 +225,7 @@ export default { }, computed: { can_submit: function () { - return this.name && this.quantity && this.price + return this.name.toString().length > 0 && this.quantity.toString().length > 0 && this.price.toString().length > 0 } }, mounted: function () { diff --git a/website/pages/merchant/items/edit/_id.vue b/website/pages/merchant/items/edit/_id.vue index 5696fd7..8688a1f 100644 --- a/website/pages/merchant/items/edit/_id.vue +++ b/website/pages/merchant/items/edit/_id.vue @@ -3,7 +3,7 @@

- Add item + Edit item

diff --git a/website/pages/merchant/warehouses/create.vue b/website/pages/merchant/warehouses/create.vue index f47a324..30916c1 100644 --- a/website/pages/merchant/warehouses/create.vue +++ b/website/pages/merchant/warehouses/create.vue @@ -12,7 +12,7 @@
- +
@@ -42,22 +42,33 @@

- +
- +

The Z values are the amount of shelfs and their respected heights from the robot's perspective. (Include bottom shelf as 0.0)

- + class="is-flex align-center mb10"> + + + + + + +
Add a shelf @@ -95,15 +106,15 @@ export default { }, data: function () { return { - name: null, + name: "", image: null, location: { - latitude: null, - longitude: null + latitude: "", + longitude: "" }, dimensions: { - x: null, - y: null, + x: "", + y: "", z: [ 0.0 ] @@ -114,6 +125,9 @@ export default { addShelf: function () { this.dimensions.z.push(this.dimensions.z[this.dimensions.z.length - 1]) }, + deleteShelf: function (i) { + this.dimensions.z.splice(i, 1) + }, uploadFile: function (event) { let file = event.target.files[0] this.createImage(file); @@ -158,7 +172,7 @@ export default { }, computed: { can_submit: function () { - return this.name && this.location.latitude && this.location.longitude && this.dimensions.x && this.dimensions.y + return this.name.toString().length > 0 && this.dimensions.x.toString().length > 0 && this.dimensions.y.toString().length > 0 } } }; From 3301f232029024a797c10fd9dc87459bb5a6b762 Mon Sep 17 00:00:00 2001 From: Oktay Sen Date: Thu, 4 Apr 2019 13:41:42 +0100 Subject: [PATCH 18/18] Removed file that was supposed to not exist. --- robot_software/rasppi_listener.py | 32 ------------------------------- 1 file changed, 32 deletions(-) delete mode 100644 robot_software/rasppi_listener.py diff --git a/robot_software/rasppi_listener.py b/robot_software/rasppi_listener.py deleted file mode 100644 index 375ce11..0000000 --- a/robot_software/rasppi_listener.py +++ /dev/null @@ -1,32 +0,0 @@ -import socket -import json - -def listen(): - PORT = 65432 # Port to listen on (non-privileged ports are > 1023) - - s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) - HOST = socket.gethostbyname(socket.gethostname()) - s.bind((HOST, PORT)) - print("Listening on {}:{}".format(HOST,PORT)) - while True: - s.listen(1) - conn, addr = s.accept() - - print('Connected by', addr) - - data = conn.recv(1024) - data = data.decode('utf-8') - data = data.split(' ') - if data[0] == 'grab': - print('grab') - elif data[0] == 'prepare': - print('prepare_grabber') - elif data[0] == 'wait_for_bump': - print('wait for bump') - raw_input() - print("BUMP!") - elif data[0] == 'lift': - print('lift to {}'.format(data[1])) - - conn.sendall(b'done') - conn.close()