Skip to content

Commit

Permalink
Moving robot code to new directory, adding linalg logic
Browse files Browse the repository at this point in the history
  • Loading branch information
NathanDuPont committed Apr 15, 2024
1 parent 2a5cd52 commit 4530458
Show file tree
Hide file tree
Showing 25 changed files with 896 additions and 28 deletions.
16 changes: 16 additions & 0 deletions .github/workflows/ci.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -38,3 +38,19 @@ jobs:
echo "$formatted_files"
exit 1
fi
test:
runs-on: ubuntu-latest
steps:
- name: 🚀 Checkout
uses: actions/checkout@v4
- name: 🌿 Setup Bazel
uses: bazel-contrib/[email protected]
with:
bazelisk-cache: true
disk-cache: ${{ github.workflow }}
repository-cache: true
bazelrc: |
build --color=yes
build --show_timestamps
- name: 🧪 Run Tests
run: bazel test //src/...
6 changes: 3 additions & 3 deletions BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -12,15 +12,15 @@ compile_pip_requirements(

# Set up commands to build all images in the project
command(
name = "api_v1_img",
command = "//src/api/v1:tarball",
name = "api_v2_img",
command = "//src/api/v2:tarball",
visibility = ["//visibility:public"],
)

multirun(
name = "build_all_imgs",
commands = [
":api_v1_img",
":api_v2_img",
],
jobs = 0,
)
Expand Down
4 changes: 2 additions & 2 deletions MODULE.bazel.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions requirements.in
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
asyncio
fastapi
jinja2
numpy
opencv-python
uvicorn[standard]
websockets
4 changes: 3 additions & 1 deletion requirements_lock.txt
Original file line number Diff line number Diff line change
Expand Up @@ -177,7 +177,9 @@ numpy==1.26.4 \
--hash=sha256:edd8b5fe47dab091176d21bb6de568acdd906d1887a4584a15a9a96a1dca06ef \
--hash=sha256:f870204a840a60da0b12273ef34f7051e98c3b5961b61b0c2c1be6dfd64fbcd3 \
--hash=sha256:ffa75af20b44f8dba823498024771d5ac50620e6915abac414251bd971b4529f
# via opencv-python
# via
# -r requirements.in
# opencv-python
opencv-python==4.9.0.80 \
--hash=sha256:1a9f0e6267de3a1a1db0c54213d022c7c8b5b9ca4b580e80bdc58516c922c9e1 \
--hash=sha256:3f16f08e02b2a2da44259c7cc712e779eff1dd8b55fdb0323e8cab09548086c0 \
Expand Down
1 change: 1 addition & 0 deletions src/api/assets/BUILD.bazel
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
# TODO implement to make assets publicly available
Binary file added src/api/assets/bloop.mp3
Binary file not shown.
Binary file added src/api/assets/bong.mp3
Binary file not shown.
4 changes: 3 additions & 1 deletion src/api/v1/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,9 @@

motors = [
DynamixelMotor(10, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0)),
DynamixelMotor(11, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0), True),
DynamixelMotor(11, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0)),
DynamixelMotor(12, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0), True),
DynamixelMotor(13, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0), True),
]
ctrl = RobotControl("/dev/ttyUSB0", 1, motors)
ctrl.init(1000000)
Expand Down
51 changes: 41 additions & 10 deletions src/api/v1/robot.py
Original file line number Diff line number Diff line change
Expand Up @@ -120,6 +120,9 @@ def __init__(
self.initialized = False

def __validate_command(self):
"""
Run validation against the motor command
"""
if not self.initialized:
raise Exception("Motor needs to be initialized before commanding")

Expand Down Expand Up @@ -182,10 +185,15 @@ def __read_bytes(self, offset: int, size: int) -> Optional[int]:
def __get_memory_property(self, property: str) -> int:
self.__validate_command()

value = self.__read_bytes(
getattr(self.address_config, property).byte_offset,
getattr(self.address_config, property).byte_size,
)
try:
config: MemorySegment = getattr(self.address_config, property)
except:
# Swallow the missing attribute error and throw a NotImplementedError for the motor instead
raise NotImplementedError(
f"FATAL | DXID: '{self.id}' | Motor does not support {property} attribute!"
)

value = self.__read_bytes(config.byte_offset, config.byte_size)

if value is None:
raise Exception(f"FATAL | DXID: '{self.id}' | Value could not be read!")
Expand All @@ -195,9 +203,17 @@ def __get_memory_property(self, property: str) -> int:
def __set_memory_property(self, property: str, value: int):
self.__validate_command()

try:
config: MemorySegment = getattr(self.address_config, property)
except:
# Swallow the missing attribute error and throw a NotImplementedError for the motor instead
raise NotImplementedError(
f"FATAL | DXID: '{self.id}' | Motor does not support {property} attribute!"
)

self.__write_bytes(
getattr(self.address_config, property).byte_offset,
getattr(self.address_config, property).byte_size,
config.byte_offset,
config.byte_size,
value,
)

Expand Down Expand Up @@ -375,12 +391,27 @@ def set_velocity(self, command: Position2D) -> bool:
math.pi / 2
)

for motor in self.motors:
# rear left
motor_0_speed = min(max(clamped_y - clamped_x - command.rotation.theta, -1), 1)

# front left
motor_1_speed = min(max(clamped_y + clamped_x - command.rotation.theta, -1), 1)

# front right
motor_2_speed = min(max(clamped_y - clamped_x - command.rotation.theta, -1), 1)

# rear right
motor_3_speed = min(max(clamped_y + clamped_x - command.rotation.theta, -1), 1)

speeds = [motor_0_speed, motor_1_speed, motor_2_speed, motor_3_speed]

for i, motor in enumerate(self.motors):
speed = speeds[i]
# TODO actual kinematics here
if clamped_y < 0:
motor.set_moving_speed((abs(clamped_y) * 1023) + 1023)
if speed < 0:
motor.set_moving_speed((abs(speed) * 1023) + 1023)
else:
motor.set_moving_speed(clamped_y * 1023)
motor.set_moving_speed(speed * 1023)

return True

Expand Down
63 changes: 52 additions & 11 deletions src/api/v2/api.py
Original file line number Diff line number Diff line change
@@ -1,27 +1,41 @@
from typing import Union
from fastapi import FastAPI, Request, WebSocket, WebSocketDisconnect
from websockets.exceptions import ConnectionClosed
from fastapi.templating import Jinja2Templates
from fastapi.middleware.cors import CORSMiddleware
from robot import (
Position2D,
RobotControl,
DynamixelMotor,
DYNAMIXEL_MX_12_ADDR_CONFIG,
)
from pygame import mixer
import asyncio
import cv2

app = FastAPI()
camera = cv2.VideoCapture(0, cv2.CAP_DSHOW)
camera = cv2.VideoCapture(0)
templates = Jinja2Templates(directory="templates")
mixer.init()

app.add_middleware(
CORSMiddleware,
allow_origins=["*"],
allow_credentials=True,
allow_methods=["*"], # Allow all HTTP methods
allow_headers=["*"], # Allow all headers
)

@app.get("/")
def read_root():
return {"Hello": "World"}


@app.get("/items/{item_id}")
def read_item(item_id: int, q: Union[str, None] = None):
return {"item_id": item_id, "q": q}
motors = [
DynamixelMotor(10, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0)),
DynamixelMotor(11, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0)),
DynamixelMotor(12, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0), True),
DynamixelMotor(13, DYNAMIXEL_MX_12_ADDR_CONFIG, Position2D(0, 0, 0), True),
]
ctrl = RobotControl("/dev/ttyUSB0", 1, motors)
ctrl.init(1000000)


# https://stackoverflow.com/a/70626324
# Not actually sure if this works with bazel
@app.websocket("/ws")
async def get_stream(websocket: WebSocket):
await websocket.accept()
Expand All @@ -36,3 +50,30 @@ async def get_stream(websocket: WebSocket):
await asyncio.sleep(0.03)
except (WebSocketDisconnect, ConnectionClosed):
print("Client disconnected")


@app.post("/command")
async def command(request: Request):
data = await request.json()

x_val = -data.get("left", 0) + data.get("right", 0)
y_val = -data.get("down", 0) + data.get("up", 0)
rl = data.get("rotateleft", 0)
rr = data.get("rotateright", 0)

ctrl.set_velocity(Position2D(x_val, y_val, rr - rl))

return {"status": "success"}


@app.post("/sfx")
async def command(request: Request):
data = await request.json()

if data.get("sfx") == "bloop":
mixer.music.load("./bloop.mp3")
elif data.get("sfx") == "bong":
mixer.music.load("./bong.mp3")
mixer.music.play()

return {"status": "success"}
18 changes: 18 additions & 0 deletions src/robot/v1/BUILD.bazel
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
load("@rules_python//python:defs.bzl", "py_library")

# Put all robot files in this library
py_library(
name = "robot_src",
srcs = [
"config.py",
"robot.py",
"structs.py",
],
deps = [
"@pypi//fastapi:pkg",
"@pypi//jinja2:pkg",
"@pypi//numpy:pkg",
"@pypi//opencv_python:pkg",
"@pypi//websockets:pkg",
],
)
24 changes: 24 additions & 0 deletions src/robot/v1/config.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
from structs import DynamixelMotorAddressConfig, MemorySegment

# Add all Dynamixel Motor Configs here
DYNAMIXEL_MX_12_ADDR_CONFIG = DynamixelMotorAddressConfig(
torque_enable=MemorySegment(24, 1),
led_enable=MemorySegment(25, 1),
d_gain=MemorySegment(26, 1),
i_gain=MemorySegment(27, 1),
p_gain=MemorySegment(28, 1),
goal_position=MemorySegment(30, 2),
moving_speed=MemorySegment(32, 2),
torque_limit=MemorySegment(34, 2),
present_position=MemorySegment(36, 2),
present_speed=MemorySegment(38, 2),
present_load=MemorySegment(40, 2),
present_input_voltage=MemorySegment(42, 1),
present_temperature=MemorySegment(43, 1),
registered=MemorySegment(44, 1),
moving=MemorySegment(46, 1),
lock=MemorySegment(47, 1),
punch=MemorySegment(48, 2),
realtime_tick=MemorySegment(50, 2),
goal_acceleration=MemorySegment(73, 1),
)
Loading

0 comments on commit 4530458

Please sign in to comment.