From 670fc798408691dfaf2985af2756cdc06c008620 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 14 Jan 2024 17:09:00 +0100 Subject: [PATCH 01/44] move sim code to robot.py --- furuta/robot.py | 96 ++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 95 insertions(+), 1 deletion(-) diff --git a/furuta/robot.py b/furuta/robot.py index eaa1b2d..e17fad8 100644 --- a/furuta/robot.py +++ b/furuta/robot.py @@ -9,7 +9,6 @@ def __init__(self, device="dev/ttyACM0", baudrate=921600): self.ser = serial.Serial(device, baudrate) def step(self, motor_command: float): - """Motor command is a float between -1 and 1.""" direction = motor_command < 0 # convert motor command to 16 bit unsigned int int_motor_command = int(np.abs(motor_command) * (2**16 - 1)) @@ -30,3 +29,98 @@ def reset_encoders(self): def close(self): self.step(0) self.ser.close() + + +class QubeDynamics: + """Solve equation M qdd + C(q, qd) = tau for qdd.""" + + def __init__( + self, + g=9.81, + Rm=8.4, + V=12.0, + km=0.042, + Mr=0.095, + Lr=0.085, + Dr=5e-6, + Mp=0.024, + Lp=0.129, + Dp=1e-6, + ): + # Gravity + self.g = g + + # Motor + self.Rm = Rm + self.V = V + + # back-emf constant (V-s/rad) + self.km = km + + # Rotary arm + self.Mr = Mr + self.Lr = Lr + self.Dr = Dr + + # Pendulum link + self.Mp = Mp + self.Lp = Lp + self.Dp = Dp + + # Init constants + self._init_const() + + def _init_const(self): + # Moments of inertia + Jr = self.Mr * self.Lr**2 / 12 # inertia about COM (kg-m^2) + Jp = self.Mp * self.Lp**2 / 12 # inertia about COM (kg-m^2) + + # Constants for equations of motion + self._c = np.zeros(5) + self._c[0] = Jr + self.Mp * self.Lr**2 + self._c[1] = 0.25 * self.Mp * self.Lp**2 + self._c[2] = 0.5 * self.Mp * self.Lp * self.Lr + self._c[3] = Jp + self._c[1] + self._c[4] = 0.5 * self.Mp * self.Lp * self.g + + @property + def params(self): + params = self.__dict__.copy() + params.pop("_c") + return params + + @params.setter + def params(self, params): + self.__dict__.update(params) + self._init_const() + + def __call__(self, state, action): + # """ + # action between 0 and 1, maps to +V and -V + # """ + th, al, thd, ald = state + voltage = action * self.V + + # Precompute some values + sin_al = np.sin(al) + sin_2al = np.sin(2 * al) + cos_al = np.cos(al) + + # Define mass matrix M = [[a, b], [b, c]] + a = self._c[0] + self._c[1] * sin_al**2 + b = self._c[2] * cos_al + c = self._c[3] + d = a * c - b * b + + # Calculate vector [x, y] = tau - C(q, qd) + trq = self.km * (voltage - self.km * thd) / self.Rm + c0 = self._c[1] * sin_2al * thd * ald - self._c[2] * sin_al * ald * ald + c1 = -0.5 * self._c[1] * sin_2al * thd * thd + self._c[4] * sin_al + x = trq - self.Dr * thd - c0 + y = -self.Dp * ald - c1 + + # Compute M^{-1} @ [x, y] + thdd = (c * x - b * y) / d + aldd = (a * y - b * x) / d + + return thdd, aldd From 6a7b3b25f3058aa9093bfa87ae32b3fc506b3a28 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 14 Jan 2024 17:45:34 +0100 Subject: [PATCH 02/44] use the new robot interface and simplify reset --- furuta/rl/envs/furuta_real.py | 114 ++++++++++------------------------ 1 file changed, 34 insertions(+), 80 deletions(-) diff --git a/furuta/rl/envs/furuta_real.py b/furuta/rl/envs/furuta_real.py index f4e9af7..b8210ac 100644 --- a/furuta/rl/envs/furuta_real.py +++ b/furuta/rl/envs/furuta_real.py @@ -1,109 +1,64 @@ -import configparser import logging from time import sleep +from typing import Optional import numpy as np -from furuta.common import VelocityFilter -from furuta.envs.furuta_base import FurutaBase -from furuta.envs.hardware.robot import FurutaRobot +from furuta.rl.envs.furuta_base import FurutaBase +from furuta.robot import Robot +from furuta.utils import ALPHA, ALPHA_DOT, THETA, THETA_DOT, VelocityFilter + +MAX_RESET_TIME = 5 # seconds class FurutaReal(FurutaBase): def __init__( self, - fs=100, - fs_ctrl=100, - action_limiter=False, - safety_th_lim=1.5, - reward="simple", - state_limits="low", - config_file="robot.ini", + control_freq=100, + reward="alpha_theta", + state_limits=None, + usb_device="/dev/ttyACM0", ): - super().__init__(fs, fs_ctrl, action_limiter, safety_th_lim, reward, state_limits) + super().__init__(control_freq, reward, state_limits) - self.robot = FurutaRobot() + self.robot = Robot(usb_device) self.vel_filt = VelocityFilter(2, dt=self.timing.dt) - self._state = self._read_state() + self._update_state(0.0) def _update_state(self, action): - self.robot.set_motor_voltage(action[0]) - - state = self._read_state() - - return state - - def _read_state(self): - # pendulum_deg_per_count = 360/self.pendulum_CPR - # p_count = self.pendulum_enc.readCounter() - # # p_count_modulo = p_count % self.pendulum_CPR - # pendulum_angle = pendulum_deg_per_count * p_count - # # pendulum_angle = (pendulum_angle + 180) % 360 - # pendulum_angle = pendulum_angle * np.pi / 180 - - # motor_deg_per_count = 360/self.motor_CPR - # m_count = self.motor_enc.readCounter() - # m_count_modulo = m_count % self.motor_CPR - # motor_angle = m_count_modulo * motor_deg_per_count - # motor_angle = (motor_angle + 180) % 360 - 180 - # motor_angle = motor_angle * np.pi / 180 - - motor_angle, pendulum_angle = self.robot.read_state() + motor_angle, pendulum_angle = self.robot.step(action) # motor_angle: theta, pendulum angle: alpha pos = np.array([motor_angle, pendulum_angle], dtype=np.float32) - vel = self.vel_filt(pos) # TODO understand unit: rad/s ? + vel = self.vel_filt(pos) state = np.concatenate([pos, vel]) + self._state = state - return state - - def get_state(self): - return self._state - - def _reset_pendulum(self, tolerance=10, still_time=1, clear=True): - pass - - def reset(self): + def reset( + self, + seed: Optional[int] = None, + options: Optional[dict] = None, + ): logging.info("Reset env...") - # reset pendulum - self._reset_pendulum(40, 0.5, False) - # reset motor - logging.debug("Reset motor") - while True: - state = self._read_state() * 180 / np.pi - motor_angle = state[0] - motor_speed = state[2] - - speed = 0.3 - if abs(motor_angle) < 10: - - if abs(motor_speed) > 10: - # braking - if motor_angle <= 0: - self.motor.set_speed(-0.3) - elif motor_angle > 0: - self.motor.set_speed(0.3) - - sleep(10 / 100) - - self.motor.set_speed(0) - self.motor.set_speed(0) + # wait for pendulum to fall back to start position + reset_time = 0 + while np.abs(self._state[ALPHA_DOT]) > 0.1 and reset_time < MAX_RESET_TIME: + sleep(0.01) + self._update_state(0.0) + reset_time += 0.01 - break - elif motor_angle >= 0: - self.motor.set_speed(-speed) - elif motor_angle < 0: - self.motor.set_speed(speed) + if reset_time >= MAX_RESET_TIME: + logging.error("Reset timeout") - # wait for pendulum to reset to start position - self._reset_pendulum(10, 1, True) + # reset both encoder, motor back to pos=0 + self.robot.reset_encoders() logging.info("Reset done") - self._state = self._read_state() - return self.get_obs() + self._update_state(0.0) + return self.get_obs(), {} # TODO: override parent render function # replace by taking webcam snapshot @@ -113,5 +68,4 @@ def reset(self): # raise NotImplementedError def close(self): - # self.motor.close() - self.robot.arduino.close() + self.robot.close() From 57289491506ed4602425edb02b3a6ee21f020e75 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 14 Jan 2024 17:46:15 +0100 Subject: [PATCH 03/44] swap rendered for the o.g cartpole renderer --- furuta/rl/envs/furuta_base.py | 290 ++++++++++++++-------------------- 1 file changed, 118 insertions(+), 172 deletions(-) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index bdd4246..05f9e26 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -1,18 +1,18 @@ from collections import namedtuple from dataclasses import dataclass -from math import cos, sin +from typing import Optional -import gym +import gymnasium as gym import numpy as np -from gym.spaces import Box +from gymnasium.spaces import Box from furuta.utils import ALPHA, ALPHA_DOT, THETA, THETA_DOT, Timing def alpha_reward(state): try: - rwd = (1 + -cos(state[ALPHA])) / 2 - except Exception as e: + rwd = (1 + -np.cos(state[ALPHA])) / 2 + except Exception as e: # TODO why would this fail? print(e) print(state) @@ -20,7 +20,7 @@ def alpha_reward(state): def alpha_theta_reward(state): - return alpha_reward(state) + (1 + cos(state[THETA])) / 2 + return alpha_reward(state) + (1 + np.cos(state[THETA])) / 2 REWARDS = { @@ -30,16 +30,25 @@ def alpha_theta_reward(state): class FurutaBase(gym.Env): - metadata = {"render.modes": ["rgb_array"]} # TODO add headless mode + metadata = { + "render_modes": ["rgb_array", "human"], + "render_fps": 50, # TODO should this be the same as the control freq? + } # TODO add headless mode? - def __init__(self, fs, reward, state_limits=None): + def __init__(self, control_freq, reward, state_limits=None, render_mode="rgb_array"): + self.render_mode = render_mode - self.timing = Timing(fs) - self.viewer = None + self.timing = Timing(control_freq) self._state = None self.reward = reward - self._reward_func = REWARDS[self.reward.name] + self.screen_width = 600 + self.screen_height = 400 + self.screen = None + self.clock = None + self.isopen = True + + self._reward_func = REWARDS[self.reward] act_max = np.array([1.0], dtype=np.float32) @@ -54,27 +63,26 @@ def __init__(self, fs, reward, state_limits=None): # Spaces self.state_space = Box( - # labels=('theta', 'alpha', 'theta_dot', 'alpha_dot'), + # ('theta', 'alpha', 'theta_dot', 'alpha_dot'), low=-self.state_max, high=self.state_max, dtype=np.float32, ) self.observation_space = Box( - # labels=('cos_th', 'sin_th', 'cos_al', 'sin_al', 'th_d', 'al_d'), + # ('cos_th', 'sin_th', 'cos_al', 'sin_al', 'th_d', 'al_d'), low=-obs_max, high=obs_max, dtype=np.float32, ) self.action_space = Box( - # labels=('action',), + # ('action',), low=-act_max, high=act_max, dtype=np.float32, ) - @profile def step(self, action): # TODO this is slow, do we even need it? # sb3 knows the env action space, probably it won't pass invalid actions @@ -92,6 +100,7 @@ def step(self, action): rwd = self._reward_func(self._state) obs = self.get_obs() + # we use this info dict to log the state in mcap files info = { "motor_angle": float(self._state[THETA]), "pendulum_angle": float(self._state[ALPHA]), @@ -101,16 +110,18 @@ def step(self, action): "action": float(action), } - # then take action + # then take action/step the sim self._update_state(action[0]) - done = not self.state_space.contains(self._state) - if done: - rwd -= self.reward.oob_penalty + terminated = not self.state_space.contains(self._state) + + # if terminated: + # rwd -= self.reward.oob_penalty - info["done"] = bool(done) + info["terminated"] = bool(terminated) + truncated = False - return obs, rwd, done, info + return obs, rwd, terminated, truncated, info def get_obs(self): return np.float32( @@ -125,174 +136,109 @@ def get_obs(self): ] ) - def reset(self): + def reset( + self, + seed: Optional[int] = None, + options: Optional[dict] = None, + ): raise NotImplementedError def _update_state(self, a): raise NotImplementedError - def render(self, mode="rgb_array"): - if self.viewer is None: - self.viewer = CartPoleSwingUpViewer(world_width=(2 * np.pi)) + def render(self): + if self.render_mode is None: + assert self.spec is not None + gym.logger.warn( + "You are calling render method without specifying any render mode. " + "You can specify the render_mode at initialization, " + f'e.g. gym.make("{self.spec.id}", render_mode="rgb_array")' + ) + return + + import pygame + from pygame import gfxdraw + + if self.screen is None: + pygame.init() + if self.render_mode == "human": + pygame.display.init() + self.screen = pygame.display.set_mode((self.screen_width, self.screen_height)) + else: # mode == "rgb_array" + self.screen = pygame.Surface((self.screen_width, self.screen_height)) + if self.clock is None: + self.clock = pygame.time.Clock() + + world_width = 2 * np.pi + scale = self.screen_width / world_width + polewidth = 10.0 + polelen = scale * (2 * 0.5) # TODO use the right pole len? + cartwidth = 50.0 + cartheight = 30.0 if self._state is None: return None - self.viewer.update(self._state) - return self.viewer.render(return_rgb_array=mode == "rgb_array") - - -@dataclass(frozen=True) -class CartParams: - """Parameters defining the Cart.""" - - width: float = 1 / 3 - height: float = 1 / 6 - mass: float = 0.5 - - -@dataclass(frozen=True) -class PoleParams: - """Parameters defining the Pole.""" - - width: float = 0.05 - length: float = 0.6 - mass: float = 0.5 - - -Screen = namedtuple("Screen", "width height") - - -class CartPoleSwingUpViewer: - """Class that encapsulates all the variables and objectecs needed to render a - CartPoleSwingUpEnv. - - It handles all the initialization and updating of each object on screen and handles calls to - the underlying gym.envs.classic_control.rendering.Viewer instance. - """ + x = self._state - screen = Screen(width=600, height=400) + self.surf = pygame.Surface((self.screen_width, self.screen_height)) + self.surf.fill((255, 255, 255)) - def __init__(self, world_width): - # TODO: make sure that's not redundant - import pyglet - from gym.envs.classic_control import rendering + l, r, t, b = -cartwidth / 2, cartwidth / 2, cartheight / 2, -cartheight / 2 + axleoffset = cartheight / 4.0 + cartx = x[THETA] * scale + self.screen_width / 2.0 # MIDDLE OF CART + carty = 100 # TOP OF CART + cart_coords = [(l, b), (l, t), (r, t), (r, b)] + cart_coords = [(c[0] + cartx, c[1] + carty) for c in cart_coords] + gfxdraw.aapolygon(self.surf, cart_coords, (0, 0, 0)) + gfxdraw.filled_polygon(self.surf, cart_coords, (0, 0, 0)) - pyglet.options["headless"] = True # noqa: F821 - - cart = CartParams() - pole = PoleParams() - - self.cart = cart - self.pole = pole - - self.world_width = world_width - screen = self.screen - scale = screen.width / self.world_width - cartwidth, cartheight = scale * cart.width, scale * cart.height - polewidth, polelength = scale * pole.width, scale * pole.length - self.viewer = rendering.Viewer(screen.width, screen.height) - self.transforms = { - "cart": rendering.Transform(), - "pole": rendering.Transform(translation=(0, 0)), - "pole_bob": rendering.Transform(), - "wheel_l": rendering.Transform(translation=(-cartwidth / 2, -cartheight / 2)), - "wheel_r": rendering.Transform(translation=(cartwidth / 2, -cartheight / 2)), - } - - self._init_track(rendering, cartheight) - self._init_cart(rendering, cartwidth, cartheight) - self._init_wheels(rendering, cartheight) - self._init_pole(rendering, polewidth, polelength) - self._init_axle(rendering, polewidth) - # Make another circle on the top of the pole - self._init_pole_bob(rendering, polewidth) - - def _init_track(self, rendering, cartheight): - screen = self.screen - carty = screen.height / 2 - track_height = carty - cartheight / 2 - cartheight / 4 - track = rendering.Line((0, track_height), (screen.width, track_height)) - track.set_color(0, 0, 0) - self.viewer.add_geom(track) - - def _init_cart(self, rendering, cartwidth, cartheight): - lef, rig, top, bot = ( - -cartwidth / 2, - cartwidth / 2, - cartheight / 2, - -cartheight / 2, - ) - cart = rendering.FilledPolygon([(lef, bot), (lef, top), (rig, top), (rig, bot)]) - cart.add_attr(self.transforms["cart"]) - cart.set_color(1, 0, 0) - self.viewer.add_geom(cart) - - def _init_pole(self, rendering, polewidth, polelength): - lef, rig, top, bot = ( + l, r, t, b = ( -polewidth / 2, polewidth / 2, - polelength - polewidth / 2, + polelen - polewidth / 2, -polewidth / 2, ) - pole = rendering.FilledPolygon([(lef, bot), (lef, top), (rig, top), (rig, bot)]) - pole.set_color(0, 0, 1) - pole.add_attr(self.transforms["pole"]) - pole.add_attr(self.transforms["cart"]) - self.viewer.add_geom(pole) - - def _init_axle(self, rendering, polewidth): - axle = rendering.make_circle(polewidth / 2) - axle.add_attr(self.transforms["pole"]) - axle.add_attr(self.transforms["cart"]) - axle.set_color(0.1, 1, 1) - self.viewer.add_geom(axle) - - def _init_pole_bob(self, rendering, polewidth): - pole_bob = rendering.make_circle(polewidth / 2) - pole_bob.add_attr(self.transforms["pole_bob"]) - pole_bob.add_attr(self.transforms["pole"]) - pole_bob.add_attr(self.transforms["cart"]) - pole_bob.set_color(0, 0, 0) - self.viewer.add_geom(pole_bob) - - def _init_wheels(self, rendering, cartheight): - wheel_l = rendering.make_circle(cartheight / 4) - wheel_r = rendering.make_circle(cartheight / 4) - wheel_l.add_attr(self.transforms["wheel_l"]) - wheel_l.add_attr(self.transforms["cart"]) - wheel_r.add_attr(self.transforms["wheel_r"]) - wheel_r.add_attr(self.transforms["cart"]) - wheel_l.set_color(0, 0, 0) # Black, (B, G, R) - wheel_r.set_color(0, 0, 0) # Black, (B, G, R) - self.viewer.add_geom(wheel_l) - self.viewer.add_geom(wheel_r) - - def update(self, state): - """Updates the positions of the objects on screen.""" - screen = self.screen - scale = screen.width / self.world_width - - th, al, _, _ = state - al = (al - np.pi) % (2 * np.pi) # change angle origin - - # keep th between -pi and pi - th = (th + np.pi) % (2 * np.pi) - np.pi - - # use motor angle (theta) as cart x position - cartx = th * scale + screen.width / 2.0 # MIDDLE OF CART - carty = screen.height / 2 - - self.transforms["cart"].set_translation(cartx, carty) - self.transforms["pole"].set_rotation(al) - self.transforms["pole_bob"].set_translation( - -self.pole.length * np.sin(al), self.pole.length * np.cos(al) + + pole_coords = [] + for coord in [(l, b), (l, t), (r, t), (r, b)]: + coord = pygame.math.Vector2(coord).rotate_rad(x[ALPHA] + np.pi) + coord = (coord[0] + cartx, coord[1] + carty + axleoffset) + pole_coords.append(coord) + gfxdraw.aapolygon(self.surf, pole_coords, (202, 152, 101)) + gfxdraw.filled_polygon(self.surf, pole_coords, (202, 152, 101)) + + gfxdraw.aacircle( + self.surf, + int(cartx), + int(carty + axleoffset), + int(polewidth / 2), + (129, 132, 203), + ) + gfxdraw.filled_circle( + self.surf, + int(cartx), + int(carty + axleoffset), + int(polewidth / 2), + (129, 132, 203), ) - def render(self, *args, **kwargs): - """Forwards the call to the underlying Viewer instance.""" - return self.viewer.render(*args, **kwargs) + gfxdraw.hline(self.surf, 0, self.screen_width, carty, (0, 0, 0)) + + self.surf = pygame.transform.flip(self.surf, False, True) + self.screen.blit(self.surf, (0, 0)) + if self.render_mode == "human": + pygame.event.pump() + self.clock.tick(self.metadata["render_fps"]) + pygame.display.flip() + + elif self.render_mode == "rgb_array": + return np.transpose(np.array(pygame.surfarray.pixels3d(self.screen)), axes=(1, 0, 2)) def close(self): - """Closes the underlying Viewer instance.""" - self.viewer.close() + if self.screen is not None: + import pygame + + pygame.display.quit() + pygame.quit() + self.isopen = False From ad05b35cfec0f26f3b6133e6a63d9158fe9a4a49 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 14 Jan 2024 17:47:04 +0100 Subject: [PATCH 04/44] move dyn code out; upgrade to gymnasium --- furuta/rl/envs/furuta_sim.py | 145 ++++++----------------------------- 1 file changed, 24 insertions(+), 121 deletions(-) diff --git a/furuta/rl/envs/furuta_sim.py b/furuta/rl/envs/furuta_sim.py index a6ae5e4..bfe1793 100644 --- a/furuta/rl/envs/furuta_sim.py +++ b/furuta/rl/envs/furuta_sim.py @@ -1,9 +1,9 @@ -from math import cos, sin +from typing import Optional -import gym +import gymnasium as gym import numpy as np -from numpy.linalg import inv +from furuta.robot import QubeDynamics from furuta.utils import ALPHA, ALPHA_DOT, THETA, THETA_DOT, VelocityFilter from .furuta_base import FurutaBase @@ -12,18 +12,17 @@ class FurutaSim(FurutaBase): def __init__( self, - fs=50, + dyn: QubeDynamics = QubeDynamics(), + control_freq=50, reward="alpha", - state_limits=None, - sim_params=None, + state_limits=[np.pi, 2 * np.pi, 20, 20], encoders_CPRs=None, velocity_filter: int = None, + render_mode="rgb_array", ): - super().__init__(fs, reward, state_limits) - self.dyn = QubeDynamics() - if sim_params: - self.dyn.params = sim_params + super().__init__(control_freq, reward, state_limits, render_mode) + self.dyn = dyn self.encoders_CPRs = encoders_CPRs @@ -44,12 +43,16 @@ def _init_state(self): # or maybe it's too slow to move even the simulated pendulum? # and maybe it should have a min voltage as well? # self._state = np.random.rand(4) # self.state_space.sample() - self._simulation_state = 0.01 * np.float32(np.random.randn(self.state_space.shape[0])) - self._state = np.zeros(self.state_space.shape[0]) + + # TODO actually sample from system bounds? + # start at zero for now + self._simulation_state = np.zeros( + 4, dtype=np.float32 + ) # 0.01 * np.float32(np.random.randn(self.state_space.shape[0])) + self._state = np.zeros(self.state_space.shape[0], dtype=np.float32) self._update_state(0) - @profile def _update_state(self, a): # ok so we simulate two things: the systems's state # and the way we would measure it @@ -57,7 +60,7 @@ def _update_state(self, a): # update the simulation state thdd, aldd = self.dyn(self._simulation_state, a) - # integrate + # TODO integrate self._simulation_state[ALPHA_DOT] += self.timing.dt * aldd self._simulation_state[THETA_DOT] += self.timing.dt * thdd self._simulation_state[ALPHA] += self.timing.dt * self._simulation_state[ALPHA_DOT] @@ -88,10 +91,14 @@ def _update_state(self, a): self._state[THETA_DOT] = self._simulation_state[THETA_DOT] self._state[ALPHA_DOT] = self._simulation_state[ALPHA_DOT] - def reset(self): + def reset( + self, + seed: Optional[int] = None, + options: Optional[dict] = None, + ): self._init_state() - obs, _, _, _ = self.step(np.array([0.0])) - return obs + obs, _, _, _, _ = self.step(np.array([0.0])) + return obs, {} class Parameterized(gym.Wrapper): @@ -103,107 +110,3 @@ def params(self): def reset(self, params): self.unwrapped.dyn.params = params return self.env.reset() - - -class QubeDynamics: - """Solve equation M qdd + C(q, qd) = tau for qdd.""" - - def __init__(self): - # Gravity - self.g = 9.81 - - # Motor - self.Rm = 8.4 # resistance (rated voltage/stall current) - self.V = 12.0 # nominal voltage - - # back-emf constant (V-s/rad) - self.km = 0.042 # (rated voltage / no load speed) - - # Rotary arm - self.Mr = 0.095 # mass (kg) - self.Lr = 0.085 # length (m) - self.Dr = 5e-6 # viscous damping (N-m-s/rad), original: 0.0015 - - # Pendulum link - self.Mp = 0.024 # mass (kg) - self.Lp = 0.129 # length (m) - self.Dp = 1e-6 # viscous damping (N-m-s/rad), original: 0.0005 - - # Init constants - self._init_const() - - def _init_const(self): - # Moments of inertia - Jr = self.Mr * self.Lr**2 / 12 # inertia about COM (kg-m^2) - Jp = self.Mp * self.Lp**2 / 12 # inertia about COM (kg-m^2) - - # Constants for equations of motion - self._c = np.zeros(5) - self._c[0] = Jr + self.Mp * self.Lr**2 - self._c[1] = 0.25 * self.Mp * self.Lp**2 - self._c[2] = 0.5 * self.Mp * self.Lp * self.Lr - self._c[3] = Jp + self._c[1] - self._c[4] = 0.5 * self.Mp * self.Lp * self.g - - @property - def params(self): - params = self.__dict__.copy() - params.pop("_c") - return params - - @params.setter - def params(self, params): - self.__dict__.update(params) - self._init_const() - - @profile - def __call__(self, state, action): - # """ - # action between 0 and 1, maps to +V and -V - # """ - th, al, thd, ald = state - voltage = action * self.V - - # Precompute some values - sin_al = sin(al) - sin_2al = sin(2 * al) - cos_al = cos(al) - - # Define mass matrix M = [[a, b], [b, c]] - a = self._c[0] + self._c[1] * sin_al**2 - b = self._c[2] * cos_al - c = self._c[3] - d = a * c - b * b - - # Calculate vector [x, y] = tau - C(q, qd) - trq = self.km * (voltage - self.km * thd) / self.Rm - c0 = self._c[1] * sin_2al * thd * ald - self._c[2] * sin_al * ald * ald - c1 = -0.5 * self._c[1] * sin_2al * thd * thd + self._c[4] * sin_al - x = trq - self.Dr * thd - c0 - y = -self.Dp * ald - c1 - - # Compute M^{-1} @ [x, y] - thdd = (c * x - b * y) / d - aldd = (a * y - b * x) / d - - # chat gpt optimized code lol - # # Define mass matrix M = [[a, b], [b, c]] - # a = self._c[0] + self._c[1] * np.sin(al) ** 2 - # b = self._c[2] * np.cos(al) - # c = self._c[3] - # M = np.array([[a, b], [b, c]]) - # Minv = inv(M) - - # # Calculate vector [x, y] = tau - C(q, qd) - # trq = self.km * (voltage - self.km * thd) / self.Rm - # c0 = self._c[1] * np.sin(2 * al) * thd * ald - self._c[2] * np.sin(al) * ald * ald - # c1 = -0.5 * self._c[1] * np.sin(2 * al) * thd * thd + self._c[4] * np.sin(al) - # x = trq - self.Dr * thd - c0 - # y = -self.Dp * ald - c1 - # v = np.array([x, y]) - - # # Compute M^{-1} @ v - # acc = np.dot(Minv, v) - # thdd, aldd = acc - - return thdd, aldd From 6edfde306a0259d2bfbb9c8d2ecb74144ff2923d Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 14 Jan 2024 17:48:47 +0100 Subject: [PATCH 05/44] upgrade to gymnasium --- furuta/rl/wrappers.py | 54 ++++++++++++++++++++++++++++--------------- furuta/utils.py | 2 +- 2 files changed, 36 insertions(+), 20 deletions(-) diff --git a/furuta/rl/wrappers.py b/furuta/rl/wrappers.py index 7d9003f..cecd246 100644 --- a/furuta/rl/wrappers.py +++ b/furuta/rl/wrappers.py @@ -1,12 +1,12 @@ import logging import time from pathlib import Path -from typing import Union +from typing import Optional, Union -import gym +import gymnasium as gym import numpy as np import wandb -from gym.spaces import Box +from gymnasium.spaces import Box from mcap_protobuf.writer import Writer from furuta.logging.protobuf.pendulum_state_pb2 import PendulumState @@ -16,13 +16,17 @@ class GentlyTerminating(gym.Wrapper): """This env wrapper sends zero command to the robot when an episode is done.""" def step(self, action): - observation, reward, done, info = self.env.step(action) - if done: + observation, reward, terminated, truncated, info = self.env.step(action) + if terminated or truncated: logging.debug("episode done, killing motor.") - self.env.motor.set_speed(0) - return observation, reward, done, info - - def reset(self): + self.unwrapped.robot.step(0.0) + return observation, reward, terminated, truncated, info + + def reset( + self, + seed: Optional[int] = None, + options: Optional[dict] = None, + ): return self.env.reset() @@ -60,7 +64,11 @@ def step(self, action): return observation, reward, done, info - def reset(self): + def reset( + self, + seed: Optional[int] = None, + options: Optional[dict] = None, + ): # create log dir if doesn't exist if not self.log_dir.exists(): self.log_dir.mkdir(parents=True) @@ -110,17 +118,21 @@ def step(self, action): if sleeping_time > 0: time.sleep(sleeping_time) else: - print("warning, loop time > dt") + logging.info("warning, loop time > dt") - obs, reward, done, info = self.env.step(np.array([0.0])) + obs, reward, terminated, truncated, info = self.env.step(np.array([0.0])) self.last = time.time() if logging.root.level == logging.DEBUG: wandb.log({**info, **{"loop time": loop_time}}) - return obs, reward, done, info + return obs, reward, terminated, truncated, info - def reset(self): + def reset( + self, + seed: Optional[int] = None, + options: Optional[dict] = None, + ): self.last = None return self.env.reset() @@ -159,7 +171,7 @@ def _continuity_cost(self, obs): return continuity_cost def step(self, action): - obs, reward, done, info = self.env.step(action) + obs, reward, terminated, truncated, info = self.env.step(action) self.history.pop(0) obs = np.concatenate([obs, action]) @@ -171,11 +183,15 @@ def step(self, action): reward -= continuity_cost info["continuity_cost"] = continuity_cost - return obs, reward, done, info + return obs, reward, terminated, truncated, info - def reset(self): + def reset( + self, + seed: Optional[int] = None, + options: Optional[dict] = None, + ): self.history = self._make_history() self.history.pop(0) - obs = np.concatenate([self.env.reset(), np.zeros_like(self.env.action_space.low)]) + obs = np.concatenate([self.env.reset()[0], np.zeros_like(self.env.action_space.low)]) self.history.append(obs) - return np.array(self.history) + return np.array(self.history), {} diff --git a/furuta/utils.py b/furuta/utils.py index bd535b2..c1206ae 100644 --- a/furuta/utils.py +++ b/furuta/utils.py @@ -1,6 +1,6 @@ # https://git.ias.informatik.tu-darmstadt.de/quanser/clients/-/blob/v0.1.1/quanser_robots/common.py import numpy as np -from gym import spaces +from gymnasium import spaces from scipy import signal THETA = 0 From 8f65b669735423b9dae06c7468e5bf8f16e6c01d Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 14 Jan 2024 17:49:16 +0100 Subject: [PATCH 06/44] add missing dependencies --- poetry.lock | 1388 +++++++++++++++++++++++++++++++++++++++++++++--- pyproject.toml | 8 +- 2 files changed, 1329 insertions(+), 67 deletions(-) diff --git a/poetry.lock b/poetry.lock index 5404f2e..592cca0 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,5 +1,17 @@ # This file is automatically @generated by Poetry 1.4.0 and should not be changed by hand. +[[package]] +name = "appdirs" +version = "1.4.4" +description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"}, + {file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"}, +] + [[package]] name = "appnope" version = "0.1.3" @@ -43,6 +55,18 @@ files = [ {file = "backcall-0.2.0.tar.gz", hash = "sha256:5cbdbf27be5e7cfadb448baf0aa95508f91f2bbc6c6437cd9cd06e2a4c215e1e"}, ] +[[package]] +name = "certifi" +version = "2023.11.17" +description = "Python package for providing Mozilla's CA Bundle." +category = "main" +optional = false +python-versions = ">=3.6" +files = [ + {file = "certifi-2023.11.17-py3-none-any.whl", hash = "sha256:e036ab49d5b79556f99cfc2d9320b34cfbe5be05c5871b51de9329f0603b0474"}, + {file = "certifi-2023.11.17.tar.gz", hash = "sha256:9b469f3a900bf28dc19b8cfbf8019bf47f7fdd1a65a1d4ffb98fc14166beb4d1"}, +] + [[package]] name = "cffi" version = "1.16.0" @@ -120,6 +144,121 @@ files = [ {file = "cfgv-3.4.0.tar.gz", hash = "sha256:e52591d4c5f5dead8e0f673fb16db7949d2cfb3f7da4582893288f0ded8fe560"}, ] +[[package]] +name = "charset-normalizer" +version = "3.3.2" +description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." +category = "main" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "charset-normalizer-3.3.2.tar.gz", hash = "sha256:f30c3cb33b24454a82faecaf01b19c18562b1e89558fb6c56de4d9118a032fd5"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:25baf083bf6f6b341f4121c2f3c548875ee6f5339300e08be3f2b2ba1721cdd3"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:06435b539f889b1f6f4ac1758871aae42dc3a8c0e24ac9e60c2384973ad73027"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9063e24fdb1e498ab71cb7419e24622516c4a04476b17a2dab57e8baa30d6e03"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6897af51655e3691ff853668779c7bad41579facacf5fd7253b0133308cf000d"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1d3193f4a680c64b4b6a9115943538edb896edc190f0b222e73761716519268e"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cd70574b12bb8a4d2aaa0094515df2463cb429d8536cfb6c7ce983246983e5a6"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8465322196c8b4d7ab6d1e049e4c5cb460d0394da4a27d23cc242fbf0034b6b5"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a9a8e9031d613fd2009c182b69c7b2c1ef8239a0efb1df3f7c8da66d5dd3d537"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:beb58fe5cdb101e3a055192ac291b7a21e3b7ef4f67fa1d74e331a7f2124341c"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:e06ed3eb3218bc64786f7db41917d4e686cc4856944f53d5bdf83a6884432e12"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:2e81c7b9c8979ce92ed306c249d46894776a909505d8f5a4ba55b14206e3222f"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:572c3763a264ba47b3cf708a44ce965d98555f618ca42c926a9c1616d8f34269"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fd1abc0d89e30cc4e02e4064dc67fcc51bd941eb395c502aac3ec19fab46b519"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-win32.whl", hash = "sha256:3d47fa203a7bd9c5b6cee4736ee84ca03b8ef23193c0d1ca99b5089f72645c73"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-win_amd64.whl", hash = "sha256:10955842570876604d404661fbccbc9c7e684caf432c09c715ec38fbae45ae09"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:802fe99cca7457642125a8a88a084cef28ff0cf9407060f7b93dca5aa25480db"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:573f6eac48f4769d667c4442081b1794f52919e7edada77495aaed9236d13a96"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:549a3a73da901d5bc3ce8d24e0600d1fa85524c10287f6004fbab87672bf3e1e"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f27273b60488abe721a075bcca6d7f3964f9f6f067c8c4c605743023d7d3944f"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1ceae2f17a9c33cb48e3263960dc5fc8005351ee19db217e9b1bb15d28c02574"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:65f6f63034100ead094b8744b3b97965785388f308a64cf8d7c34f2f2e5be0c4"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:753f10e867343b4511128c6ed8c82f7bec3bd026875576dfd88483c5c73b2fd8"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4a78b2b446bd7c934f5dcedc588903fb2f5eec172f3d29e52a9096a43722adfc"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e537484df0d8f426ce2afb2d0f8e1c3d0b114b83f8850e5f2fbea0e797bd82ae"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:eb6904c354526e758fda7167b33005998fb68c46fbc10e013ca97f21ca5c8887"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:deb6be0ac38ece9ba87dea880e438f25ca3eddfac8b002a2ec3d9183a454e8ae"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:4ab2fe47fae9e0f9dee8c04187ce5d09f48eabe611be8259444906793ab7cbce"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:80402cd6ee291dcb72644d6eac93785fe2c8b9cb30893c1af5b8fdd753b9d40f"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-win32.whl", hash = "sha256:7cd13a2e3ddeed6913a65e66e94b51d80a041145a026c27e6bb76c31a853c6ab"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-win_amd64.whl", hash = "sha256:663946639d296df6a2bb2aa51b60a2454ca1cb29835324c640dafb5ff2131a77"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:0b2b64d2bb6d3fb9112bafa732def486049e63de9618b5843bcdd081d8144cd8"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:ddbb2551d7e0102e7252db79ba445cdab71b26640817ab1e3e3648dad515003b"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:55086ee1064215781fff39a1af09518bc9255b50d6333f2e4c74ca09fac6a8f6"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f4a014bc36d3c57402e2977dada34f9c12300af536839dc38c0beab8878f38a"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a10af20b82360ab00827f916a6058451b723b4e65030c5a18577c8b2de5b3389"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8d756e44e94489e49571086ef83b2bb8ce311e730092d2c34ca8f7d925cb20aa"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:90d558489962fd4918143277a773316e56c72da56ec7aa3dc3dbbe20fdfed15b"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6ac7ffc7ad6d040517be39eb591cac5ff87416c2537df6ba3cba3bae290c0fed"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:7ed9e526742851e8d5cc9e6cf41427dfc6068d4f5a3bb03659444b4cabf6bc26"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:8bdb58ff7ba23002a4c5808d608e4e6c687175724f54a5dade5fa8c67b604e4d"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:6b3251890fff30ee142c44144871185dbe13b11bab478a88887a639655be1068"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:b4a23f61ce87adf89be746c8a8974fe1c823c891d8f86eb218bb957c924bb143"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:efcb3f6676480691518c177e3b465bcddf57cea040302f9f4e6e191af91174d4"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-win32.whl", hash = "sha256:d965bba47ddeec8cd560687584e88cf699fd28f192ceb452d1d7ee807c5597b7"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-win_amd64.whl", hash = "sha256:96b02a3dc4381e5494fad39be677abcb5e6634bf7b4fa83a6dd3112607547001"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:95f2a5796329323b8f0512e09dbb7a1860c46a39da62ecb2324f116fa8fdc85c"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c002b4ffc0be611f0d9da932eb0f704fe2602a9a949d1f738e4c34c75b0863d5"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a981a536974bbc7a512cf44ed14938cf01030a99e9b3a06dd59578882f06f985"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3287761bc4ee9e33561a7e058c72ac0938c4f57fe49a09eae428fd88aafe7bb6"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:42cb296636fcc8b0644486d15c12376cb9fa75443e00fb25de0b8602e64c1714"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0a55554a2fa0d408816b3b5cedf0045f4b8e1a6065aec45849de2d6f3f8e9786"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:c083af607d2515612056a31f0a8d9e0fcb5876b7bfc0abad3ecd275bc4ebc2d5"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:87d1351268731db79e0f8e745d92493ee2841c974128ef629dc518b937d9194c"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:bd8f7df7d12c2db9fab40bdd87a7c09b1530128315d047a086fa3ae3435cb3a8"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:c180f51afb394e165eafe4ac2936a14bee3eb10debc9d9e4db8958fe36afe711"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:8c622a5fe39a48f78944a87d4fb8a53ee07344641b0562c540d840748571b811"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-win32.whl", hash = "sha256:db364eca23f876da6f9e16c9da0df51aa4f104a972735574842618b8c6d999d4"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-win_amd64.whl", hash = "sha256:86216b5cee4b06df986d214f664305142d9c76df9b6512be2738aa72a2048f99"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:6463effa3186ea09411d50efc7d85360b38d5f09b870c48e4600f63af490e56a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6c4caeef8fa63d06bd437cd4bdcf3ffefe6738fb1b25951440d80dc7df8c03ac"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:37e55c8e51c236f95b033f6fb391d7d7970ba5fe7ff453dad675e88cf303377a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fb69256e180cb6c8a894fee62b3afebae785babc1ee98b81cdf68bbca1987f33"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ae5f4161f18c61806f411a13b0310bea87f987c7d2ecdbdaad0e94eb2e404238"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b2b0a0c0517616b6869869f8c581d4eb2dd83a4d79e0ebcb7d373ef9956aeb0a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:45485e01ff4d3630ec0d9617310448a8702f70e9c01906b0d0118bdf9d124cf2"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:eb00ed941194665c332bf8e078baf037d6c35d7c4f3102ea2d4f16ca94a26dc8"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:2127566c664442652f024c837091890cb1942c30937add288223dc895793f898"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a50aebfa173e157099939b17f18600f72f84eed3049e743b68ad15bd69b6bf99"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:4d0d1650369165a14e14e1e47b372cfcb31d6ab44e6e33cb2d4e57265290044d"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:923c0c831b7cfcb071580d3f46c4baf50f174be571576556269530f4bbd79d04"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:06a81e93cd441c56a9b65d8e1d043daeb97a3d0856d177d5c90ba85acb3db087"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-win32.whl", hash = "sha256:6ef1d82a3af9d3eecdba2321dc1b3c238245d890843e040e41e470ffa64c3e25"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-win_amd64.whl", hash = "sha256:eb8821e09e916165e160797a6c17edda0679379a4be5c716c260e836e122f54b"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:c235ebd9baae02f1b77bcea61bce332cb4331dc3617d254df3323aa01ab47bd4"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5b4c145409bef602a690e7cfad0a15a55c13320ff7a3ad7ca59c13bb8ba4d45d"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:68d1f8a9e9e37c1223b656399be5d6b448dea850bed7d0f87a8311f1ff3dabb0"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:22afcb9f253dac0696b5a4be4a1c0f8762f8239e21b99680099abd9b2b1b2269"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e27ad930a842b4c5eb8ac0016b0a54f5aebbe679340c26101df33424142c143c"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1f79682fbe303db92bc2b1136016a38a42e835d932bab5b3b1bfcfbf0640e519"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b261ccdec7821281dade748d088bb6e9b69e6d15b30652b74cbbac25e280b796"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:122c7fa62b130ed55f8f285bfd56d5f4b4a5b503609d181f9ad85e55c89f4185"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:d0eccceffcb53201b5bfebb52600a5fb483a20b61da9dbc885f8b103cbe7598c"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:9f96df6923e21816da7e0ad3fd47dd8f94b2a5ce594e00677c0013018b813458"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:7f04c839ed0b6b98b1a7501a002144b76c18fb1c1850c8b98d458ac269e26ed2"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:34d1c8da1e78d2e001f363791c98a272bb734000fcef47a491c1e3b0505657a8"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ff8fa367d09b717b2a17a052544193ad76cd49979c805768879cb63d9ca50561"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-win32.whl", hash = "sha256:aed38f6e4fb3f5d6bf81bfa990a07806be9d83cf7bacef998ab1a9bd660a581f"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-win_amd64.whl", hash = "sha256:b01b88d45a6fcb69667cd6d2f7a9aeb4bf53760d7fc536bf679ec94fe9f3ff3d"}, + {file = "charset_normalizer-3.3.2-py3-none-any.whl", hash = "sha256:3e4d1f6587322d2788836a99c69062fbb091331ec940e02d12d179c1d53e25fc"}, +] + +[[package]] +name = "click" +version = "8.1.7" +description = "Composable command line interface toolkit" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "click-8.1.7-py3-none-any.whl", hash = "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28"}, + {file = "click-8.1.7.tar.gz", hash = "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "platform_system == \"Windows\""} + [[package]] name = "cloudpickle" version = "3.0.0" @@ -162,69 +301,6 @@ traitlets = ">=4" [package.extras] test = ["pytest"] -[[package]] -name = "contourpy" -version = "1.1.0" -description = "Python library for calculating contours of 2D quadrilateral grids" -category = "main" -optional = false -python-versions = ">=3.8" -files = [ - {file = "contourpy-1.1.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:89f06eff3ce2f4b3eb24c1055a26981bffe4e7264acd86f15b97e40530b794bc"}, - {file = "contourpy-1.1.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:dffcc2ddec1782dd2f2ce1ef16f070861af4fb78c69862ce0aab801495dda6a3"}, - {file = "contourpy-1.1.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:25ae46595e22f93592d39a7eac3d638cda552c3e1160255258b695f7b58e5655"}, - {file = "contourpy-1.1.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:17cfaf5ec9862bc93af1ec1f302457371c34e688fbd381f4035a06cd47324f48"}, - {file = "contourpy-1.1.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:18a64814ae7bce73925131381603fff0116e2df25230dfc80d6d690aa6e20b37"}, - {file = "contourpy-1.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:90c81f22b4f572f8a2110b0b741bb64e5a6427e0a198b2cdc1fbaf85f352a3aa"}, - {file = "contourpy-1.1.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:53cc3a40635abedbec7f1bde60f8c189c49e84ac180c665f2cd7c162cc454baa"}, - {file = "contourpy-1.1.0-cp310-cp310-win32.whl", hash = "sha256:9b2dd2ca3ac561aceef4c7c13ba654aaa404cf885b187427760d7f7d4c57cff8"}, - {file = "contourpy-1.1.0-cp310-cp310-win_amd64.whl", hash = "sha256:1f795597073b09d631782e7245016a4323cf1cf0b4e06eef7ea6627e06a37ff2"}, - {file = "contourpy-1.1.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:0b7b04ed0961647691cfe5d82115dd072af7ce8846d31a5fac6c142dcce8b882"}, - {file = "contourpy-1.1.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:27bc79200c742f9746d7dd51a734ee326a292d77e7d94c8af6e08d1e6c15d545"}, - {file = "contourpy-1.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:052cc634bf903c604ef1a00a5aa093c54f81a2612faedaa43295809ffdde885e"}, - {file = "contourpy-1.1.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9382a1c0bc46230fb881c36229bfa23d8c303b889b788b939365578d762b5c18"}, - {file = "contourpy-1.1.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e5cec36c5090e75a9ac9dbd0ff4a8cf7cecd60f1b6dc23a374c7d980a1cd710e"}, - {file = "contourpy-1.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1f0cbd657e9bde94cd0e33aa7df94fb73c1ab7799378d3b3f902eb8eb2e04a3a"}, - {file = "contourpy-1.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:181cbace49874f4358e2929aaf7ba84006acb76694102e88dd15af861996c16e"}, - {file = "contourpy-1.1.0-cp311-cp311-win32.whl", hash = "sha256:edb989d31065b1acef3828a3688f88b2abb799a7db891c9e282df5ec7e46221b"}, - {file = "contourpy-1.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:fb3b7d9e6243bfa1efb93ccfe64ec610d85cfe5aec2c25f97fbbd2e58b531256"}, - {file = "contourpy-1.1.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:bcb41692aa09aeb19c7c213411854402f29f6613845ad2453d30bf421fe68fed"}, - {file = "contourpy-1.1.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:5d123a5bc63cd34c27ff9c7ac1cd978909e9c71da12e05be0231c608048bb2ae"}, - {file = "contourpy-1.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:62013a2cf68abc80dadfd2307299bfa8f5aa0dcaec5b2954caeb5fa094171103"}, - {file = "contourpy-1.1.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0b6616375d7de55797d7a66ee7d087efe27f03d336c27cf1f32c02b8c1a5ac70"}, - {file = "contourpy-1.1.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:317267d915490d1e84577924bd61ba71bf8681a30e0d6c545f577363157e5e94"}, - {file = "contourpy-1.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d551f3a442655f3dcc1285723f9acd646ca5858834efeab4598d706206b09c9f"}, - {file = "contourpy-1.1.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:e7a117ce7df5a938fe035cad481b0189049e8d92433b4b33aa7fc609344aafa1"}, - {file = "contourpy-1.1.0-cp38-cp38-win32.whl", hash = "sha256:108dfb5b3e731046a96c60bdc46a1a0ebee0760418951abecbe0fc07b5b93b27"}, - {file = "contourpy-1.1.0-cp38-cp38-win_amd64.whl", hash = "sha256:d4f26b25b4f86087e7d75e63212756c38546e70f2a92d2be44f80114826e1cd4"}, - {file = "contourpy-1.1.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bc00bb4225d57bff7ebb634646c0ee2a1298402ec10a5fe7af79df9a51c1bfd9"}, - {file = "contourpy-1.1.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:189ceb1525eb0655ab8487a9a9c41f42a73ba52d6789754788d1883fb06b2d8a"}, - {file = "contourpy-1.1.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9f2931ed4741f98f74b410b16e5213f71dcccee67518970c42f64153ea9313b9"}, - {file = "contourpy-1.1.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:30f511c05fab7f12e0b1b7730ebdc2ec8deedcfb505bc27eb570ff47c51a8f15"}, - {file = "contourpy-1.1.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:143dde50520a9f90e4a2703f367cf8ec96a73042b72e68fcd184e1279962eb6f"}, - {file = "contourpy-1.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e94bef2580e25b5fdb183bf98a2faa2adc5b638736b2c0a4da98691da641316a"}, - {file = "contourpy-1.1.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ed614aea8462735e7d70141374bd7650afd1c3f3cb0c2dbbcbe44e14331bf002"}, - {file = "contourpy-1.1.0-cp39-cp39-win32.whl", hash = "sha256:71551f9520f008b2950bef5f16b0e3587506ef4f23c734b71ffb7b89f8721999"}, - {file = "contourpy-1.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:438ba416d02f82b692e371858143970ed2eb6337d9cdbbede0d8ad9f3d7dd17d"}, - {file = "contourpy-1.1.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:a698c6a7a432789e587168573a864a7ea374c6be8d4f31f9d87c001d5a843493"}, - {file = "contourpy-1.1.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:397b0ac8a12880412da3551a8cb5a187d3298a72802b45a3bd1805e204ad8439"}, - {file = "contourpy-1.1.0-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:a67259c2b493b00e5a4d0f7bfae51fb4b3371395e47d079a4446e9b0f4d70e76"}, - {file = "contourpy-1.1.0-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:2b836d22bd2c7bb2700348e4521b25e077255ebb6ab68e351ab5aa91ca27e027"}, - {file = "contourpy-1.1.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:084eaa568400cfaf7179b847ac871582199b1b44d5699198e9602ecbbb5f6104"}, - {file = "contourpy-1.1.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:911ff4fd53e26b019f898f32db0d4956c9d227d51338fb3b03ec72ff0084ee5f"}, - {file = "contourpy-1.1.0.tar.gz", hash = "sha256:e53046c3863828d21d531cc3b53786e6580eb1ba02477e8681009b6aa0870b21"}, -] - -[package.dependencies] -numpy = ">=1.16" - -[package.extras] -bokeh = ["bokeh", "selenium"] -docs = ["furo", "sphinx-copybutton"] -mypy = ["contourpy[bokeh,docs]", "docutils-stubs", "mypy (==1.2.0)", "types-Pillow"] -test = ["Pillow", "contourpy[test-no-images]", "matplotlib"] -test-no-images = ["pytest", "pytest-cov", "wurlitzer"] - [[package]] name = "contourpy" version = "1.1.1" @@ -288,7 +364,10 @@ files = [ ] [package.dependencies] -numpy = {version = ">=1.16,<2.0", markers = "python_version <= \"3.11\""} +numpy = [ + {version = ">=1.16,<2.0", markers = "python_version <= \"3.11\""}, + {version = ">=1.26.0rc1,<2.0", markers = "python_version >= \"3.12\""}, +] [package.extras] bokeh = ["bokeh", "selenium"] @@ -365,6 +444,21 @@ files = [ {file = "distlib-0.3.7.tar.gz", hash = "sha256:9dafe54b34a028eafd95039d5e5d4851a13734540f1331060d31c9916e7147a8"}, ] +[[package]] +name = "docker-pycreds" +version = "0.4.0" +description = "Python bindings for the docker credentials store API" +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "docker-pycreds-0.4.0.tar.gz", hash = "sha256:6ce3270bcaf404cc4c3e27e4b6c70d3521deae82fb508767870fdbf772d584d4"}, + {file = "docker_pycreds-0.4.0-py2.py3-none-any.whl", hash = "sha256:7266112468627868005106ec19cd0d722702d2b7d5912a28e19b826c3d37af49"}, +] + +[package.dependencies] +six = ">=1.4.0" + [[package]] name = "exceptiongroup" version = "1.2.0" @@ -490,6 +584,75 @@ ufo = ["fs (>=2.2.0,<3)"] unicode = ["unicodedata2 (>=15.1.0)"] woff = ["brotli (>=1.0.1)", "brotlicffi (>=0.8.0)", "zopfli (>=0.1.4)"] +[[package]] +name = "fsspec" +version = "2023.12.2" +description = "File-system specification" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "fsspec-2023.12.2-py3-none-any.whl", hash = "sha256:d800d87f72189a745fa3d6b033b9dc4a34ad069f60ca60b943a63599f5501960"}, + {file = "fsspec-2023.12.2.tar.gz", hash = "sha256:8548d39e8810b59c38014934f6b31e57f40c1b20f911f4cc2b85389c7e9bf0cb"}, +] + +[package.extras] +abfs = ["adlfs"] +adl = ["adlfs"] +arrow = ["pyarrow (>=1)"] +dask = ["dask", "distributed"] +devel = ["pytest", "pytest-cov"] +dropbox = ["dropbox", "dropboxdrivefs", "requests"] +full = ["adlfs", "aiohttp (!=4.0.0a0,!=4.0.0a1)", "dask", "distributed", "dropbox", "dropboxdrivefs", "fusepy", "gcsfs", "libarchive-c", "ocifs", "panel", "paramiko", "pyarrow (>=1)", "pygit2", "requests", "s3fs", "smbprotocol", "tqdm"] +fuse = ["fusepy"] +gcs = ["gcsfs"] +git = ["pygit2"] +github = ["requests"] +gs = ["gcsfs"] +gui = ["panel"] +hdfs = ["pyarrow (>=1)"] +http = ["aiohttp (!=4.0.0a0,!=4.0.0a1)", "requests"] +libarchive = ["libarchive-c"] +oci = ["ocifs"] +s3 = ["s3fs"] +sftp = ["paramiko"] +smb = ["smbprotocol"] +ssh = ["paramiko"] +tqdm = ["tqdm"] + +[[package]] +name = "gitdb" +version = "4.0.11" +description = "Git Object Database" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "gitdb-4.0.11-py3-none-any.whl", hash = "sha256:81a3407ddd2ee8df444cbacea00e2d038e40150acfa3001696fe0dcf1d3adfa4"}, + {file = "gitdb-4.0.11.tar.gz", hash = "sha256:bf5421126136d6d0af55bc1e7c1af1c397a34f5b7bd79e776cd3e89785c2b04b"}, +] + +[package.dependencies] +smmap = ">=3.0.1,<6" + +[[package]] +name = "gitpython" +version = "3.1.41" +description = "GitPython is a Python library used to interact with Git repositories" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "GitPython-3.1.41-py3-none-any.whl", hash = "sha256:c36b6634d069b3f719610175020a9aed919421c87552185b085e04fbbdb10b7c"}, + {file = "GitPython-3.1.41.tar.gz", hash = "sha256:ed66e624884f76df22c8e16066d567aaa5a37d5b5fa19db2c6df6f7156db9048"}, +] + +[package.dependencies] +gitdb = ">=4.0.1,<5" + +[package.extras] +test = ["black", "coverage[toml]", "ddt (>=1.1.1,!=1.4.3)", "mock", "mypy", "pre-commit", "pytest (>=7.3.1)", "pytest-cov", "pytest-instafail", "pytest-mock", "pytest-sugar", "sumtypes"] + [[package]] name = "gymnasium" version = "0.29.1" @@ -537,6 +700,18 @@ files = [ [package.extras] license = ["ukkonen"] +[[package]] +name = "idna" +version = "3.6" +description = "Internationalized Domain Names in Applications (IDNA)" +category = "main" +optional = false +python-versions = ">=3.5" +files = [ + {file = "idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f"}, + {file = "idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca"}, +] + [[package]] name = "importlib-metadata" version = "6.8.0" @@ -682,6 +857,24 @@ docs = ["Jinja2 (==2.11.3)", "MarkupSafe (==1.1.1)", "Pygments (==2.8.1)", "alab qa = ["flake8 (==5.0.4)", "mypy (==0.971)", "types-setuptools (==67.2.0.1)"] testing = ["Django", "attrs", "colorama", "docopt", "pytest (<7.0.0)"] +[[package]] +name = "jinja2" +version = "3.1.3" +description = "A very fast and expressive template engine." +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "Jinja2-3.1.3-py3-none-any.whl", hash = "sha256:7d6d50dd97d52cbc355597bd845fabfbac3f551e1f99619e39a35ce8c370b5fa"}, + {file = "Jinja2-3.1.3.tar.gz", hash = "sha256:ac8bd6544d4bb2c9792bf3a159e80bba8fda7f07e81bc3aed565432d5925ba90"}, +] + +[package.dependencies] +MarkupSafe = ">=2.0" + +[package.extras] +i18n = ["Babel (>=2.7)"] + [[package]] name = "jupyter-client" version = "8.6.0" @@ -841,6 +1034,127 @@ files = [ {file = "kiwisolver-1.4.5.tar.gz", hash = "sha256:e57e563a57fb22a142da34f38acc2fc1a5c864bc29ca1517a88abc963e60d6ec"}, ] +[[package]] +name = "lz4" +version = "4.3.3" +description = "LZ4 Bindings for Python" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "lz4-4.3.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:b891880c187e96339474af2a3b2bfb11a8e4732ff5034be919aa9029484cd201"}, + {file = "lz4-4.3.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:222a7e35137d7539c9c33bb53fcbb26510c5748779364014235afc62b0ec797f"}, + {file = "lz4-4.3.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f76176492ff082657ada0d0f10c794b6da5800249ef1692b35cf49b1e93e8ef7"}, + {file = "lz4-4.3.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f1d18718f9d78182c6b60f568c9a9cec8a7204d7cb6fad4e511a2ef279e4cb05"}, + {file = "lz4-4.3.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6cdc60e21ec70266947a48839b437d46025076eb4b12c76bd47f8e5eb8a75dcc"}, + {file = "lz4-4.3.3-cp310-cp310-win32.whl", hash = "sha256:c81703b12475da73a5d66618856d04b1307e43428a7e59d98cfe5a5d608a74c6"}, + {file = "lz4-4.3.3-cp310-cp310-win_amd64.whl", hash = "sha256:43cf03059c0f941b772c8aeb42a0813d68d7081c009542301637e5782f8a33e2"}, + {file = "lz4-4.3.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:30e8c20b8857adef7be045c65f47ab1e2c4fabba86a9fa9a997d7674a31ea6b6"}, + {file = "lz4-4.3.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:2f7b1839f795315e480fb87d9bc60b186a98e3e5d17203c6e757611ef7dcef61"}, + {file = "lz4-4.3.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:edfd858985c23523f4e5a7526ca6ee65ff930207a7ec8a8f57a01eae506aaee7"}, + {file = "lz4-4.3.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0e9c410b11a31dbdc94c05ac3c480cb4b222460faf9231f12538d0074e56c563"}, + {file = "lz4-4.3.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d2507ee9c99dbddd191c86f0e0c8b724c76d26b0602db9ea23232304382e1f21"}, + {file = "lz4-4.3.3-cp311-cp311-win32.whl", hash = "sha256:f180904f33bdd1e92967923a43c22899e303906d19b2cf8bb547db6653ea6e7d"}, + {file = "lz4-4.3.3-cp311-cp311-win_amd64.whl", hash = "sha256:b14d948e6dce389f9a7afc666d60dd1e35fa2138a8ec5306d30cd2e30d36b40c"}, + {file = "lz4-4.3.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:e36cd7b9d4d920d3bfc2369840da506fa68258f7bb176b8743189793c055e43d"}, + {file = "lz4-4.3.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:31ea4be9d0059c00b2572d700bf2c1bc82f241f2c3282034a759c9a4d6ca4dc2"}, + {file = "lz4-4.3.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:33c9a6fd20767ccaf70649982f8f3eeb0884035c150c0b818ea660152cf3c809"}, + {file = "lz4-4.3.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bca8fccc15e3add173da91be8f34121578dc777711ffd98d399be35487c934bf"}, + {file = "lz4-4.3.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e7d84b479ddf39fe3ea05387f10b779155fc0990125f4fb35d636114e1c63a2e"}, + {file = "lz4-4.3.3-cp312-cp312-win32.whl", hash = "sha256:337cb94488a1b060ef1685187d6ad4ba8bc61d26d631d7ba909ee984ea736be1"}, + {file = "lz4-4.3.3-cp312-cp312-win_amd64.whl", hash = "sha256:5d35533bf2cee56f38ced91f766cd0038b6abf46f438a80d50c52750088be93f"}, + {file = "lz4-4.3.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:363ab65bf31338eb364062a15f302fc0fab0a49426051429866d71c793c23394"}, + {file = "lz4-4.3.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:0a136e44a16fc98b1abc404fbabf7f1fada2bdab6a7e970974fb81cf55b636d0"}, + {file = "lz4-4.3.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:abc197e4aca8b63f5ae200af03eb95fb4b5055a8f990079b5bdf042f568469dd"}, + {file = "lz4-4.3.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:56f4fe9c6327adb97406f27a66420b22ce02d71a5c365c48d6b656b4aaeb7775"}, + {file = "lz4-4.3.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f0e822cd7644995d9ba248cb4b67859701748a93e2ab7fc9bc18c599a52e4604"}, + {file = "lz4-4.3.3-cp38-cp38-win32.whl", hash = "sha256:24b3206de56b7a537eda3a8123c644a2b7bf111f0af53bc14bed90ce5562d1aa"}, + {file = "lz4-4.3.3-cp38-cp38-win_amd64.whl", hash = "sha256:b47839b53956e2737229d70714f1d75f33e8ac26e52c267f0197b3189ca6de24"}, + {file = "lz4-4.3.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6756212507405f270b66b3ff7f564618de0606395c0fe10a7ae2ffcbbe0b1fba"}, + {file = "lz4-4.3.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ee9ff50557a942d187ec85462bb0960207e7ec5b19b3b48949263993771c6205"}, + {file = "lz4-4.3.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2b901c7784caac9a1ded4555258207d9e9697e746cc8532129f150ffe1f6ba0d"}, + {file = "lz4-4.3.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b6d9ec061b9eca86e4dcc003d93334b95d53909afd5a32c6e4f222157b50c071"}, + {file = "lz4-4.3.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f4c7bf687303ca47d69f9f0133274958fd672efaa33fb5bcde467862d6c621f0"}, + {file = "lz4-4.3.3-cp39-cp39-win32.whl", hash = "sha256:054b4631a355606e99a42396f5db4d22046a3397ffc3269a348ec41eaebd69d2"}, + {file = "lz4-4.3.3-cp39-cp39-win_amd64.whl", hash = "sha256:eac9af361e0d98335a02ff12fb56caeb7ea1196cf1a49dbf6f17828a131da807"}, + {file = "lz4-4.3.3.tar.gz", hash = "sha256:01fe674ef2889dbb9899d8a67361e0c4a2c833af5aeb37dd505727cf5d2a131e"}, +] + +[package.extras] +docs = ["sphinx (>=1.6.0)", "sphinx-bootstrap-theme"] +flake8 = ["flake8"] +tests = ["psutil", "pytest (!=3.3.0)", "pytest-cov"] + +[[package]] +name = "markupsafe" +version = "2.1.3" +description = "Safely add untrusted strings to HTML/XML markup." +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "MarkupSafe-2.1.3-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:cd0f502fe016460680cd20aaa5a76d241d6f35a1c3350c474bac1273803893fa"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:e09031c87a1e51556fdcb46e5bd4f59dfb743061cf93c4d6831bf894f125eb57"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:68e78619a61ecf91e76aa3e6e8e33fc4894a2bebe93410754bd28fce0a8a4f9f"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:65c1a9bcdadc6c28eecee2c119465aebff8f7a584dd719facdd9e825ec61ab52"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:525808b8019e36eb524b8c68acdd63a37e75714eac50e988180b169d64480a00"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:962f82a3086483f5e5f64dbad880d31038b698494799b097bc59c2edf392fce6"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:aa7bd130efab1c280bed0f45501b7c8795f9fdbeb02e965371bbef3523627779"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:c9c804664ebe8f83a211cace637506669e7890fec1b4195b505c214e50dd4eb7"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-win32.whl", hash = "sha256:10bbfe99883db80bdbaff2dcf681dfc6533a614f700da1287707e8a5d78a8431"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-win_amd64.whl", hash = "sha256:1577735524cdad32f9f694208aa75e422adba74f1baee7551620e43a3141f559"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:ad9e82fb8f09ade1c3e1b996a6337afac2b8b9e365f926f5a61aacc71adc5b3c"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3c0fae6c3be832a0a0473ac912810b2877c8cb9d76ca48de1ed31e1c68386575"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b076b6226fb84157e3f7c971a47ff3a679d837cf338547532ab866c57930dbee"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bfce63a9e7834b12b87c64d6b155fdd9b3b96191b6bd334bf37db7ff1fe457f2"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:338ae27d6b8745585f87218a3f23f1512dbf52c26c28e322dbe54bcede54ccb9"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e4dd52d80b8c83fdce44e12478ad2e85c64ea965e75d66dbeafb0a3e77308fcc"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:df0be2b576a7abbf737b1575f048c23fb1d769f267ec4358296f31c2479db8f9"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:5bbe06f8eeafd38e5d0a4894ffec89378b6c6a625ff57e3028921f8ff59318ac"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-win32.whl", hash = "sha256:dd15ff04ffd7e05ffcb7fe79f1b98041b8ea30ae9234aed2a9168b5797c3effb"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-win_amd64.whl", hash = "sha256:134da1eca9ec0ae528110ccc9e48041e0828d79f24121a1a146161103c76e686"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:f698de3fd0c4e6972b92290a45bd9b1536bffe8c6759c62471efaa8acb4c37bc"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:aa57bd9cf8ae831a362185ee444e15a93ecb2e344c8e52e4d721ea3ab6ef1823"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffcc3f7c66b5f5b7931a5aa68fc9cecc51e685ef90282f4a82f0f5e9b704ad11"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:47d4f1c5f80fc62fdd7777d0d40a2e9dda0a05883ab11374334f6c4de38adffd"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1f67c7038d560d92149c060157d623c542173016c4babc0c1913cca0564b9939"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:9aad3c1755095ce347e26488214ef77e0485a3c34a50c5a5e2471dff60b9dd9c"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:14ff806850827afd6b07a5f32bd917fb7f45b046ba40c57abdb636674a8b559c"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8f9293864fe09b8149f0cc42ce56e3f0e54de883a9de90cd427f191c346eb2e1"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-win32.whl", hash = "sha256:715d3562f79d540f251b99ebd6d8baa547118974341db04f5ad06d5ea3eb8007"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-win_amd64.whl", hash = "sha256:1b8dd8c3fd14349433c79fa8abeb573a55fc0fdd769133baac1f5e07abf54aeb"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8e254ae696c88d98da6555f5ace2279cf7cd5b3f52be2b5cf97feafe883b58d2"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb0932dc158471523c9637e807d9bfb93e06a95cbf010f1a38b98623b929ef2b"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9402b03f1a1b4dc4c19845e5c749e3ab82d5078d16a2a4c2cd2df62d57bb0707"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ca379055a47383d02a5400cb0d110cef0a776fc644cda797db0c5696cfd7e18e"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:b7ff0f54cb4ff66dd38bebd335a38e2c22c41a8ee45aa608efc890ac3e3931bc"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:c011a4149cfbcf9f03994ec2edffcb8b1dc2d2aede7ca243746df97a5d41ce48"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:56d9f2ecac662ca1611d183feb03a3fa4406469dafe241673d521dd5ae92a155"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-win32.whl", hash = "sha256:8758846a7e80910096950b67071243da3e5a20ed2546e6392603c096778d48e0"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-win_amd64.whl", hash = "sha256:787003c0ddb00500e49a10f2844fac87aa6ce977b90b0feaaf9de23c22508b24"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:2ef12179d3a291be237280175b542c07a36e7f60718296278d8593d21ca937d4"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:2c1b19b3aaacc6e57b7e25710ff571c24d6c3613a45e905b1fde04d691b98ee0"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8afafd99945ead6e075b973fefa56379c5b5c53fd8937dad92c662da5d8fd5ee"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8c41976a29d078bb235fea9b2ecd3da465df42a562910f9022f1a03107bd02be"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d080e0a5eb2529460b30190fcfcc4199bd7f827663f858a226a81bc27beaa97e"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:69c0f17e9f5a7afdf2cc9fb2d1ce6aabdb3bafb7f38017c0b77862bcec2bbad8"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:504b320cd4b7eff6f968eddf81127112db685e81f7e36e75f9f84f0df46041c3"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:42de32b22b6b804f42c5d98be4f7e5e977ecdd9ee9b660fda1a3edf03b11792d"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-win32.whl", hash = "sha256:ceb01949af7121f9fc39f7d27f91be8546f3fb112c608bc4029aef0bab86a2a5"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-win_amd64.whl", hash = "sha256:1b40069d487e7edb2676d3fbdb2b0829ffa2cd63a2ec26c4938b2d34391b4ecc"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:8023faf4e01efadfa183e863fefde0046de576c6f14659e8782065bcece22198"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6b2b56950d93e41f33b4223ead100ea0fe11f8e6ee5f641eb753ce4b77a7042b"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9dcdfd0eaf283af041973bff14a2e143b8bd64e069f4c383416ecd79a81aab58"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:05fb21170423db021895e1ea1e1f3ab3adb85d1c2333cbc2310f2a26bc77272e"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:282c2cb35b5b673bbcadb33a585408104df04f14b2d9b01d4c345a3b92861c2c"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:ab4a0df41e7c16a1392727727e7998a467472d0ad65f3ad5e6e765015df08636"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:7ef3cb2ebbf91e330e3bb937efada0edd9003683db6b57bb108c4001f37a02ea"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:0a4e4a1aff6c7ac4cd55792abf96c915634c2b97e3cc1c7129578aa68ebd754e"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-win32.whl", hash = "sha256:fec21693218efe39aa7f8599346e90c705afa52c5b31ae019b2e57e8f6542bb2"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-win_amd64.whl", hash = "sha256:3fd4abcb888d15a94f32b75d8fd18ee162ca0c064f35b11134be77050296d6ba"}, + {file = "MarkupSafe-2.1.3.tar.gz", hash = "sha256:af598ed32d6ae86f1b747b82783958b1a4ab8f617b06fe68795c7f026abbdcad"}, +] + [[package]] name = "matplotlib" version = "3.7.3" @@ -926,6 +1240,56 @@ files = [ [package.dependencies] traitlets = "*" +[[package]] +name = "mcap" +version = "1.1.1" +description = "MCAP libraries for Python" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "mcap-1.1.1-py3-none-any.whl", hash = "sha256:d494988b76124b01de42089a02183b598f80e9bae3d44f5d30dd469752fa8b58"}, + {file = "mcap-1.1.1.tar.gz", hash = "sha256:699cde9f98350bece709a416591a2acb4ce5c0d8b07773b846f6b416bd993efa"}, +] + +[package.dependencies] +lz4 = "*" +zstandard = "*" + +[[package]] +name = "mcap-protobuf-support" +version = "0.4.1" +description = "Protobuf support for the Python MCAP library" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "mcap-protobuf-support-0.4.1.tar.gz", hash = "sha256:7b13619b067e718192df53f97fd6360529af1f4dd5da55c6bf3c2fc00fef5f3a"}, + {file = "mcap_protobuf_support-0.4.1-py3-none-any.whl", hash = "sha256:3e1da96851ab55b31664a7778bb7ab0bc38c8e9c2618444d3c535e8411900abf"}, +] + +[package.dependencies] +mcap = ">=0.0.14" +protobuf = ">=3.8,<4.22.0 || >=4.25.0" + +[[package]] +name = "mpmath" +version = "1.3.0" +description = "Python library for arbitrary-precision floating-point arithmetic" +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "mpmath-1.3.0-py3-none-any.whl", hash = "sha256:a0b2b9fe80bbcd81a6647ff13108738cfb482d481d826cc0e02f5b35e5c88d2c"}, + {file = "mpmath-1.3.0.tar.gz", hash = "sha256:7a28eb2a9774d00c7bc92411c19a89209d5da7c4c9a9e227be8330a23a25b91f"}, +] + +[package.extras] +develop = ["codecov", "pycodestyle", "pytest (>=4.6)", "pytest-cov", "wheel"] +docs = ["sphinx"] +gmpy = ["gmpy2 (>=2.1.0a4)"] +tests = ["pytest (>=4.6)"] + [[package]] name = "nest-asyncio" version = "1.5.8" @@ -938,6 +1302,25 @@ files = [ {file = "nest_asyncio-1.5.8.tar.gz", hash = "sha256:25aa2ca0d2a5b5531956b9e273b45cf664cae2b145101d73b86b199978d48fdb"}, ] +[[package]] +name = "networkx" +version = "3.1" +description = "Python package for creating and manipulating graphs and networks" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "networkx-3.1-py3-none-any.whl", hash = "sha256:4f33f68cb2afcf86f28a45f43efc27a9386b535d567d2127f8f61d51dec58d36"}, + {file = "networkx-3.1.tar.gz", hash = "sha256:de346335408f84de0eada6ff9fafafff9bcda11f0a0dfaa931133debb146ab61"}, +] + +[package.extras] +default = ["matplotlib (>=3.4)", "numpy (>=1.20)", "pandas (>=1.3)", "scipy (>=1.8)"] +developer = ["mypy (>=1.1)", "pre-commit (>=3.2)"] +doc = ["nb2plots (>=0.6)", "numpydoc (>=1.5)", "pillow (>=9.4)", "pydata-sphinx-theme (>=0.13)", "sphinx (>=6.1)", "sphinx-gallery (>=0.12)", "texext (>=0.6.7)"] +extra = ["lxml (>=4.6)", "pydot (>=1.4.2)", "pygraphviz (>=1.10)", "sympy (>=1.10)"] +test = ["codecov (>=2.1)", "pytest (>=7.2)", "pytest-cov (>=4.0)"] + [[package]] name = "nodeenv" version = "1.8.0" @@ -991,6 +1374,234 @@ files = [ {file = "numpy-1.24.4.tar.gz", hash = "sha256:80f5e3a4e498641401868df4208b74581206afbee7cf7b8329daae82676d9463"}, ] +[[package]] +name = "numpy" +version = "1.26.3" +description = "Fundamental package for array computing in Python" +category = "main" +optional = false +python-versions = ">=3.9" +files = [ + {file = "numpy-1.26.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:806dd64230dbbfaca8a27faa64e2f414bf1c6622ab78cc4264f7f5f028fee3bf"}, + {file = "numpy-1.26.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:02f98011ba4ab17f46f80f7f8f1c291ee7d855fcef0a5a98db80767a468c85cd"}, + {file = "numpy-1.26.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6d45b3ec2faed4baca41c76617fcdcfa4f684ff7a151ce6fc78ad3b6e85af0a6"}, + {file = "numpy-1.26.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bdd2b45bf079d9ad90377048e2747a0c82351989a2165821f0c96831b4a2a54b"}, + {file = "numpy-1.26.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:211ddd1e94817ed2d175b60b6374120244a4dd2287f4ece45d49228b4d529178"}, + {file = "numpy-1.26.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:b1240f767f69d7c4c8a29adde2310b871153df9b26b5cb2b54a561ac85146485"}, + {file = "numpy-1.26.3-cp310-cp310-win32.whl", hash = "sha256:21a9484e75ad018974a2fdaa216524d64ed4212e418e0a551a2d83403b0531d3"}, + {file = "numpy-1.26.3-cp310-cp310-win_amd64.whl", hash = "sha256:9e1591f6ae98bcfac2a4bbf9221c0b92ab49762228f38287f6eeb5f3f55905ce"}, + {file = "numpy-1.26.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b831295e5472954104ecb46cd98c08b98b49c69fdb7040483aff799a755a7374"}, + {file = "numpy-1.26.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:9e87562b91f68dd8b1c39149d0323b42e0082db7ddb8e934ab4c292094d575d6"}, + {file = "numpy-1.26.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8c66d6fec467e8c0f975818c1796d25c53521124b7cfb760114be0abad53a0a2"}, + {file = "numpy-1.26.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f25e2811a9c932e43943a2615e65fc487a0b6b49218899e62e426e7f0a57eeda"}, + {file = "numpy-1.26.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:af36e0aa45e25c9f57bf684b1175e59ea05d9a7d3e8e87b7ae1a1da246f2767e"}, + {file = "numpy-1.26.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:51c7f1b344f302067b02e0f5b5d2daa9ed4a721cf49f070280ac202738ea7f00"}, + {file = "numpy-1.26.3-cp311-cp311-win32.whl", hash = "sha256:7ca4f24341df071877849eb2034948459ce3a07915c2734f1abb4018d9c49d7b"}, + {file = "numpy-1.26.3-cp311-cp311-win_amd64.whl", hash = "sha256:39763aee6dfdd4878032361b30b2b12593fb445ddb66bbac802e2113eb8a6ac4"}, + {file = "numpy-1.26.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:a7081fd19a6d573e1a05e600c82a1c421011db7935ed0d5c483e9dd96b99cf13"}, + {file = "numpy-1.26.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:12c70ac274b32bc00c7f61b515126c9205323703abb99cd41836e8125ea0043e"}, + {file = "numpy-1.26.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7f784e13e598e9594750b2ef6729bcd5a47f6cfe4a12cca13def35e06d8163e3"}, + {file = "numpy-1.26.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5f24750ef94d56ce6e33e4019a8a4d68cfdb1ef661a52cdaee628a56d2437419"}, + {file = "numpy-1.26.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:77810ef29e0fb1d289d225cabb9ee6cf4d11978a00bb99f7f8ec2132a84e0166"}, + {file = "numpy-1.26.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8ed07a90f5450d99dad60d3799f9c03c6566709bd53b497eb9ccad9a55867f36"}, + {file = "numpy-1.26.3-cp312-cp312-win32.whl", hash = "sha256:f73497e8c38295aaa4741bdfa4fda1a5aedda5473074369eca10626835445511"}, + {file = "numpy-1.26.3-cp312-cp312-win_amd64.whl", hash = "sha256:da4b0c6c699a0ad73c810736303f7fbae483bcb012e38d7eb06a5e3b432c981b"}, + {file = "numpy-1.26.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:1666f634cb3c80ccbd77ec97bc17337718f56d6658acf5d3b906ca03e90ce87f"}, + {file = "numpy-1.26.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:18c3319a7d39b2c6a9e3bb75aab2304ab79a811ac0168a671a62e6346c29b03f"}, + {file = "numpy-1.26.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0b7e807d6888da0db6e7e75838444d62495e2b588b99e90dd80c3459594e857b"}, + {file = "numpy-1.26.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b4d362e17bcb0011738c2d83e0a65ea8ce627057b2fdda37678f4374a382a137"}, + {file = "numpy-1.26.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:b8c275f0ae90069496068c714387b4a0eba5d531aace269559ff2b43655edd58"}, + {file = "numpy-1.26.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:cc0743f0302b94f397a4a65a660d4cd24267439eb16493fb3caad2e4389bccbb"}, + {file = "numpy-1.26.3-cp39-cp39-win32.whl", hash = "sha256:9bc6d1a7f8cedd519c4b7b1156d98e051b726bf160715b769106661d567b3f03"}, + {file = "numpy-1.26.3-cp39-cp39-win_amd64.whl", hash = "sha256:867e3644e208c8922a3be26fc6bbf112a035f50f0a86497f98f228c50c607bb2"}, + {file = "numpy-1.26.3-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:3c67423b3703f8fbd90f5adaa37f85b5794d3366948efe9a5190a5f3a83fc34e"}, + {file = "numpy-1.26.3-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:46f47ee566d98849323f01b349d58f2557f02167ee301e5e28809a8c0e27a2d0"}, + {file = "numpy-1.26.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:a8474703bffc65ca15853d5fd4d06b18138ae90c17c8d12169968e998e448bb5"}, + {file = "numpy-1.26.3.tar.gz", hash = "sha256:697df43e2b6310ecc9d95f05d5ef20eacc09c7c4ecc9da3f235d39e71b7da1e4"}, +] + +[[package]] +name = "nvidia-cublas-cu12" +version = "12.1.3.1" +description = "CUBLAS native runtime libraries" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_cublas_cu12-12.1.3.1-py3-none-manylinux1_x86_64.whl", hash = "sha256:ee53ccca76a6fc08fb9701aa95b6ceb242cdaab118c3bb152af4e579af792728"}, + {file = "nvidia_cublas_cu12-12.1.3.1-py3-none-win_amd64.whl", hash = "sha256:2b964d60e8cf11b5e1073d179d85fa340c120e99b3067558f3cf98dd69d02906"}, +] + +[[package]] +name = "nvidia-cuda-cupti-cu12" +version = "12.1.105" +description = "CUDA profiling tools runtime libs." +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_cuda_cupti_cu12-12.1.105-py3-none-manylinux1_x86_64.whl", hash = "sha256:e54fde3983165c624cb79254ae9818a456eb6e87a7fd4d56a2352c24ee542d7e"}, + {file = "nvidia_cuda_cupti_cu12-12.1.105-py3-none-win_amd64.whl", hash = "sha256:bea8236d13a0ac7190bd2919c3e8e6ce1e402104276e6f9694479e48bb0eb2a4"}, +] + +[[package]] +name = "nvidia-cuda-nvrtc-cu12" +version = "12.1.105" +description = "NVRTC native runtime libraries" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_cuda_nvrtc_cu12-12.1.105-py3-none-manylinux1_x86_64.whl", hash = "sha256:339b385f50c309763ca65456ec75e17bbefcbbf2893f462cb8b90584cd27a1c2"}, + {file = "nvidia_cuda_nvrtc_cu12-12.1.105-py3-none-win_amd64.whl", hash = "sha256:0a98a522d9ff138b96c010a65e145dc1b4850e9ecb75a0172371793752fd46ed"}, +] + +[[package]] +name = "nvidia-cuda-runtime-cu12" +version = "12.1.105" +description = "CUDA Runtime native Libraries" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_cuda_runtime_cu12-12.1.105-py3-none-manylinux1_x86_64.whl", hash = "sha256:6e258468ddf5796e25f1dc591a31029fa317d97a0a94ed93468fc86301d61e40"}, + {file = "nvidia_cuda_runtime_cu12-12.1.105-py3-none-win_amd64.whl", hash = "sha256:dfb46ef84d73fababab44cf03e3b83f80700d27ca300e537f85f636fac474344"}, +] + +[[package]] +name = "nvidia-cudnn-cu12" +version = "8.9.2.26" +description = "cuDNN runtime libraries" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_cudnn_cu12-8.9.2.26-py3-none-manylinux1_x86_64.whl", hash = "sha256:5ccb288774fdfb07a7e7025ffec286971c06d8d7b4fb162525334616d7629ff9"}, +] + +[package.dependencies] +nvidia-cublas-cu12 = "*" + +[[package]] +name = "nvidia-cufft-cu12" +version = "11.0.2.54" +description = "CUFFT native runtime libraries" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_cufft_cu12-11.0.2.54-py3-none-manylinux1_x86_64.whl", hash = "sha256:794e3948a1aa71fd817c3775866943936774d1c14e7628c74f6f7417224cdf56"}, + {file = "nvidia_cufft_cu12-11.0.2.54-py3-none-win_amd64.whl", hash = "sha256:d9ac353f78ff89951da4af698f80870b1534ed69993f10a4cf1d96f21357e253"}, +] + +[[package]] +name = "nvidia-curand-cu12" +version = "10.3.2.106" +description = "CURAND native runtime libraries" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_curand_cu12-10.3.2.106-py3-none-manylinux1_x86_64.whl", hash = "sha256:9d264c5036dde4e64f1de8c50ae753237c12e0b1348738169cd0f8a536c0e1e0"}, + {file = "nvidia_curand_cu12-10.3.2.106-py3-none-win_amd64.whl", hash = "sha256:75b6b0c574c0037839121317e17fd01f8a69fd2ef8e25853d826fec30bdba74a"}, +] + +[[package]] +name = "nvidia-cusolver-cu12" +version = "11.4.5.107" +description = "CUDA solver native runtime libraries" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_cusolver_cu12-11.4.5.107-py3-none-manylinux1_x86_64.whl", hash = "sha256:8a7ec542f0412294b15072fa7dab71d31334014a69f953004ea7a118206fe0dd"}, + {file = "nvidia_cusolver_cu12-11.4.5.107-py3-none-win_amd64.whl", hash = "sha256:74e0c3a24c78612192a74fcd90dd117f1cf21dea4822e66d89e8ea80e3cd2da5"}, +] + +[package.dependencies] +nvidia-cublas-cu12 = "*" +nvidia-cusparse-cu12 = "*" +nvidia-nvjitlink-cu12 = "*" + +[[package]] +name = "nvidia-cusparse-cu12" +version = "12.1.0.106" +description = "CUSPARSE native runtime libraries" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_cusparse_cu12-12.1.0.106-py3-none-manylinux1_x86_64.whl", hash = "sha256:f3b50f42cf363f86ab21f720998517a659a48131e8d538dc02f8768237bd884c"}, + {file = "nvidia_cusparse_cu12-12.1.0.106-py3-none-win_amd64.whl", hash = "sha256:b798237e81b9719373e8fae8d4f091b70a0cf09d9d85c95a557e11df2d8e9a5a"}, +] + +[package.dependencies] +nvidia-nvjitlink-cu12 = "*" + +[[package]] +name = "nvidia-nccl-cu12" +version = "2.18.1" +description = "NVIDIA Collective Communication Library (NCCL) Runtime" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_nccl_cu12-2.18.1-py3-none-manylinux1_x86_64.whl", hash = "sha256:1a6c4acefcbebfa6de320f412bf7866de856e786e0462326ba1bac40de0b5e71"}, +] + +[[package]] +name = "nvidia-nvjitlink-cu12" +version = "12.3.101" +description = "Nvidia JIT LTO Library" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_nvjitlink_cu12-12.3.101-py3-none-manylinux1_x86_64.whl", hash = "sha256:64335a8088e2b9d196ae8665430bc6a2b7e6ef2eb877a9c735c804bd4ff6467c"}, + {file = "nvidia_nvjitlink_cu12-12.3.101-py3-none-win_amd64.whl", hash = "sha256:1b2e317e437433753530792f13eece58f0aec21a2b05903be7bffe58a606cbd1"}, +] + +[[package]] +name = "nvidia-nvtx-cu12" +version = "12.1.105" +description = "NVIDIA Tools Extension" +category = "main" +optional = false +python-versions = ">=3" +files = [ + {file = "nvidia_nvtx_cu12-12.1.105-py3-none-manylinux1_x86_64.whl", hash = "sha256:dc21cf308ca5691e7c04d962e213f8a4aa9bbfa23d95412f452254c2caeb09e5"}, + {file = "nvidia_nvtx_cu12-12.1.105-py3-none-win_amd64.whl", hash = "sha256:65f4d98982b31b60026e0e6de73fbdfc09d08a96f4656dd3665ca616a11e1e82"}, +] + +[[package]] +name = "opencv-python" +version = "4.9.0.80" +description = "Wrapper package for OpenCV python bindings." +category = "main" +optional = false +python-versions = ">=3.6" +files = [ + {file = "opencv-python-4.9.0.80.tar.gz", hash = "sha256:1a9f0e6267de3a1a1db0c54213d022c7c8b5b9ca4b580e80bdc58516c922c9e1"}, + {file = "opencv_python-4.9.0.80-cp37-abi3-macosx_10_16_x86_64.whl", hash = "sha256:7e5f7aa4486651a6ebfa8ed4b594b65bd2d2f41beeb4241a3e4b1b85acbbbadb"}, + {file = "opencv_python-4.9.0.80-cp37-abi3-macosx_11_0_arm64.whl", hash = "sha256:71dfb9555ccccdd77305fc3dcca5897fbf0cf28b297c51ee55e079c065d812a3"}, + {file = "opencv_python-4.9.0.80-cp37-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7b34a52e9da36dda8c151c6394aed602e4b17fa041df0b9f5b93ae10b0fcca2a"}, + {file = "opencv_python-4.9.0.80-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e4088cab82b66a3b37ffc452976b14a3c599269c247895ae9ceb4066d8188a57"}, + {file = "opencv_python-4.9.0.80-cp37-abi3-win32.whl", hash = "sha256:dcf000c36dd1651118a2462257e3a9e76db789a78432e1f303c7bac54f63ef6c"}, + {file = "opencv_python-4.9.0.80-cp37-abi3-win_amd64.whl", hash = "sha256:3f16f08e02b2a2da44259c7cc712e779eff1dd8b55fdb0323e8cab09548086c0"}, +] + +[package.dependencies] +numpy = [ + {version = ">=1.21.0", markers = "python_version <= \"3.9\" and platform_system == \"Darwin\" and platform_machine == \"arm64\""}, + {version = ">=1.21.2", markers = "python_version >= \"3.10\""}, + {version = ">=1.21.4", markers = "python_version >= \"3.10\" and platform_system == \"Darwin\""}, + {version = ">=1.23.5", markers = "python_version >= \"3.11\""}, + {version = ">=1.26.0", markers = "python_version >= \"3.12\""}, + {version = ">=1.19.3", markers = "python_version >= \"3.6\" and platform_system == \"Linux\" and platform_machine == \"aarch64\" or python_version >= \"3.9\""}, + {version = ">=1.17.0", markers = "python_version >= \"3.7\""}, + {version = ">=1.17.3", markers = "python_version >= \"3.8\""}, +] + [[package]] name = "packaging" version = "23.2" @@ -1003,6 +1614,74 @@ files = [ {file = "packaging-23.2.tar.gz", hash = "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5"}, ] +[[package]] +name = "pandas" +version = "2.0.3" +description = "Powerful data structures for data analysis, time series, and statistics" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pandas-2.0.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:e4c7c9f27a4185304c7caf96dc7d91bc60bc162221152de697c98eb0b2648dd8"}, + {file = "pandas-2.0.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:f167beed68918d62bffb6ec64f2e1d8a7d297a038f86d4aed056b9493fca407f"}, + {file = "pandas-2.0.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ce0c6f76a0f1ba361551f3e6dceaff06bde7514a374aa43e33b588ec10420183"}, + {file = "pandas-2.0.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba619e410a21d8c387a1ea6e8a0e49bb42216474436245718d7f2e88a2f8d7c0"}, + {file = "pandas-2.0.3-cp310-cp310-win32.whl", hash = "sha256:3ef285093b4fe5058eefd756100a367f27029913760773c8bf1d2d8bebe5d210"}, + {file = "pandas-2.0.3-cp310-cp310-win_amd64.whl", hash = "sha256:9ee1a69328d5c36c98d8e74db06f4ad518a1840e8ccb94a4ba86920986bb617e"}, + {file = "pandas-2.0.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b084b91d8d66ab19f5bb3256cbd5ea661848338301940e17f4492b2ce0801fe8"}, + {file = "pandas-2.0.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:37673e3bdf1551b95bf5d4ce372b37770f9529743d2498032439371fc7b7eb26"}, + {file = "pandas-2.0.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b9cb1e14fdb546396b7e1b923ffaeeac24e4cedd14266c3497216dd4448e4f2d"}, + {file = "pandas-2.0.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d9cd88488cceb7635aebb84809d087468eb33551097d600c6dad13602029c2df"}, + {file = "pandas-2.0.3-cp311-cp311-win32.whl", hash = "sha256:694888a81198786f0e164ee3a581df7d505024fbb1f15202fc7db88a71d84ebd"}, + {file = "pandas-2.0.3-cp311-cp311-win_amd64.whl", hash = "sha256:6a21ab5c89dcbd57f78d0ae16630b090eec626360085a4148693def5452d8a6b"}, + {file = "pandas-2.0.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:9e4da0d45e7f34c069fe4d522359df7d23badf83abc1d1cef398895822d11061"}, + {file = "pandas-2.0.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:32fca2ee1b0d93dd71d979726b12b61faa06aeb93cf77468776287f41ff8fdc5"}, + {file = "pandas-2.0.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:258d3624b3ae734490e4d63c430256e716f488c4fcb7c8e9bde2d3aa46c29089"}, + {file = "pandas-2.0.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9eae3dc34fa1aa7772dd3fc60270d13ced7346fcbcfee017d3132ec625e23bb0"}, + {file = "pandas-2.0.3-cp38-cp38-win32.whl", hash = "sha256:f3421a7afb1a43f7e38e82e844e2bca9a6d793d66c1a7f9f0ff39a795bbc5e02"}, + {file = "pandas-2.0.3-cp38-cp38-win_amd64.whl", hash = "sha256:69d7f3884c95da3a31ef82b7618af5710dba95bb885ffab339aad925c3e8ce78"}, + {file = "pandas-2.0.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5247fb1ba347c1261cbbf0fcfba4a3121fbb4029d95d9ef4dc45406620b25c8b"}, + {file = "pandas-2.0.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:81af086f4543c9d8bb128328b5d32e9986e0c84d3ee673a2ac6fb57fd14f755e"}, + {file = "pandas-2.0.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1994c789bf12a7c5098277fb43836ce090f1073858c10f9220998ac74f37c69b"}, + {file = "pandas-2.0.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5ec591c48e29226bcbb316e0c1e9423622bc7a4eaf1ef7c3c9fa1a3981f89641"}, + {file = "pandas-2.0.3-cp39-cp39-win32.whl", hash = "sha256:04dbdbaf2e4d46ca8da896e1805bc04eb85caa9a82e259e8eed00254d5e0c682"}, + {file = "pandas-2.0.3-cp39-cp39-win_amd64.whl", hash = "sha256:1168574b036cd8b93abc746171c9b4f1b83467438a5e45909fed645cf8692dbc"}, + {file = "pandas-2.0.3.tar.gz", hash = "sha256:c02f372a88e0d17f36d3093a644c73cfc1788e876a7c4bcb4020a77512e2043c"}, +] + +[package.dependencies] +numpy = [ + {version = ">=1.20.3", markers = "python_version < \"3.10\""}, + {version = ">=1.21.0", markers = "python_version >= \"3.10\""}, + {version = ">=1.23.2", markers = "python_version >= \"3.11\""}, +] +python-dateutil = ">=2.8.2" +pytz = ">=2020.1" +tzdata = ">=2022.1" + +[package.extras] +all = ["PyQt5 (>=5.15.1)", "SQLAlchemy (>=1.4.16)", "beautifulsoup4 (>=4.9.3)", "bottleneck (>=1.3.2)", "brotlipy (>=0.7.0)", "fastparquet (>=0.6.3)", "fsspec (>=2021.07.0)", "gcsfs (>=2021.07.0)", "html5lib (>=1.1)", "hypothesis (>=6.34.2)", "jinja2 (>=3.0.0)", "lxml (>=4.6.3)", "matplotlib (>=3.6.1)", "numba (>=0.53.1)", "numexpr (>=2.7.3)", "odfpy (>=1.4.1)", "openpyxl (>=3.0.7)", "pandas-gbq (>=0.15.0)", "psycopg2 (>=2.8.6)", "pyarrow (>=7.0.0)", "pymysql (>=1.0.2)", "pyreadstat (>=1.1.2)", "pytest (>=7.3.2)", "pytest-asyncio (>=0.17.0)", "pytest-xdist (>=2.2.0)", "python-snappy (>=0.6.0)", "pyxlsb (>=1.0.8)", "qtpy (>=2.2.0)", "s3fs (>=2021.08.0)", "scipy (>=1.7.1)", "tables (>=3.6.1)", "tabulate (>=0.8.9)", "xarray (>=0.21.0)", "xlrd (>=2.0.1)", "xlsxwriter (>=1.4.3)", "zstandard (>=0.15.2)"] +aws = ["s3fs (>=2021.08.0)"] +clipboard = ["PyQt5 (>=5.15.1)", "qtpy (>=2.2.0)"] +compression = ["brotlipy (>=0.7.0)", "python-snappy (>=0.6.0)", "zstandard (>=0.15.2)"] +computation = ["scipy (>=1.7.1)", "xarray (>=0.21.0)"] +excel = ["odfpy (>=1.4.1)", "openpyxl (>=3.0.7)", "pyxlsb (>=1.0.8)", "xlrd (>=2.0.1)", "xlsxwriter (>=1.4.3)"] +feather = ["pyarrow (>=7.0.0)"] +fss = ["fsspec (>=2021.07.0)"] +gcp = ["gcsfs (>=2021.07.0)", "pandas-gbq (>=0.15.0)"] +hdf5 = ["tables (>=3.6.1)"] +html = ["beautifulsoup4 (>=4.9.3)", "html5lib (>=1.1)", "lxml (>=4.6.3)"] +mysql = ["SQLAlchemy (>=1.4.16)", "pymysql (>=1.0.2)"] +output-formatting = ["jinja2 (>=3.0.0)", "tabulate (>=0.8.9)"] +parquet = ["pyarrow (>=7.0.0)"] +performance = ["bottleneck (>=1.3.2)", "numba (>=0.53.1)", "numexpr (>=2.7.1)"] +plot = ["matplotlib (>=3.6.1)"] +postgresql = ["SQLAlchemy (>=1.4.16)", "psycopg2 (>=2.8.6)"] +spss = ["pyreadstat (>=1.1.2)"] +sql-other = ["SQLAlchemy (>=1.4.16)"] +test = ["hypothesis (>=6.34.2)", "pytest (>=7.3.2)", "pytest-asyncio (>=0.17.0)", "pytest-xdist (>=2.2.0)"] +xml = ["lxml (>=4.6.3)"] + [[package]] name = "parso" version = "0.8.3" @@ -1180,6 +1859,27 @@ files = [ [package.dependencies] wcwidth = "*" +[[package]] +name = "protobuf" +version = "4.25.2" +description = "" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "protobuf-4.25.2-cp310-abi3-win32.whl", hash = "sha256:b50c949608682b12efb0b2717f53256f03636af5f60ac0c1d900df6213910fd6"}, + {file = "protobuf-4.25.2-cp310-abi3-win_amd64.whl", hash = "sha256:8f62574857ee1de9f770baf04dde4165e30b15ad97ba03ceac65f760ff018ac9"}, + {file = "protobuf-4.25.2-cp37-abi3-macosx_10_9_universal2.whl", hash = "sha256:2db9f8fa64fbdcdc93767d3cf81e0f2aef176284071507e3ede160811502fd3d"}, + {file = "protobuf-4.25.2-cp37-abi3-manylinux2014_aarch64.whl", hash = "sha256:10894a2885b7175d3984f2be8d9850712c57d5e7587a2410720af8be56cdaf62"}, + {file = "protobuf-4.25.2-cp37-abi3-manylinux2014_x86_64.whl", hash = "sha256:fc381d1dd0516343f1440019cedf08a7405f791cd49eef4ae1ea06520bc1c020"}, + {file = "protobuf-4.25.2-cp38-cp38-win32.whl", hash = "sha256:33a1aeef4b1927431d1be780e87b641e322b88d654203a9e9d93f218ee359e61"}, + {file = "protobuf-4.25.2-cp38-cp38-win_amd64.whl", hash = "sha256:47f3de503fe7c1245f6f03bea7e8d3ec11c6c4a2ea9ef910e3221c8a15516d62"}, + {file = "protobuf-4.25.2-cp39-cp39-win32.whl", hash = "sha256:5e5c933b4c30a988b52e0b7c02641760a5ba046edc5e43d3b94a74c9fc57c1b3"}, + {file = "protobuf-4.25.2-cp39-cp39-win_amd64.whl", hash = "sha256:d66a769b8d687df9024f2985d5137a337f957a0916cf5464d1513eee96a63ff0"}, + {file = "protobuf-4.25.2-py3-none-any.whl", hash = "sha256:a8b7a98d4ce823303145bf3c1a8bdb0f2f4642a414b196f04ad9853ed0c8f830"}, + {file = "protobuf-4.25.2.tar.gz", hash = "sha256:fe599e175cb347efc8ee524bcd4b902d11f7262c0e569ececcb89995c15f0a5e"}, +] + [[package]] name = "psutil" version = "5.9.6" @@ -1248,6 +1948,73 @@ files = [ {file = "pycparser-2.21.tar.gz", hash = "sha256:e644fdec12f7872f86c58ff790da456218b10f863970249516d60a5eaca77206"}, ] +[[package]] +name = "pygame" +version = "2.5.2" +description = "Python Game Development" +category = "main" +optional = false +python-versions = ">=3.6" +files = [ + {file = "pygame-2.5.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:a0769eb628c818761755eb0a0ca8216b95270ea8cbcbc82227e39ac9644643da"}, + {file = "pygame-2.5.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:ed9a3d98adafa0805ccbaaff5d2996a2b5795381285d8437a4a5d248dbd12b4a"}, + {file = "pygame-2.5.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f30d1618672a55e8c6669281ba264464b3ab563158e40d89e8c8b3faa0febebd"}, + {file = "pygame-2.5.2-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:39690e9be9baf58b7359d1f3b2336e1fd6f92fedbbce42987be5df27f8d30718"}, + {file = "pygame-2.5.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:03879ec299c9f4ba23901b2649a96b2143f0a5d787f0b6c39469989e2320caf1"}, + {file = "pygame-2.5.2-cp310-cp310-win32.whl", hash = "sha256:74e1d6284100e294f445832e6f6343be4fe4748decc4f8a51131ae197dae8584"}, + {file = "pygame-2.5.2-cp310-cp310-win_amd64.whl", hash = "sha256:485239c7d32265fd35b76ae8f64f34b0637ae11e69d76de15710c4b9edcc7c8d"}, + {file = "pygame-2.5.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:34646ca20e163dc6f6cf8170f1e12a2e41726780112594ac061fa448cf7ccd75"}, + {file = "pygame-2.5.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3b8a6e351665ed26ea791f0e1fd649d3f483e8681892caef9d471f488f9ea5ee"}, + {file = "pygame-2.5.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:dc346965847aef00013fa2364f41a64f068cd096dcc7778fc306ca3735f0eedf"}, + {file = "pygame-2.5.2-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:35632035fd81261f2d797fa810ea8c46111bd78ceb6089d52b61ed7dc3c5d05f"}, + {file = "pygame-2.5.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0e24d05184e4195fe5ebcdce8b18ecb086f00182b9ae460a86682d312ce8d31f"}, + {file = "pygame-2.5.2-cp311-cp311-win32.whl", hash = "sha256:f02c1c7505af18d426d355ac9872bd5c916b27f7b0fe224749930662bea47a50"}, + {file = "pygame-2.5.2-cp311-cp311-win_amd64.whl", hash = "sha256:6d58c8cf937815d3b7cdc0fa9590c5129cb2c9658b72d00e8a4568dea2ff1d42"}, + {file = "pygame-2.5.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:1a2a43802bb5e89ce2b3b775744e78db4f9a201bf8d059b946c61722840ceea8"}, + {file = "pygame-2.5.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:1c289f2613c44fe70a1e40769de4a49c5ab5a29b9376f1692bb1a15c9c1c9bfa"}, + {file = "pygame-2.5.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:074aa6c6e110c925f7f27f00c7733c6303407edc61d738882985091d1eb2ef17"}, + {file = "pygame-2.5.2-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:fe0228501ec616779a0b9c4299e837877783e18df294dd690b9ab0eed3d8aaab"}, + {file = "pygame-2.5.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:31648d38ecdc2335ffc0e38fb18a84b3339730521505dac68514f83a1092e3f4"}, + {file = "pygame-2.5.2-cp312-cp312-win32.whl", hash = "sha256:224c308856334bc792f696e9278e50d099a87c116f7fc314cd6aa3ff99d21592"}, + {file = "pygame-2.5.2-cp312-cp312-win_amd64.whl", hash = "sha256:dd2d2650faf54f9a0f5bd0db8409f79609319725f8f08af6507a0609deadcad4"}, + {file = "pygame-2.5.2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:9b30bc1220c457169571aac998e54b013aaeb732d2fd8744966cb1cfab1f61d1"}, + {file = "pygame-2.5.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:78fcd7643358b886a44127ff7dec9041c056c212b3a98977674f83f99e9b12d3"}, + {file = "pygame-2.5.2-cp36-cp36m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:35cf093a51cb294ede56c29d4acf41538c00f297fcf78a9b186fb7d23c0577b6"}, + {file = "pygame-2.5.2-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6fe323acbf53a0195c8c98b1b941eba7ac24e3e2b28ae48e8cda566f15fc4945"}, + {file = "pygame-2.5.2-cp36-cp36m-win32.whl", hash = "sha256:5697528266b4716d9cdd44a5a1d210f4d86ef801d0f64ca5da5d0816704009d9"}, + {file = "pygame-2.5.2-cp36-cp36m-win_amd64.whl", hash = "sha256:edda1f7cff4806a4fa39e0e8ccd75f38d1d340fa5fc52d8582ade87aca247d92"}, + {file = "pygame-2.5.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:9bd738fd4ecc224769d0b4a719f96900a86578e26e0105193658a32966df2aae"}, + {file = "pygame-2.5.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:30a8d7cf12363b4140bf2f93b5eec4028376ca1d0fe4b550588f836279485308"}, + {file = "pygame-2.5.2-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:bc12e4dea3e88ea8a553de6d56a37b704dbe2aed95105889f6afeb4b96e62097"}, + {file = "pygame-2.5.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2b34c73cb328024f8db3cb6487a37e54000148988275d8d6e5adf99d9323c937"}, + {file = "pygame-2.5.2-cp37-cp37m-win32.whl", hash = "sha256:7d0a2794649defa57ef50b096a99f7113d3d0c2e32d1426cafa7d618eadce4c7"}, + {file = "pygame-2.5.2-cp37-cp37m-win_amd64.whl", hash = "sha256:41f8779f52e0f6e6e6ccb8f0b5536e432bf386ee29c721a1c22cada7767b0cef"}, + {file = "pygame-2.5.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:677e37bc0ea7afd89dde5a88ced4458aa8656159c70a576eea68b5622ee1997b"}, + {file = "pygame-2.5.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:47a8415d2bd60e6909823b5643a1d4ef5cc29417d817f2a214b255f6fa3a1e4c"}, + {file = "pygame-2.5.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ff21201df6278b8ca2e948fb148ffe88f5481fd03760f381dd61e45954c7dff"}, + {file = "pygame-2.5.2-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d29a84b2e02814b9ba925357fd2e1df78efe5e1aa64dc3051eaed95d2b96eafd"}, + {file = "pygame-2.5.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d78485c4d21133d6b2fbb504cd544ca655e50b6eb551d2995b3aa6035928adda"}, + {file = "pygame-2.5.2-cp38-cp38-win32.whl", hash = "sha256:d851247239548aa357c4a6840fb67adc2d570ce7cb56988d036a723d26b48bff"}, + {file = "pygame-2.5.2-cp38-cp38-win_amd64.whl", hash = "sha256:88d1cdacc2d3471eceab98bf0c93c14d3a8461f93e58e3d926f20d4de3a75554"}, + {file = "pygame-2.5.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:4f1559e7efe4efb9dc19d2d811d702f325d9605f9f6f9ececa39ee6890c798f5"}, + {file = "pygame-2.5.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:cf2191b756ceb0e8458a761d0c665b0c70b538570449e0d39b75a5ba94ac5cf0"}, + {file = "pygame-2.5.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6cf2257447ce7f2d6de37e5fb019d2bbe32ed05a5721ace8bc78c2d9beaf3aee"}, + {file = "pygame-2.5.2-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d75cbbfaba2b81434d62631d0b08b85fab16cf4a36e40b80298d3868927e1299"}, + {file = "pygame-2.5.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:daca456d5b9f52e088e06a127dec182b3638a775684fb2260f25d664351cf1ae"}, + {file = "pygame-2.5.2-cp39-cp39-win32.whl", hash = "sha256:3b3e619e33d11c297d7a57a82db40681f9c2c3ae1d5bf06003520b4fe30c435d"}, + {file = "pygame-2.5.2-cp39-cp39-win_amd64.whl", hash = "sha256:1822d534bb7fe756804647b6da2c9ea5d7a62d8796b2e15d172d3be085de28c6"}, + {file = "pygame-2.5.2-pp36-pypy36_pp73-win32.whl", hash = "sha256:e708fc8f709a0fe1d1876489345f2e443d47f3976d33455e2e1e937f972f8677"}, + {file = "pygame-2.5.2-pp37-pypy37_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c13edebc43c240fb0532969e914f0ccefff5ae7e50b0b788d08ad2c15ef793e4"}, + {file = "pygame-2.5.2-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:263b4a7cbfc9fe2055abc21b0251cc17dea6dff750f0e1c598919ff350cdbffe"}, + {file = "pygame-2.5.2-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:e58e2b0c791041e4bccafa5bd7650623ba1592b8fe62ae0a276b7d0ecb314b6c"}, + {file = "pygame-2.5.2-pp38-pypy38_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a0bd67426c02ffe6c9827fc4bcbda9442fbc451d29b17c83a3c088c56fef2c90"}, + {file = "pygame-2.5.2-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9dcff6cbba1584cf7732ce1dbdd044406cd4f6e296d13bcb7fba963fb4aeefc9"}, + {file = "pygame-2.5.2-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:ce4b6c0bfe44d00bb0998a6517bd0cf9455f642f30f91bc671ad41c05bf6f6ae"}, + {file = "pygame-2.5.2-pp39-pypy39_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:68c4e8e60b725ffc7a6c6ecd9bb5fcc5ed2d6e0e2a2c4a29a8454856ef16ad63"}, + {file = "pygame-2.5.2-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1f3849f97372a3381c66955f99a0d58485ccd513c3d00c030b869094ce6997a6"}, + {file = "pygame-2.5.2.tar.gz", hash = "sha256:c1b89eb5d539e7ac5cf75513125fb5f2f0a2d918b1fd6e981f23bf0ac1b1c24a"}, +] + [[package]] name = "pygments" version = "2.16.1" @@ -1331,6 +2098,18 @@ files = [ [package.dependencies] six = ">=1.5" +[[package]] +name = "pytz" +version = "2023.3.post1" +description = "World timezone definitions, modern and historical" +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "pytz-2023.3.post1-py2.py3-none-any.whl", hash = "sha256:ce42d816b81b68506614c11e8937d3aa9e41007ceb50bfdcb0749b921bf646c7"}, + {file = "pytz-2023.3.post1.tar.gz", hash = "sha256:7b4fddbeb94a1eba4b557da24f19fdf9db575192544270a9101d8509f9f43d7b"}, +] + [[package]] name = "pywin32" version = "306" @@ -1521,6 +2300,218 @@ files = [ [package.dependencies] cffi = {version = "*", markers = "implementation_name == \"pypy\""} +[[package]] +name = "requests" +version = "2.31.0" +description = "Python HTTP for Humans." +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "requests-2.31.0-py3-none-any.whl", hash = "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f"}, + {file = "requests-2.31.0.tar.gz", hash = "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1"}, +] + +[package.dependencies] +certifi = ">=2017.4.17" +charset-normalizer = ">=2,<4" +idna = ">=2.5,<4" +urllib3 = ">=1.21.1,<3" + +[package.extras] +socks = ["PySocks (>=1.5.6,!=1.5.7)"] +use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] + +[[package]] +name = "scipy" +version = "1.11.4" +description = "Fundamental algorithms for scientific computing in Python" +category = "main" +optional = false +python-versions = ">=3.9" +files = [ + {file = "scipy-1.11.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:bc9a714581f561af0848e6b69947fda0614915f072dfd14142ed1bfe1b806710"}, + {file = "scipy-1.11.4-cp310-cp310-macosx_12_0_arm64.whl", hash = "sha256:cf00bd2b1b0211888d4dc75656c0412213a8b25e80d73898083f402b50f47e41"}, + {file = "scipy-1.11.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b9999c008ccf00e8fbcce1236f85ade5c569d13144f77a1946bef8863e8f6eb4"}, + {file = "scipy-1.11.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:933baf588daa8dc9a92c20a0be32f56d43faf3d1a60ab11b3f08c356430f6e56"}, + {file = "scipy-1.11.4-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:8fce70f39076a5aa62e92e69a7f62349f9574d8405c0a5de6ed3ef72de07f446"}, + {file = "scipy-1.11.4-cp310-cp310-win_amd64.whl", hash = "sha256:6550466fbeec7453d7465e74d4f4b19f905642c89a7525571ee91dd7adabb5a3"}, + {file = "scipy-1.11.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:f313b39a7e94f296025e3cffc2c567618174c0b1dde173960cf23808f9fae4be"}, + {file = "scipy-1.11.4-cp311-cp311-macosx_12_0_arm64.whl", hash = "sha256:1b7c3dca977f30a739e0409fb001056484661cb2541a01aba0bb0029f7b68db8"}, + {file = "scipy-1.11.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:00150c5eae7b610c32589dda259eacc7c4f1665aedf25d921907f4d08a951b1c"}, + {file = "scipy-1.11.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:530f9ad26440e85766509dbf78edcfe13ffd0ab7fec2560ee5c36ff74d6269ff"}, + {file = "scipy-1.11.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:5e347b14fe01003d3b78e196e84bd3f48ffe4c8a7b8a1afbcb8f5505cb710993"}, + {file = "scipy-1.11.4-cp311-cp311-win_amd64.whl", hash = "sha256:acf8ed278cc03f5aff035e69cb511741e0418681d25fbbb86ca65429c4f4d9cd"}, + {file = "scipy-1.11.4-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:028eccd22e654b3ea01ee63705681ee79933652b2d8f873e7949898dda6d11b6"}, + {file = "scipy-1.11.4-cp312-cp312-macosx_12_0_arm64.whl", hash = "sha256:2c6ff6ef9cc27f9b3db93a6f8b38f97387e6e0591600369a297a50a8e96e835d"}, + {file = "scipy-1.11.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b030c6674b9230d37c5c60ab456e2cf12f6784596d15ce8da9365e70896effc4"}, + {file = "scipy-1.11.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ad669df80528aeca5f557712102538f4f37e503f0c5b9541655016dd0932ca79"}, + {file = "scipy-1.11.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:ce7fff2e23ab2cc81ff452a9444c215c28e6305f396b2ba88343a567feec9660"}, + {file = "scipy-1.11.4-cp312-cp312-win_amd64.whl", hash = "sha256:36750b7733d960d7994888f0d148d31ea3017ac15eef664194b4ef68d36a4a97"}, + {file = "scipy-1.11.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6e619aba2df228a9b34718efb023966da781e89dd3d21637b27f2e54db0410d7"}, + {file = "scipy-1.11.4-cp39-cp39-macosx_12_0_arm64.whl", hash = "sha256:f3cd9e7b3c2c1ec26364856f9fbe78695fe631150f94cd1c22228456404cf1ec"}, + {file = "scipy-1.11.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d10e45a6c50211fe256da61a11c34927c68f277e03138777bdebedd933712fea"}, + {file = "scipy-1.11.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:91af76a68eeae0064887a48e25c4e616fa519fa0d38602eda7e0f97d65d57937"}, + {file = "scipy-1.11.4-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:6df1468153a31cf55ed5ed39647279beb9cfb5d3f84369453b49e4b8502394fd"}, + {file = "scipy-1.11.4-cp39-cp39-win_amd64.whl", hash = "sha256:ee410e6de8f88fd5cf6eadd73c135020bfbbbdfcd0f6162c36a7638a1ea8cc65"}, + {file = "scipy-1.11.4.tar.gz", hash = "sha256:90a2b78e7f5733b9de748f589f09225013685f9b218275257f8a8168ededaeaa"}, +] + +[package.dependencies] +numpy = ">=1.21.6,<1.28.0" + +[package.extras] +dev = ["click", "cython-lint (>=0.12.2)", "doit (>=0.36.0)", "mypy", "pycodestyle", "pydevtool", "rich-click", "ruff", "types-psutil", "typing_extensions"] +doc = ["jupytext", "matplotlib (>2)", "myst-nb", "numpydoc", "pooch", "pydata-sphinx-theme (==0.9.0)", "sphinx (!=4.1.0)", "sphinx-design (>=0.2.0)"] +test = ["asv", "gmpy2", "mpmath", "pooch", "pytest", "pytest-cov", "pytest-timeout", "pytest-xdist", "scikit-umfpack", "threadpoolctl"] + +[[package]] +name = "sentry-sdk" +version = "1.39.2" +description = "Python client for Sentry (https://sentry.io)" +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "sentry-sdk-1.39.2.tar.gz", hash = "sha256:24c83b0b41c887d33328a9166f5950dc37ad58f01c9f2fbff6b87a6f1094170c"}, + {file = "sentry_sdk-1.39.2-py2.py3-none-any.whl", hash = "sha256:acaf597b30258fc7663063b291aa99e58f3096e91fe1e6634f4b79f9c1943e8e"}, +] + +[package.dependencies] +certifi = "*" +urllib3 = {version = ">=1.26.11", markers = "python_version >= \"3.6\""} + +[package.extras] +aiohttp = ["aiohttp (>=3.5)"] +arq = ["arq (>=0.23)"] +asyncpg = ["asyncpg (>=0.23)"] +beam = ["apache-beam (>=2.12)"] +bottle = ["bottle (>=0.12.13)"] +celery = ["celery (>=3)"] +chalice = ["chalice (>=1.16.0)"] +clickhouse-driver = ["clickhouse-driver (>=0.2.0)"] +django = ["django (>=1.8)"] +falcon = ["falcon (>=1.4)"] +fastapi = ["fastapi (>=0.79.0)"] +flask = ["blinker (>=1.1)", "flask (>=0.11)", "markupsafe"] +grpcio = ["grpcio (>=1.21.1)"] +httpx = ["httpx (>=0.16.0)"] +huey = ["huey (>=2)"] +loguru = ["loguru (>=0.5)"] +opentelemetry = ["opentelemetry-distro (>=0.35b0)"] +opentelemetry-experimental = ["opentelemetry-distro (>=0.40b0,<1.0)", "opentelemetry-instrumentation-aiohttp-client (>=0.40b0,<1.0)", "opentelemetry-instrumentation-django (>=0.40b0,<1.0)", "opentelemetry-instrumentation-fastapi (>=0.40b0,<1.0)", "opentelemetry-instrumentation-flask (>=0.40b0,<1.0)", "opentelemetry-instrumentation-requests (>=0.40b0,<1.0)", "opentelemetry-instrumentation-sqlite3 (>=0.40b0,<1.0)", "opentelemetry-instrumentation-urllib (>=0.40b0,<1.0)"] +pure-eval = ["asttokens", "executing", "pure_eval"] +pymongo = ["pymongo (>=3.1)"] +pyspark = ["pyspark (>=2.4.4)"] +quart = ["blinker (>=1.1)", "quart (>=0.16.1)"] +rq = ["rq (>=0.6)"] +sanic = ["sanic (>=0.8)"] +sqlalchemy = ["sqlalchemy (>=1.2)"] +starlette = ["starlette (>=0.19.1)"] +starlite = ["starlite (>=1.48)"] +tornado = ["tornado (>=5)"] + +[[package]] +name = "setproctitle" +version = "1.3.3" +description = "A Python module to customize the process title" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "setproctitle-1.3.3-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:897a73208da48db41e687225f355ce993167079eda1260ba5e13c4e53be7f754"}, + {file = "setproctitle-1.3.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:8c331e91a14ba4076f88c29c777ad6b58639530ed5b24b5564b5ed2fd7a95452"}, + {file = "setproctitle-1.3.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bbbd6c7de0771c84b4aa30e70b409565eb1fc13627a723ca6be774ed6b9d9fa3"}, + {file = "setproctitle-1.3.3-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c05ac48ef16ee013b8a326c63e4610e2430dbec037ec5c5b58fcced550382b74"}, + {file = "setproctitle-1.3.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1342f4fdb37f89d3e3c1c0a59d6ddbedbde838fff5c51178a7982993d238fe4f"}, + {file = "setproctitle-1.3.3-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fc74e84fdfa96821580fb5e9c0b0777c1c4779434ce16d3d62a9c4d8c710df39"}, + {file = "setproctitle-1.3.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:9617b676b95adb412bb69645d5b077d664b6882bb0d37bfdafbbb1b999568d85"}, + {file = "setproctitle-1.3.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:6a249415f5bb88b5e9e8c4db47f609e0bf0e20a75e8d744ea787f3092ba1f2d0"}, + {file = "setproctitle-1.3.3-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:38da436a0aaace9add67b999eb6abe4b84397edf4a78ec28f264e5b4c9d53cd5"}, + {file = "setproctitle-1.3.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:da0d57edd4c95bf221b2ebbaa061e65b1788f1544977288bdf95831b6e44e44d"}, + {file = "setproctitle-1.3.3-cp310-cp310-win32.whl", hash = "sha256:a1fcac43918b836ace25f69b1dca8c9395253ad8152b625064415b1d2f9be4fb"}, + {file = "setproctitle-1.3.3-cp310-cp310-win_amd64.whl", hash = "sha256:200620c3b15388d7f3f97e0ae26599c0c378fdf07ae9ac5a13616e933cbd2086"}, + {file = "setproctitle-1.3.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:334f7ed39895d692f753a443102dd5fed180c571eb6a48b2a5b7f5b3564908c8"}, + {file = "setproctitle-1.3.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:950f6476d56ff7817a8fed4ab207727fc5260af83481b2a4b125f32844df513a"}, + {file = "setproctitle-1.3.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:195c961f54a09eb2acabbfc90c413955cf16c6e2f8caa2adbf2237d1019c7dd8"}, + {file = "setproctitle-1.3.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f05e66746bf9fe6a3397ec246fe481096664a9c97eb3fea6004735a4daf867fd"}, + {file = "setproctitle-1.3.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b5901a31012a40ec913265b64e48c2a4059278d9f4e6be628441482dd13fb8b5"}, + {file = "setproctitle-1.3.3-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:64286f8a995f2cd934082b398fc63fca7d5ffe31f0e27e75b3ca6b4efda4e353"}, + {file = "setproctitle-1.3.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:184239903bbc6b813b1a8fc86394dc6ca7d20e2ebe6f69f716bec301e4b0199d"}, + {file = "setproctitle-1.3.3-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:664698ae0013f986118064b6676d7dcd28fefd0d7d5a5ae9497cbc10cba48fa5"}, + {file = "setproctitle-1.3.3-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:e5119a211c2e98ff18b9908ba62a3bd0e3fabb02a29277a7232a6fb4b2560aa0"}, + {file = "setproctitle-1.3.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:417de6b2e214e837827067048f61841f5d7fc27926f2e43954567094051aff18"}, + {file = "setproctitle-1.3.3-cp311-cp311-win32.whl", hash = "sha256:6a143b31d758296dc2f440175f6c8e0b5301ced3b0f477b84ca43cdcf7f2f476"}, + {file = "setproctitle-1.3.3-cp311-cp311-win_amd64.whl", hash = "sha256:a680d62c399fa4b44899094027ec9a1bdaf6f31c650e44183b50d4c4d0ccc085"}, + {file = "setproctitle-1.3.3-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:d4460795a8a7a391e3567b902ec5bdf6c60a47d791c3b1d27080fc203d11c9dc"}, + {file = "setproctitle-1.3.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:bdfd7254745bb737ca1384dee57e6523651892f0ea2a7344490e9caefcc35e64"}, + {file = "setproctitle-1.3.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:477d3da48e216d7fc04bddab67b0dcde633e19f484a146fd2a34bb0e9dbb4a1e"}, + {file = "setproctitle-1.3.3-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ab2900d111e93aff5df9fddc64cf51ca4ef2c9f98702ce26524f1acc5a786ae7"}, + {file = "setproctitle-1.3.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:088b9efc62d5aa5d6edf6cba1cf0c81f4488b5ce1c0342a8b67ae39d64001120"}, + {file = "setproctitle-1.3.3-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a6d50252377db62d6a0bb82cc898089916457f2db2041e1d03ce7fadd4a07381"}, + {file = "setproctitle-1.3.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:87e668f9561fd3a457ba189edfc9e37709261287b52293c115ae3487a24b92f6"}, + {file = "setproctitle-1.3.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:287490eb90e7a0ddd22e74c89a92cc922389daa95babc833c08cf80c84c4df0a"}, + {file = "setproctitle-1.3.3-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:4fe1c49486109f72d502f8be569972e27f385fe632bd8895f4730df3c87d5ac8"}, + {file = "setproctitle-1.3.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:4a6ba2494a6449b1f477bd3e67935c2b7b0274f2f6dcd0f7c6aceae10c6c6ba3"}, + {file = "setproctitle-1.3.3-cp312-cp312-win32.whl", hash = "sha256:2df2b67e4b1d7498632e18c56722851ba4db5d6a0c91aaf0fd395111e51cdcf4"}, + {file = "setproctitle-1.3.3-cp312-cp312-win_amd64.whl", hash = "sha256:f38d48abc121263f3b62943f84cbaede05749047e428409c2c199664feb6abc7"}, + {file = "setproctitle-1.3.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:816330675e3504ae4d9a2185c46b573105d2310c20b19ea2b4596a9460a4f674"}, + {file = "setproctitle-1.3.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:68f960bc22d8d8e4ac886d1e2e21ccbd283adcf3c43136161c1ba0fa509088e0"}, + {file = "setproctitle-1.3.3-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:00e6e7adff74796ef12753ff399491b8827f84f6c77659d71bd0b35870a17d8f"}, + {file = "setproctitle-1.3.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:53bc0d2358507596c22b02db079618451f3bd720755d88e3cccd840bafb4c41c"}, + {file = "setproctitle-1.3.3-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ad6d20f9541f5f6ac63df553b6d7a04f313947f550eab6a61aa758b45f0d5657"}, + {file = "setproctitle-1.3.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:c1c84beab776b0becaa368254801e57692ed749d935469ac10e2b9b825dbdd8e"}, + {file = "setproctitle-1.3.3-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:507e8dc2891021350eaea40a44ddd887c9f006e6b599af8d64a505c0f718f170"}, + {file = "setproctitle-1.3.3-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:b1067647ac7aba0b44b591936118a22847bda3c507b0a42d74272256a7a798e9"}, + {file = "setproctitle-1.3.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:2e71f6365744bf53714e8bd2522b3c9c1d83f52ffa6324bd7cbb4da707312cd8"}, + {file = "setproctitle-1.3.3-cp37-cp37m-win32.whl", hash = "sha256:7f1d36a1e15a46e8ede4e953abb104fdbc0845a266ec0e99cc0492a4364f8c44"}, + {file = "setproctitle-1.3.3-cp37-cp37m-win_amd64.whl", hash = "sha256:c9a402881ec269d0cc9c354b149fc29f9ec1a1939a777f1c858cdb09c7a261df"}, + {file = "setproctitle-1.3.3-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:ff814dea1e5c492a4980e3e7d094286077054e7ea116cbeda138819db194b2cd"}, + {file = "setproctitle-1.3.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:accb66d7b3ccb00d5cd11d8c6e07055a4568a24c95cf86109894dcc0c134cc89"}, + {file = "setproctitle-1.3.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:554eae5a5b28f02705b83a230e9d163d645c9a08914c0ad921df363a07cf39b1"}, + {file = "setproctitle-1.3.3-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a911b26264dbe9e8066c7531c0591cfab27b464459c74385b276fe487ca91c12"}, + {file = "setproctitle-1.3.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2982efe7640c4835f7355fdb4da313ad37fb3b40f5c69069912f8048f77b28c8"}, + {file = "setproctitle-1.3.3-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:df3f4274b80709d8bcab2f9a862973d453b308b97a0b423a501bcd93582852e3"}, + {file = "setproctitle-1.3.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:af2c67ae4c795d1674a8d3ac1988676fa306bcfa1e23fddb5e0bd5f5635309ca"}, + {file = "setproctitle-1.3.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:af4061f67fd7ec01624c5e3c21f6b7af2ef0e6bab7fbb43f209e6506c9ce0092"}, + {file = "setproctitle-1.3.3-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:37a62cbe16d4c6294e84670b59cf7adcc73faafe6af07f8cb9adaf1f0e775b19"}, + {file = "setproctitle-1.3.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:a83ca086fbb017f0d87f240a8f9bbcf0809f3b754ee01cec928fff926542c450"}, + {file = "setproctitle-1.3.3-cp38-cp38-win32.whl", hash = "sha256:059f4ce86f8cc92e5860abfc43a1dceb21137b26a02373618d88f6b4b86ba9b2"}, + {file = "setproctitle-1.3.3-cp38-cp38-win_amd64.whl", hash = "sha256:ab92e51cd4a218208efee4c6d37db7368fdf182f6e7ff148fb295ecddf264287"}, + {file = "setproctitle-1.3.3-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:c7951820b77abe03d88b114b998867c0f99da03859e5ab2623d94690848d3e45"}, + {file = "setproctitle-1.3.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5bc94cf128676e8fac6503b37763adb378e2b6be1249d207630f83fc325d9b11"}, + {file = "setproctitle-1.3.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1f5d9027eeda64d353cf21a3ceb74bb1760bd534526c9214e19f052424b37e42"}, + {file = "setproctitle-1.3.3-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2e4a8104db15d3462e29d9946f26bed817a5b1d7a47eabca2d9dc2b995991503"}, + {file = "setproctitle-1.3.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c32c41ace41f344d317399efff4cffb133e709cec2ef09c99e7a13e9f3b9483c"}, + {file = "setproctitle-1.3.3-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cbf16381c7bf7f963b58fb4daaa65684e10966ee14d26f5cc90f07049bfd8c1e"}, + {file = "setproctitle-1.3.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:e18b7bd0898398cc97ce2dfc83bb192a13a087ef6b2d5a8a36460311cb09e775"}, + {file = "setproctitle-1.3.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:69d565d20efe527bd8a9b92e7f299ae5e73b6c0470f3719bd66f3cd821e0d5bd"}, + {file = "setproctitle-1.3.3-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:ddedd300cd690a3b06e7eac90ed4452348b1348635777ce23d460d913b5b63c3"}, + {file = "setproctitle-1.3.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:415bfcfd01d1fbf5cbd75004599ef167a533395955305f42220a585f64036081"}, + {file = "setproctitle-1.3.3-cp39-cp39-win32.whl", hash = "sha256:21112fcd2195d48f25760f0eafa7a76510871bbb3b750219310cf88b04456ae3"}, + {file = "setproctitle-1.3.3-cp39-cp39-win_amd64.whl", hash = "sha256:5a740f05d0968a5a17da3d676ce6afefebeeeb5ce137510901bf6306ba8ee002"}, + {file = "setproctitle-1.3.3-pp310-pypy310_pp73-macosx_10_9_x86_64.whl", hash = "sha256:6b9e62ddb3db4b5205c0321dd69a406d8af9ee1693529d144e86bd43bcb4b6c0"}, + {file = "setproctitle-1.3.3-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9e3b99b338598de0bd6b2643bf8c343cf5ff70db3627af3ca427a5e1a1a90dd9"}, + {file = "setproctitle-1.3.3-pp310-pypy310_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:38ae9a02766dad331deb06855fb7a6ca15daea333b3967e214de12cfae8f0ef5"}, + {file = "setproctitle-1.3.3-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:200ede6fd11233085ba9b764eb055a2a191fb4ffb950c68675ac53c874c22e20"}, + {file = "setproctitle-1.3.3-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:0d3a953c50776751e80fe755a380a64cb14d61e8762bd43041ab3f8cc436092f"}, + {file = "setproctitle-1.3.3-pp37-pypy37_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e5e08e232b78ba3ac6bc0d23ce9e2bee8fad2be391b7e2da834fc9a45129eb87"}, + {file = "setproctitle-1.3.3-pp37-pypy37_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f1da82c3e11284da4fcbf54957dafbf0655d2389cd3d54e4eaba636faf6d117a"}, + {file = "setproctitle-1.3.3-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:aeaa71fb9568ebe9b911ddb490c644fbd2006e8c940f21cb9a1e9425bd709574"}, + {file = "setproctitle-1.3.3-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:59335d000c6250c35989394661eb6287187854e94ac79ea22315469ee4f4c244"}, + {file = "setproctitle-1.3.3-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c3ba57029c9c50ecaf0c92bb127224cc2ea9fda057b5d99d3f348c9ec2855ad3"}, + {file = "setproctitle-1.3.3-pp38-pypy38_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d876d355c53d975c2ef9c4f2487c8f83dad6aeaaee1b6571453cb0ee992f55f6"}, + {file = "setproctitle-1.3.3-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:224602f0939e6fb9d5dd881be1229d485f3257b540f8a900d4271a2c2aa4e5f4"}, + {file = "setproctitle-1.3.3-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:d7f27e0268af2d7503386e0e6be87fb9b6657afd96f5726b733837121146750d"}, + {file = "setproctitle-1.3.3-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f5e7266498cd31a4572378c61920af9f6b4676a73c299fce8ba93afd694f8ae7"}, + {file = "setproctitle-1.3.3-pp39-pypy39_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:33c5609ad51cd99d388e55651b19148ea99727516132fb44680e1f28dd0d1de9"}, + {file = "setproctitle-1.3.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:eae8988e78192fd1a3245a6f4f382390b61bce6cfcc93f3809726e4c885fa68d"}, + {file = "setproctitle-1.3.3.tar.gz", hash = "sha256:c913e151e7ea01567837ff037a23ca8740192880198b7fbb90b16d181607caae"}, +] + +[package.extras] +test = ["pytest"] + [[package]] name = "setuptools" version = "68.2.2" @@ -1590,6 +2581,44 @@ files = [ {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"}, ] +[[package]] +name = "smmap" +version = "5.0.1" +description = "A pure Python implementation of a sliding window memory map manager" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "smmap-5.0.1-py3-none-any.whl", hash = "sha256:e6d8668fa5f93e706934a62d7b4db19c8d9eb8cf2adbb75ef1b675aa332b69da"}, + {file = "smmap-5.0.1.tar.gz", hash = "sha256:dceeb6c0028fdb6734471eb07c0cd2aae706ccaecab45965ee83f11c8d3b1f62"}, +] + +[[package]] +name = "stable-baselines3" +version = "2.2.1" +description = "Pytorch version of Stable Baselines, implementations of reinforcement learning algorithms." +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "stable_baselines3-2.2.1-py3-none-any.whl", hash = "sha256:ccb3405e244c3f0b284b3487436d99c782f1243229aa466cb4ceba93807e7428"}, + {file = "stable_baselines3-2.2.1.tar.gz", hash = "sha256:7db00bea41c82448d9243529059a9bc8131c0fe751098bd05e3d8822d2978a4d"}, +] + +[package.dependencies] +cloudpickle = "*" +gymnasium = ">=0.28.1,<0.30" +matplotlib = "*" +numpy = ">=1.20" +pandas = "*" +torch = ">=1.13" + +[package.extras] +docs = ["sphinx (>=5,<8)", "sphinx-autobuild", "sphinx-copybutton", "sphinx-rtd-theme (>=1.3.0)", "sphinxcontrib.spelling"] +extra = ["autorom[accept-rom-license] (>=0.6.1,<0.7.0)", "opencv-python", "pillow", "psutil", "pygame", "rich", "shimmy[atari] (>=1.3.0,<1.4.0)", "tensorboard (>=2.9.1)", "tqdm"] +extra-no-roms = ["opencv-python", "pillow", "psutil", "pygame", "rich", "shimmy[atari] (>=1.3.0,<1.4.0)", "tensorboard (>=2.9.1)", "tqdm"] +tests = ["black (>=23.9.1,<24)", "mypy", "pytest", "pytest-cov", "pytest-env", "pytest-xdist", "ruff (>=0.0.288)"] + [[package]] name = "stack-data" version = "0.6.3" @@ -1610,6 +2639,21 @@ pure-eval = "*" [package.extras] tests = ["cython", "littleutils", "pygments", "pytest", "typeguard"] +[[package]] +name = "sympy" +version = "1.12" +description = "Computer algebra system (CAS) in Python" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "sympy-1.12-py3-none-any.whl", hash = "sha256:c3588cd4295d0c0f603d0f2ae780587e64e2efeedb3521e46b9bb1d08d184fa5"}, + {file = "sympy-1.12.tar.gz", hash = "sha256:ebf595c8dac3e0fdc4152c51878b498396ec7f30e7a914d6071e674d49420fb8"}, +] + +[package.dependencies] +mpmath = ">=0.19" + [[package]] name = "tomli" version = "2.0.1" @@ -1622,6 +2666,60 @@ files = [ {file = "tomli-2.0.1.tar.gz", hash = "sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f"}, ] +[[package]] +name = "torch" +version = "2.1.2" +description = "Tensors and Dynamic neural networks in Python with strong GPU acceleration" +category = "main" +optional = false +python-versions = ">=3.8.0" +files = [ + {file = "torch-2.1.2-cp310-cp310-manylinux1_x86_64.whl", hash = "sha256:3a871edd6c02dae77ad810335c0833391c1a4ce49af21ea8cf0f6a5d2096eea8"}, + {file = "torch-2.1.2-cp310-cp310-manylinux2014_aarch64.whl", hash = "sha256:bef6996c27d8f6e92ea4e13a772d89611da0e103b48790de78131e308cf73076"}, + {file = "torch-2.1.2-cp310-cp310-win_amd64.whl", hash = "sha256:0e13034fd5fb323cbbc29e56d0637a3791e50dd589616f40c79adfa36a5a35a1"}, + {file = "torch-2.1.2-cp310-none-macosx_10_9_x86_64.whl", hash = "sha256:d9b535cad0df3d13997dbe8bd68ac33e0e3ae5377639c9881948e40794a61403"}, + {file = "torch-2.1.2-cp310-none-macosx_11_0_arm64.whl", hash = "sha256:f9a55d55af02826ebfbadf4e9b682f0f27766bc33df8236b48d28d705587868f"}, + {file = "torch-2.1.2-cp311-cp311-manylinux1_x86_64.whl", hash = "sha256:a6ebbe517097ef289cc7952783588c72de071d4b15ce0f8b285093f0916b1162"}, + {file = "torch-2.1.2-cp311-cp311-manylinux2014_aarch64.whl", hash = "sha256:8f32ce591616a30304f37a7d5ea80b69ca9e1b94bba7f308184bf616fdaea155"}, + {file = "torch-2.1.2-cp311-cp311-win_amd64.whl", hash = "sha256:e0ee6cf90c8970e05760f898d58f9ac65821c37ffe8b04269ec787aa70962b69"}, + {file = "torch-2.1.2-cp311-none-macosx_10_9_x86_64.whl", hash = "sha256:76d37967c31c99548ad2c4d3f2cf191db48476f2e69b35a0937137116da356a1"}, + {file = "torch-2.1.2-cp311-none-macosx_11_0_arm64.whl", hash = "sha256:e2d83f07b4aac983453ea5bf8f9aa9dacf2278a8d31247f5d9037f37befc60e4"}, + {file = "torch-2.1.2-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:f41fe0c7ecbf903a568c73486139a75cfab287a0f6c17ed0698fdea7a1e8641d"}, + {file = "torch-2.1.2-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:e3225f47d50bb66f756fe9196a768055d1c26b02154eb1f770ce47a2578d3aa7"}, + {file = "torch-2.1.2-cp38-cp38-win_amd64.whl", hash = "sha256:33d59cd03cb60106857f6c26b36457793637512998666ee3ce17311f217afe2b"}, + {file = "torch-2.1.2-cp38-none-macosx_10_9_x86_64.whl", hash = "sha256:8e221deccd0def6c2badff6be403e0c53491805ed9915e2c029adbcdb87ab6b5"}, + {file = "torch-2.1.2-cp38-none-macosx_11_0_arm64.whl", hash = "sha256:05b18594f60a911a0c4f023f38a8bda77131fba5fd741bda626e97dcf5a3dd0a"}, + {file = "torch-2.1.2-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:9ca96253b761e9aaf8e06fb30a66ee301aecbf15bb5a303097de1969077620b6"}, + {file = "torch-2.1.2-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:d93ba70f67b08c2ae5598ee711cbc546a1bc8102cef938904b8c85c2089a51a0"}, + {file = "torch-2.1.2-cp39-cp39-win_amd64.whl", hash = "sha256:255b50bc0608db177e6a3cc118961d77de7e5105f07816585fa6f191f33a9ff3"}, + {file = "torch-2.1.2-cp39-none-macosx_10_9_x86_64.whl", hash = "sha256:6984cd5057c0c977b3c9757254e989d3f1124f4ce9d07caa6cb637783c71d42a"}, + {file = "torch-2.1.2-cp39-none-macosx_11_0_arm64.whl", hash = "sha256:bc195d7927feabc0eb7c110e457c955ed2ab616f3c7c28439dd4188cf589699f"}, +] + +[package.dependencies] +filelock = "*" +fsspec = "*" +jinja2 = "*" +networkx = "*" +nvidia-cublas-cu12 = {version = "12.1.3.1", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-cuda-cupti-cu12 = {version = "12.1.105", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-cuda-nvrtc-cu12 = {version = "12.1.105", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-cuda-runtime-cu12 = {version = "12.1.105", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-cudnn-cu12 = {version = "8.9.2.26", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-cufft-cu12 = {version = "11.0.2.54", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-curand-cu12 = {version = "10.3.2.106", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-cusolver-cu12 = {version = "11.4.5.107", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-cusparse-cu12 = {version = "12.1.0.106", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-nccl-cu12 = {version = "2.18.1", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +nvidia-nvtx-cu12 = {version = "12.1.105", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +sympy = "*" +triton = {version = "2.1.0", markers = "platform_system == \"Linux\" and platform_machine == \"x86_64\""} +typing-extensions = "*" + +[package.extras] +dynamo = ["jinja2"] +opt-einsum = ["opt-einsum (>=3.3)"] + [[package]] name = "tornado" version = "6.3.3" @@ -1659,6 +2757,32 @@ files = [ docs = ["myst-parser", "pydata-sphinx-theme", "sphinx"] test = ["argcomplete (>=3.0.3)", "mypy (>=1.6.0)", "pre-commit", "pytest (>=7.0,<7.5)", "pytest-mock", "pytest-mypy-testing"] +[[package]] +name = "triton" +version = "2.1.0" +description = "A language and compiler for custom Deep Learning operations" +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "triton-2.1.0-0-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:66439923a30d5d48399b08a9eae10370f6c261a5ec864a64983bae63152d39d7"}, + {file = "triton-2.1.0-0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:919b06453f0033ea52c13eaf7833de0e57db3178d23d4e04f9fc71c4f2c32bf8"}, + {file = "triton-2.1.0-0-cp37-cp37m-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:ae4bb8a91de790e1866405211c4d618379781188f40d5c4c399766914e84cd94"}, + {file = "triton-2.1.0-0-cp38-cp38-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:39f6fb6bdccb3e98f3152e3fbea724f1aeae7d749412bbb1fa9c441d474eba26"}, + {file = "triton-2.1.0-0-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:21544e522c02005a626c8ad63d39bdff2f31d41069592919ef281e964ed26446"}, + {file = "triton-2.1.0-0-pp37-pypy37_pp73-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:143582ca31dd89cd982bd3bf53666bab1c7527d41e185f9e3d8a3051ce1b663b"}, + {file = "triton-2.1.0-0-pp38-pypy38_pp73-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:82fc5aeeedf6e36be4e4530cbdcba81a09d65c18e02f52dc298696d45721f3bd"}, + {file = "triton-2.1.0-0-pp39-pypy39_pp73-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:81a96d110a738ff63339fc892ded095b31bd0d205e3aace262af8400d40b6fa8"}, +] + +[package.dependencies] +filelock = "*" + +[package.extras] +build = ["cmake (>=3.18)", "lit"] +tests = ["autopep8", "flake8", "isort", "numpy", "pytest", "scipy (>=1.7.1)"] +tutorials = ["matplotlib", "pandas", "tabulate"] + [[package]] name = "typing-extensions" version = "4.8.0" @@ -1671,6 +2795,35 @@ files = [ {file = "typing_extensions-4.8.0.tar.gz", hash = "sha256:df8e4339e9cb77357558cbdbceca33c303714cf861d1eef15e1070055ae8b7ef"}, ] +[[package]] +name = "tzdata" +version = "2023.4" +description = "Provider of IANA time zone data" +category = "main" +optional = false +python-versions = ">=2" +files = [ + {file = "tzdata-2023.4-py2.py3-none-any.whl", hash = "sha256:aa3ace4329eeacda5b7beb7ea08ece826c28d761cda36e747cfbf97996d39bf3"}, + {file = "tzdata-2023.4.tar.gz", hash = "sha256:dd54c94f294765522c77399649b4fefd95522479a664a0cec87f41bebc6148c9"}, +] + +[[package]] +name = "urllib3" +version = "2.1.0" +description = "HTTP library with thread-safe connection pooling, file post, and more." +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "urllib3-2.1.0-py3-none-any.whl", hash = "sha256:55901e917a5896a349ff771be919f8bd99aff50b79fe58fec595eb37bbc56bb3"}, + {file = "urllib3-2.1.0.tar.gz", hash = "sha256:df7aa8afb0148fa78488e7899b2c59b5f4ffcfa82e6c54ccb9dd37c1d7b52d54"}, +] + +[package.extras] +brotli = ["brotli (>=1.0.9)", "brotlicffi (>=0.8.0)"] +socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"] +zstd = ["zstandard (>=0.18.0)"] + [[package]] name = "virtualenv" version = "20.24.6" @@ -1692,6 +2845,47 @@ platformdirs = ">=3.9.1,<4" docs = ["furo (>=2023.7.26)", "proselint (>=0.13)", "sphinx (>=7.1.2)", "sphinx-argparse (>=0.4)", "sphinxcontrib-towncrier (>=0.2.1a0)", "towncrier (>=23.6)"] test = ["covdefaults (>=2.3)", "coverage (>=7.2.7)", "coverage-enable-subprocess (>=1)", "flaky (>=3.7)", "packaging (>=23.1)", "pytest (>=7.4)", "pytest-env (>=0.8.2)", "pytest-freezer (>=0.4.8)", "pytest-mock (>=3.11.1)", "pytest-randomly (>=3.12)", "pytest-timeout (>=2.1)", "setuptools (>=68)", "time-machine (>=2.10)"] +[[package]] +name = "wandb" +version = "0.16.2" +description = "A CLI and library for interacting with the Weights & Biases API." +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "wandb-0.16.2-py3-none-any.whl", hash = "sha256:6b119cf3c01f35e7276b62d052128e5320621d182c9eb5796a12cf62a9b3134f"}, + {file = "wandb-0.16.2.tar.gz", hash = "sha256:e40cd79ea6272fe4762a80b9f47b172e141daeb3b56eb9d1e192ebd10752e64e"}, +] + +[package.dependencies] +appdirs = ">=1.4.3" +Click = ">=7.1,<8.0.0 || >8.0.0" +docker-pycreds = ">=0.4.0" +GitPython = ">=1.0.0,<3.1.29 || >3.1.29" +protobuf = [ + {version = ">=3.15.0,<4.21.0 || >4.21.0,<5", markers = "python_version == \"3.9\" and sys_platform == \"linux\""}, + {version = ">=3.19.0,<4.21.0 || >4.21.0,<5", markers = "python_version > \"3.9\" or sys_platform != \"linux\""}, +] +psutil = ">=5.0.0" +PyYAML = "*" +requests = ">=2.0.0,<3" +sentry-sdk = ">=1.0.0" +setproctitle = "*" +setuptools = "*" +typing-extensions = {version = "*", markers = "python_version < \"3.10\""} + +[package.extras] +async = ["httpx (>=0.23.0)"] +aws = ["boto3"] +azure = ["azure-identity", "azure-storage-blob"] +gcp = ["google-cloud-storage"] +kubeflow = ["google-cloud-storage", "kubernetes", "minio", "sh"] +launch = ["PyYAML (>=6.0.0)", "awscli", "azure-containerregistry", "azure-identity", "azure-storage-blob", "boto3", "botocore", "chardet", "google-auth", "google-cloud-aiplatform", "google-cloud-artifact-registry", "google-cloud-compute", "google-cloud-storage", "iso8601", "kubernetes", "kubernetes-asyncio", "nbconvert", "nbformat", "optuna", "pydantic", "typing-extensions"] +media = ["bokeh", "moviepy", "numpy", "pillow", "plotly (>=5.18.0)", "rdkit-pypi", "soundfile"] +models = ["cloudpickle"] +perf = ["orjson"] +sweeps = ["sweeps (>=0.2.0)"] + [[package]] name = "wcwidth" version = "0.2.9" @@ -1720,7 +2914,69 @@ files = [ docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-lint"] testing = ["big-O", "jaraco.functools", "jaraco.itertools", "more-itertools", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-ignore-flaky", "pytest-mypy (>=0.9.1)", "pytest-ruff"] +[[package]] +name = "zstandard" +version = "0.22.0" +description = "Zstandard bindings for Python" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "zstandard-0.22.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:275df437ab03f8c033b8a2c181e51716c32d831082d93ce48002a5227ec93019"}, + {file = "zstandard-0.22.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:2ac9957bc6d2403c4772c890916bf181b2653640da98f32e04b96e4d6fb3252a"}, + {file = "zstandard-0.22.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fe3390c538f12437b859d815040763abc728955a52ca6ff9c5d4ac707c4ad98e"}, + {file = "zstandard-0.22.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1958100b8a1cc3f27fa21071a55cb2ed32e9e5df4c3c6e661c193437f171cba2"}, + {file = "zstandard-0.22.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:93e1856c8313bc688d5df069e106a4bc962eef3d13372020cc6e3ebf5e045202"}, + {file = "zstandard-0.22.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:1a90ba9a4c9c884bb876a14be2b1d216609385efb180393df40e5172e7ecf356"}, + {file = "zstandard-0.22.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:3db41c5e49ef73641d5111554e1d1d3af106410a6c1fb52cf68912ba7a343a0d"}, + {file = "zstandard-0.22.0-cp310-cp310-win32.whl", hash = "sha256:d8593f8464fb64d58e8cb0b905b272d40184eac9a18d83cf8c10749c3eafcd7e"}, + {file = "zstandard-0.22.0-cp310-cp310-win_amd64.whl", hash = "sha256:f1a4b358947a65b94e2501ce3e078bbc929b039ede4679ddb0460829b12f7375"}, + {file = "zstandard-0.22.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:589402548251056878d2e7c8859286eb91bd841af117dbe4ab000e6450987e08"}, + {file = "zstandard-0.22.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a97079b955b00b732c6f280d5023e0eefe359045e8b83b08cf0333af9ec78f26"}, + {file = "zstandard-0.22.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:445b47bc32de69d990ad0f34da0e20f535914623d1e506e74d6bc5c9dc40bb09"}, + {file = "zstandard-0.22.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:33591d59f4956c9812f8063eff2e2c0065bc02050837f152574069f5f9f17775"}, + {file = "zstandard-0.22.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:888196c9c8893a1e8ff5e89b8f894e7f4f0e64a5af4d8f3c410f0319128bb2f8"}, + {file = "zstandard-0.22.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:53866a9d8ab363271c9e80c7c2e9441814961d47f88c9bc3b248142c32141d94"}, + {file = "zstandard-0.22.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:4ac59d5d6910b220141c1737b79d4a5aa9e57466e7469a012ed42ce2d3995e88"}, + {file = "zstandard-0.22.0-cp311-cp311-win32.whl", hash = "sha256:2b11ea433db22e720758cba584c9d661077121fcf60ab43351950ded20283440"}, + {file = "zstandard-0.22.0-cp311-cp311-win_amd64.whl", hash = "sha256:11f0d1aab9516a497137b41e3d3ed4bbf7b2ee2abc79e5c8b010ad286d7464bd"}, + {file = "zstandard-0.22.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:6c25b8eb733d4e741246151d895dd0308137532737f337411160ff69ca24f93a"}, + {file = "zstandard-0.22.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f9b2cde1cd1b2a10246dbc143ba49d942d14fb3d2b4bccf4618d475c65464912"}, + {file = "zstandard-0.22.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a88b7df61a292603e7cd662d92565d915796b094ffb3d206579aaebac6b85d5f"}, + {file = "zstandard-0.22.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:466e6ad8caefb589ed281c076deb6f0cd330e8bc13c5035854ffb9c2014b118c"}, + {file = "zstandard-0.22.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a1d67d0d53d2a138f9e29d8acdabe11310c185e36f0a848efa104d4e40b808e4"}, + {file = "zstandard-0.22.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:39b2853efc9403927f9065cc48c9980649462acbdf81cd4f0cb773af2fd734bc"}, + {file = "zstandard-0.22.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8a1b2effa96a5f019e72874969394edd393e2fbd6414a8208fea363a22803b45"}, + {file = "zstandard-0.22.0-cp312-cp312-win32.whl", hash = "sha256:88c5b4b47a8a138338a07fc94e2ba3b1535f69247670abfe422de4e0b344aae2"}, + {file = "zstandard-0.22.0-cp312-cp312-win_amd64.whl", hash = "sha256:de20a212ef3d00d609d0b22eb7cc798d5a69035e81839f549b538eff4105d01c"}, + {file = "zstandard-0.22.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d75f693bb4e92c335e0645e8845e553cd09dc91616412d1d4650da835b5449df"}, + {file = "zstandard-0.22.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:36a47636c3de227cd765e25a21dc5dace00539b82ddd99ee36abae38178eff9e"}, + {file = "zstandard-0.22.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:68953dc84b244b053c0d5f137a21ae8287ecf51b20872eccf8eaac0302d3e3b0"}, + {file = "zstandard-0.22.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2612e9bb4977381184bb2463150336d0f7e014d6bb5d4a370f9a372d21916f69"}, + {file = "zstandard-0.22.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:23d2b3c2b8e7e5a6cb7922f7c27d73a9a615f0a5ab5d0e03dd533c477de23004"}, + {file = "zstandard-0.22.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:1d43501f5f31e22baf822720d82b5547f8a08f5386a883b32584a185675c8fbf"}, + {file = "zstandard-0.22.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:a493d470183ee620a3df1e6e55b3e4de8143c0ba1b16f3ded83208ea8ddfd91d"}, + {file = "zstandard-0.22.0-cp38-cp38-win32.whl", hash = "sha256:7034d381789f45576ec3f1fa0e15d741828146439228dc3f7c59856c5bcd3292"}, + {file = "zstandard-0.22.0-cp38-cp38-win_amd64.whl", hash = "sha256:d8fff0f0c1d8bc5d866762ae95bd99d53282337af1be9dc0d88506b340e74b73"}, + {file = "zstandard-0.22.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:2fdd53b806786bd6112d97c1f1e7841e5e4daa06810ab4b284026a1a0e484c0b"}, + {file = "zstandard-0.22.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:73a1d6bd01961e9fd447162e137ed949c01bdb830dfca487c4a14e9742dccc93"}, + {file = "zstandard-0.22.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9501f36fac6b875c124243a379267d879262480bf85b1dbda61f5ad4d01b75a3"}, + {file = "zstandard-0.22.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:48f260e4c7294ef275744210a4010f116048e0c95857befb7462e033f09442fe"}, + {file = "zstandard-0.22.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:959665072bd60f45c5b6b5d711f15bdefc9849dd5da9fb6c873e35f5d34d8cfb"}, + {file = "zstandard-0.22.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:d22fdef58976457c65e2796e6730a3ea4a254f3ba83777ecfc8592ff8d77d303"}, + {file = "zstandard-0.22.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:a7ccf5825fd71d4542c8ab28d4d482aace885f5ebe4b40faaa290eed8e095a4c"}, + {file = "zstandard-0.22.0-cp39-cp39-win32.whl", hash = "sha256:f058a77ef0ece4e210bb0450e68408d4223f728b109764676e1a13537d056bb0"}, + {file = "zstandard-0.22.0-cp39-cp39-win_amd64.whl", hash = "sha256:e9e9d4e2e336c529d4c435baad846a181e39a982f823f7e4495ec0b0ec8538d2"}, + {file = "zstandard-0.22.0.tar.gz", hash = "sha256:8226a33c542bcb54cd6bd0a366067b610b41713b64c9abec1bc4533d69f51e70"}, +] + +[package.dependencies] +cffi = {version = ">=1.11", markers = "platform_python_implementation == \"PyPy\""} + +[package.extras] +cffi = ["cffi (>=1.11)"] + [metadata] lock-version = "2.0" -python-versions = "^3.8" -content-hash = "7a66902cdbc87792136b16bced15762a8c29007b3f07af05192bbdad5d54f0cd" +python-versions = ">=3.9,<4.0" +content-hash = "3a6cdd7ca2420e9dbc35fb15231ffcd6579f6a7a94f5183edaecc3b7c2c38656" diff --git a/pyproject.toml b/pyproject.toml index 6d95111..351eefa 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -7,13 +7,19 @@ readme = "README.md" packages = [{include = "furuta"}] [tool.poetry.dependencies] -python = "^3.8" +python = ">=3.9,<4.0" gymnasium = "^0.29.1" pyserial = "^3.5" simple-pid = "^2.0.0" ipykernel = "^6.26.0" matplotlib = "3.7.3" pre-commit = "^3.5.0" +stable-baselines3 = "^2.2.1" +wandb = "^0.16.2" +mcap-protobuf-support = "^0.4.1" +scipy = "^1.11.4" +pygame = "^2.5.2" +opencv-python = "^4.9.0.80" [tool.poetry.group.dev.dependencies] From 91c70c791973324d48d48d7dee65a050057b6836 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 14 Jan 2024 17:49:35 +0100 Subject: [PATCH 07/44] reorg package structure --- furuta/__init__.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/furuta/__init__.py b/furuta/__init__.py index c7a6b36..7e3162b 100644 --- a/furuta/__init__.py +++ b/furuta/__init__.py @@ -2,9 +2,9 @@ register( id="FurutaReal-v0", - entry_point="furuta_gym.envs:FurutaReal", + entry_point="furuta.rl.envs.furuta_real:FurutaReal", ) register( id="FurutaSim-v0", - entry_point="furuta_gym.envs:FurutaSim", + entry_point="furuta.rl.envs.furuta_sim:FurutaSim", ) From 0b502f8a045f8415a55504c4a96d2c3160176c7f Mon Sep 17 00:00:00 2001 From: Armand Date: Mon, 15 Jan 2024 13:44:03 +0100 Subject: [PATCH 08/44] setup training on sim to check everything ok --- .gitignore | 1 + furuta/rl/algos.py | 4 +- furuta/rl/envs/furuta_sim.py | 19 +- furuta/rl/utils.py | 46 ++ poetry.lock | 524 +++++++++++++++++--- pyproject.toml | 3 + scripts/configs/algo/ppo.yaml | 4 + scripts/configs/algo/sac.yaml | 6 +- scripts/configs/experiment/sim.yaml | 2 +- scripts/configs/train.yaml | 7 +- scripts/configs/wrappers/base_wrappers.yaml | 4 +- scripts/configs/wrappers/real_wrappers.yaml | 3 +- scripts/train.py | 76 ++- 13 files changed, 571 insertions(+), 128 deletions(-) create mode 100644 furuta/rl/utils.py create mode 100644 scripts/configs/algo/ppo.yaml diff --git a/.gitignore b/.gitignore index fd1709d..8aa8029 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,5 @@ **/videos/* +**/outputs/* **/wandb/* **/runs/* **/models/* diff --git a/furuta/rl/algos.py b/furuta/rl/algos.py index 552d8fa..d06bebd 100644 --- a/furuta/rl/algos.py +++ b/furuta/rl/algos.py @@ -6,11 +6,11 @@ # check if they all have the train freq param # check if they have other tuple args # check if it would be cleaner for sb3 to accept list instead of tuple? -# TODO also does having the sb3 import means i need sb3 when importing from everywhere else in the package? class SAC(stable_baselines3.SAC): def __init__(self, **kwargs): # sb3 expects tuple, omegaconf returns list # so we need to convert kwarg train_freq from tuple to list - kwargs.update({"train_freq": tuple(kwargs["train_freq"])}) + if "train_freq" in kwargs and type(kwargs["train_freq"]) == list: + kwargs.update({"train_freq": tuple(kwargs["train_freq"])}) super().__init__(**kwargs) diff --git a/furuta/rl/envs/furuta_sim.py b/furuta/rl/envs/furuta_sim.py index bfe1793..b71e30a 100644 --- a/furuta/rl/envs/furuta_sim.py +++ b/furuta/rl/envs/furuta_sim.py @@ -31,7 +31,7 @@ def __init__( else: self.vel_filt = None - def _init_state(self): + def _init_state(self, random_init: bool = True): # TODO could also sample from state space # though we also use it as upper speed limit # the two use case are kind of conflicting @@ -43,15 +43,8 @@ def _init_state(self): # or maybe it's too slow to move even the simulated pendulum? # and maybe it should have a min voltage as well? # self._state = np.random.rand(4) # self.state_space.sample() - - # TODO actually sample from system bounds? - # start at zero for now - self._simulation_state = np.zeros( - 4, dtype=np.float32 - ) # 0.01 * np.float32(np.random.randn(self.state_space.shape[0])) - self._state = np.zeros(self.state_space.shape[0], dtype=np.float32) - - self._update_state(0) + self._simulation_state = 0.01 * np.float32(np.random.randn(self.state_space.shape[0])) + self._state = self._simulation_state.copy() def _update_state(self, a): # ok so we simulate two things: the systems's state @@ -96,8 +89,10 @@ def reset( seed: Optional[int] = None, options: Optional[dict] = None, ): - self._init_state() - obs, _, _, _, _ = self.step(np.array([0.0])) + if options is None: + options = {} + self._init_state(options.get("random_init", True)) + obs = self.get_obs() return obs, {} diff --git a/furuta/rl/utils.py b/furuta/rl/utils.py new file mode 100644 index 0000000..e8e6216 --- /dev/null +++ b/furuta/rl/utils.py @@ -0,0 +1,46 @@ +import logging +import os +import random +from pathlib import Path + +import numpy as np +import torch +import wandb + + +def seed_everything(env, seed, cudnn_deterministic): + env.action_space.seed(seed) + env.observation_space.seed(seed) + random.seed(seed) + np.random.seed(seed) + torch.manual_seed(seed) + torch.backends.cudnn.deterministic = cudnn_deterministic + + +def download_artifact_file(artifact_alias, filename): + """Download artifact and returns path to filename. + + :param artifact_name: wandb artifact alias + :param filename: filename in the artifact + """ + logging.info(f"loading {filename} from {artifact_alias}") + + artifact = wandb.use_artifact(artifact_alias) + artifact_dir = Path(artifact.download()) + filepath = artifact_dir / filename + + assert filepath.is_file(), f"{artifact_alias} doesn't contain {filename}" + + return filepath + + +def upload_file_to_artifacts(pth, artifact_name, artifact_type): + logging.info(f"Saving {pth} to {artifact_name}") + if not isinstance(pth, Path): + pth = Path(pth) + + assert os.path.isfile(pth), f"{pth} is not a file" + + artifact = wandb.Artifact(artifact_name, type=artifact_type) + artifact.add_file(pth) + wandb.log_artifact(artifact) diff --git a/poetry.lock b/poetry.lock index 592cca0..2a90370 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,5 +1,28 @@ # This file is automatically @generated by Poetry 1.4.0 and should not be changed by hand. +[[package]] +name = "absl-py" +version = "2.0.0" +description = "Abseil Python Common Libraries, see https://github.com/abseil/abseil-py." +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "absl-py-2.0.0.tar.gz", hash = "sha256:d9690211c5fcfefcdd1a45470ac2b5c5acd45241c3af71eed96bc5441746c0d5"}, + {file = "absl_py-2.0.0-py3-none-any.whl", hash = "sha256:9a28abb62774ae4e8edbe2dd4c49ffcd45a6a848952a5eccc6a49f3f0fc1e2f3"}, +] + +[[package]] +name = "antlr4-python3-runtime" +version = "4.9.3" +description = "ANTLR 4.9.3 runtime for Python 3.7" +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "antlr4-python3-runtime-4.9.3.tar.gz", hash = "sha256:f224469b4168294902bb1efa80a8bf7855f24c99aef99cbefc1bcd3cce77881b"}, +] + [[package]] name = "appdirs" version = "1.4.4" @@ -55,6 +78,18 @@ files = [ {file = "backcall-0.2.0.tar.gz", hash = "sha256:5cbdbf27be5e7cfadb448baf0aa95508f91f2bbc6c6437cd9cd06e2a4c215e1e"}, ] +[[package]] +name = "cachetools" +version = "5.3.2" +description = "Extensible memoizing collections and decorators" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "cachetools-5.3.2-py3-none-any.whl", hash = "sha256:861f35a13a451f94e301ce2bec7cac63e881232ccce7ed67fab9b5df4d3beaa1"}, + {file = "cachetools-5.3.2.tar.gz", hash = "sha256:086ee420196f7b2ab9ca2db2520aca326318b68fe5ba8bc4d49cca91add450f2"}, +] + [[package]] name = "certifi" version = "2023.11.17" @@ -422,14 +457,14 @@ files = [ [[package]] name = "decorator" -version = "5.1.1" +version = "4.4.2" description = "Decorators for Humans" category = "main" optional = false -python-versions = ">=3.5" +python-versions = ">=2.6, !=3.0.*, !=3.1.*" files = [ - {file = "decorator-5.1.1-py3-none-any.whl", hash = "sha256:b8c3f85900b9dc423225913c5aace94729fe1fa9763b38939a95226f02d37186"}, - {file = "decorator-5.1.1.tar.gz", hash = "sha256:637996211036b6385ef91435e4fae22989472f9d571faba8927ba8253acbc330"}, + {file = "decorator-4.4.2-py2.py3-none-any.whl", hash = "sha256:41fa54c2a0cc4ba648be4fd43cff00aedf5b9465c9bf18d64325bc225f08f760"}, + {file = "decorator-4.4.2.tar.gz", hash = "sha256:e3a62f0520172440ca0dcc823749319382e377f37f140a0b99ef45fecb84bfe7"}, ] [[package]] @@ -653,6 +688,116 @@ gitdb = ">=4.0.1,<5" [package.extras] test = ["black", "coverage[toml]", "ddt (>=1.1.1,!=1.4.3)", "mock", "mypy", "pre-commit", "pytest (>=7.3.1)", "pytest-cov", "pytest-instafail", "pytest-mock", "pytest-sugar", "sumtypes"] +[[package]] +name = "google-auth" +version = "2.26.2" +description = "Google Authentication Library" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "google-auth-2.26.2.tar.gz", hash = "sha256:97327dbbf58cccb58fc5a1712bba403ae76668e64814eb30f7316f7e27126b81"}, + {file = "google_auth-2.26.2-py2.py3-none-any.whl", hash = "sha256:3f445c8ce9b61ed6459aad86d8ccdba4a9afed841b2d1451a11ef4db08957424"}, +] + +[package.dependencies] +cachetools = ">=2.0.0,<6.0" +pyasn1-modules = ">=0.2.1" +rsa = ">=3.1.4,<5" + +[package.extras] +aiohttp = ["aiohttp (>=3.6.2,<4.0.0.dev0)", "requests (>=2.20.0,<3.0.0.dev0)"] +enterprise-cert = ["cryptography (==36.0.2)", "pyopenssl (==22.0.0)"] +pyopenssl = ["cryptography (>=38.0.3)", "pyopenssl (>=20.0.0)"] +reauth = ["pyu2f (>=0.1.5)"] +requests = ["requests (>=2.20.0,<3.0.0.dev0)"] + +[[package]] +name = "google-auth-oauthlib" +version = "1.2.0" +description = "Google Authentication Library" +category = "main" +optional = false +python-versions = ">=3.6" +files = [ + {file = "google-auth-oauthlib-1.2.0.tar.gz", hash = "sha256:292d2d3783349f2b0734a0a0207b1e1e322ac193c2c09d8f7c613fb7cc501ea8"}, + {file = "google_auth_oauthlib-1.2.0-py2.py3-none-any.whl", hash = "sha256:297c1ce4cb13a99b5834c74a1fe03252e1e499716718b190f56bcb9c4abc4faf"}, +] + +[package.dependencies] +google-auth = ">=2.15.0" +requests-oauthlib = ">=0.7.0" + +[package.extras] +tool = ["click (>=6.0.0)"] + +[[package]] +name = "grpcio" +version = "1.60.0" +description = "HTTP/2-based RPC framework" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "grpcio-1.60.0-cp310-cp310-linux_armv7l.whl", hash = "sha256:d020cfa595d1f8f5c6b343530cd3ca16ae5aefdd1e832b777f9f0eb105f5b139"}, + {file = "grpcio-1.60.0-cp310-cp310-macosx_12_0_universal2.whl", hash = "sha256:b98f43fcdb16172dec5f4b49f2fece4b16a99fd284d81c6bbac1b3b69fcbe0ff"}, + {file = "grpcio-1.60.0-cp310-cp310-manylinux_2_17_aarch64.whl", hash = "sha256:20e7a4f7ded59097c84059d28230907cd97130fa74f4a8bfd1d8e5ba18c81491"}, + {file = "grpcio-1.60.0-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:452ca5b4afed30e7274445dd9b441a35ece656ec1600b77fff8c216fdf07df43"}, + {file = "grpcio-1.60.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:43e636dc2ce9ece583b3e2ca41df5c983f4302eabc6d5f9cd04f0562ee8ec1ae"}, + {file = "grpcio-1.60.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:6e306b97966369b889985a562ede9d99180def39ad42c8014628dd3cc343f508"}, + {file = "grpcio-1.60.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:f897c3b127532e6befdcf961c415c97f320d45614daf84deba0a54e64ea2457b"}, + {file = "grpcio-1.60.0-cp310-cp310-win32.whl", hash = "sha256:b87efe4a380887425bb15f220079aa8336276398dc33fce38c64d278164f963d"}, + {file = "grpcio-1.60.0-cp310-cp310-win_amd64.whl", hash = "sha256:a9c7b71211f066908e518a2ef7a5e211670761651039f0d6a80d8d40054047df"}, + {file = "grpcio-1.60.0-cp311-cp311-linux_armv7l.whl", hash = "sha256:fb464479934778d7cc5baf463d959d361954d6533ad34c3a4f1d267e86ee25fd"}, + {file = "grpcio-1.60.0-cp311-cp311-macosx_10_10_universal2.whl", hash = "sha256:4b44d7e39964e808b071714666a812049765b26b3ea48c4434a3b317bac82f14"}, + {file = "grpcio-1.60.0-cp311-cp311-manylinux_2_17_aarch64.whl", hash = "sha256:90bdd76b3f04bdb21de5398b8a7c629676c81dfac290f5f19883857e9371d28c"}, + {file = "grpcio-1.60.0-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:91229d7203f1ef0ab420c9b53fe2ca5c1fbeb34f69b3bc1b5089466237a4a134"}, + {file = "grpcio-1.60.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3b36a2c6d4920ba88fa98075fdd58ff94ebeb8acc1215ae07d01a418af4c0253"}, + {file = "grpcio-1.60.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:297eef542156d6b15174a1231c2493ea9ea54af8d016b8ca7d5d9cc65cfcc444"}, + {file = "grpcio-1.60.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:87c9224acba0ad8bacddf427a1c2772e17ce50b3042a789547af27099c5f751d"}, + {file = "grpcio-1.60.0-cp311-cp311-win32.whl", hash = "sha256:95ae3e8e2c1b9bf671817f86f155c5da7d49a2289c5cf27a319458c3e025c320"}, + {file = "grpcio-1.60.0-cp311-cp311-win_amd64.whl", hash = "sha256:467a7d31554892eed2aa6c2d47ded1079fc40ea0b9601d9f79204afa8902274b"}, + {file = "grpcio-1.60.0-cp312-cp312-linux_armv7l.whl", hash = "sha256:a7152fa6e597c20cb97923407cf0934e14224af42c2b8d915f48bc3ad2d9ac18"}, + {file = "grpcio-1.60.0-cp312-cp312-macosx_10_10_universal2.whl", hash = "sha256:7db16dd4ea1b05ada504f08d0dca1cd9b926bed3770f50e715d087c6f00ad748"}, + {file = "grpcio-1.60.0-cp312-cp312-manylinux_2_17_aarch64.whl", hash = "sha256:b0571a5aef36ba9177e262dc88a9240c866d903a62799e44fd4aae3f9a2ec17e"}, + {file = "grpcio-1.60.0-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6fd9584bf1bccdfff1512719316efa77be235469e1e3295dce64538c4773840b"}, + {file = "grpcio-1.60.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d6a478581b1a1a8fdf3318ecb5f4d0cda41cacdffe2b527c23707c9c1b8fdb55"}, + {file = "grpcio-1.60.0-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:77c8a317f0fd5a0a2be8ed5cbe5341537d5c00bb79b3bb27ba7c5378ba77dbca"}, + {file = "grpcio-1.60.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:1c30bb23a41df95109db130a6cc1b974844300ae2e5d68dd4947aacba5985aa5"}, + {file = "grpcio-1.60.0-cp312-cp312-win32.whl", hash = "sha256:2aef56e85901c2397bd557c5ba514f84de1f0ae5dd132f5d5fed042858115951"}, + {file = "grpcio-1.60.0-cp312-cp312-win_amd64.whl", hash = "sha256:e381fe0c2aa6c03b056ad8f52f8efca7be29fb4d9ae2f8873520843b6039612a"}, + {file = "grpcio-1.60.0-cp37-cp37m-linux_armv7l.whl", hash = "sha256:92f88ca1b956eb8427a11bb8b4a0c0b2b03377235fc5102cb05e533b8693a415"}, + {file = "grpcio-1.60.0-cp37-cp37m-macosx_10_10_universal2.whl", hash = "sha256:e278eafb406f7e1b1b637c2cf51d3ad45883bb5bd1ca56bc05e4fc135dfdaa65"}, + {file = "grpcio-1.60.0-cp37-cp37m-manylinux_2_17_aarch64.whl", hash = "sha256:a48edde788b99214613e440fce495bbe2b1e142a7f214cce9e0832146c41e324"}, + {file = "grpcio-1.60.0-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:de2ad69c9a094bf37c1102b5744c9aec6cf74d2b635558b779085d0263166454"}, + {file = "grpcio-1.60.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:073f959c6f570797272f4ee9464a9997eaf1e98c27cb680225b82b53390d61e6"}, + {file = "grpcio-1.60.0-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:c826f93050c73e7769806f92e601e0efdb83ec8d7c76ddf45d514fee54e8e619"}, + {file = "grpcio-1.60.0-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:9e30be89a75ee66aec7f9e60086fadb37ff8c0ba49a022887c28c134341f7179"}, + {file = "grpcio-1.60.0-cp37-cp37m-win_amd64.whl", hash = "sha256:b0fb2d4801546598ac5cd18e3ec79c1a9af8b8f2a86283c55a5337c5aeca4b1b"}, + {file = "grpcio-1.60.0-cp38-cp38-linux_armv7l.whl", hash = "sha256:9073513ec380434eb8d21970e1ab3161041de121f4018bbed3146839451a6d8e"}, + {file = "grpcio-1.60.0-cp38-cp38-macosx_10_10_universal2.whl", hash = "sha256:74d7d9fa97809c5b892449b28a65ec2bfa458a4735ddad46074f9f7d9550ad13"}, + {file = "grpcio-1.60.0-cp38-cp38-manylinux_2_17_aarch64.whl", hash = "sha256:1434ca77d6fed4ea312901122dc8da6c4389738bf5788f43efb19a838ac03ead"}, + {file = "grpcio-1.60.0-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e61e76020e0c332a98290323ecfec721c9544f5b739fab925b6e8cbe1944cf19"}, + {file = "grpcio-1.60.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:675997222f2e2f22928fbba640824aebd43791116034f62006e19730715166c0"}, + {file = "grpcio-1.60.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:5208a57eae445ae84a219dfd8b56e04313445d146873117b5fa75f3245bc1390"}, + {file = "grpcio-1.60.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:428d699c8553c27e98f4d29fdc0f0edc50e9a8a7590bfd294d2edb0da7be3629"}, + {file = "grpcio-1.60.0-cp38-cp38-win32.whl", hash = "sha256:83f2292ae292ed5a47cdcb9821039ca8e88902923198f2193f13959360c01860"}, + {file = "grpcio-1.60.0-cp38-cp38-win_amd64.whl", hash = "sha256:705a68a973c4c76db5d369ed573fec3367d7d196673fa86614b33d8c8e9ebb08"}, + {file = "grpcio-1.60.0-cp39-cp39-linux_armv7l.whl", hash = "sha256:c193109ca4070cdcaa6eff00fdb5a56233dc7610216d58fb81638f89f02e4968"}, + {file = "grpcio-1.60.0-cp39-cp39-macosx_10_10_universal2.whl", hash = "sha256:676e4a44e740deaba0f4d95ba1d8c5c89a2fcc43d02c39f69450b1fa19d39590"}, + {file = "grpcio-1.60.0-cp39-cp39-manylinux_2_17_aarch64.whl", hash = "sha256:5ff21e000ff2f658430bde5288cb1ac440ff15c0d7d18b5fb222f941b46cb0d2"}, + {file = "grpcio-1.60.0-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4c86343cf9ff7b2514dd229bdd88ebba760bd8973dac192ae687ff75e39ebfab"}, + {file = "grpcio-1.60.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0fd3b3968ffe7643144580f260f04d39d869fcc2cddb745deef078b09fd2b328"}, + {file = "grpcio-1.60.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:30943b9530fe3620e3b195c03130396cd0ee3a0d10a66c1bee715d1819001eaf"}, + {file = "grpcio-1.60.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:b10241250cb77657ab315270b064a6c7f1add58af94befa20687e7c8d8603ae6"}, + {file = "grpcio-1.60.0-cp39-cp39-win32.whl", hash = "sha256:79a050889eb8d57a93ed21d9585bb63fca881666fc709f5d9f7f9372f5e7fd03"}, + {file = "grpcio-1.60.0-cp39-cp39-win_amd64.whl", hash = "sha256:8a97a681e82bc11a42d4372fe57898d270a2707f36c45c6676e49ce0d5c41353"}, + {file = "grpcio-1.60.0.tar.gz", hash = "sha256:2199165a1affb666aa24adf0c97436686d0a61bc5fc113c037701fb7c7fceb96"}, +] + +[package.extras] +protobuf = ["grpcio-tools (>=1.60.0)"] + [[package]] name = "gymnasium" version = "0.29.1" @@ -685,6 +830,23 @@ other = ["lz4 (>=3.1.0)", "matplotlib (>=3.0)", "moviepy (>=1.0.0)", "opencv-pyt testing = ["pytest (==7.1.3)", "scipy (>=1.7.3)"] toy-text = ["pygame (>=2.1.3)", "pygame (>=2.1.3)"] +[[package]] +name = "hydra-core" +version = "1.3.2" +description = "A framework for elegantly configuring complex applications" +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "hydra-core-1.3.2.tar.gz", hash = "sha256:8a878ed67216997c3e9d88a8e72e7b4767e81af37afb4ea3334b269a4390a824"}, + {file = "hydra_core-1.3.2-py3-none-any.whl", hash = "sha256:fa0238a9e31df3373b35b0bfb672c34cc92718d21f81311d8996a16de1141d8b"}, +] + +[package.dependencies] +antlr4-python3-runtime = ">=4.9.0,<4.10.0" +omegaconf = ">=2.2,<2.4" +packaging = "*" + [[package]] name = "identify" version = "2.5.31" @@ -712,6 +874,58 @@ files = [ {file = "idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca"}, ] +[[package]] +name = "imageio" +version = "2.33.1" +description = "Library for reading and writing a wide range of image, video, scientific, and volumetric data formats." +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "imageio-2.33.1-py3-none-any.whl", hash = "sha256:c5094c48ccf6b2e6da8b4061cd95e1209380afafcbeae4a4e280938cce227e1d"}, + {file = "imageio-2.33.1.tar.gz", hash = "sha256:78722d40b137bd98f5ec7312119f8aea9ad2049f76f434748eb306b6937cc1ce"}, +] + +[package.dependencies] +numpy = "*" +pillow = ">=8.3.2" + +[package.extras] +all-plugins = ["astropy", "av", "imageio-ffmpeg", "pillow-heif", "psutil", "tifffile"] +all-plugins-pypy = ["av", "imageio-ffmpeg", "pillow-heif", "psutil", "tifffile"] +build = ["wheel"] +dev = ["black", "flake8", "fsspec[github]", "pytest", "pytest-cov"] +docs = ["numpydoc", "pydata-sphinx-theme", "sphinx (<6)"] +ffmpeg = ["imageio-ffmpeg", "psutil"] +fits = ["astropy"] +full = ["astropy", "av", "black", "flake8", "fsspec[github]", "gdal", "imageio-ffmpeg", "itk", "numpydoc", "pillow-heif", "psutil", "pydata-sphinx-theme", "pytest", "pytest-cov", "sphinx (<6)", "tifffile", "wheel"] +gdal = ["gdal"] +itk = ["itk"] +linting = ["black", "flake8"] +pillow-heif = ["pillow-heif"] +pyav = ["av"] +test = ["fsspec[github]", "pytest", "pytest-cov"] +tifffile = ["tifffile"] + +[[package]] +name = "imageio-ffmpeg" +version = "0.4.9" +description = "FFMPEG wrapper for Python" +category = "main" +optional = false +python-versions = ">=3.5" +files = [ + {file = "imageio-ffmpeg-0.4.9.tar.gz", hash = "sha256:39bcd1660118ef360fa4047456501071364661aa9d9021d3d26c58f1ee2081f5"}, + {file = "imageio_ffmpeg-0.4.9-py3-none-macosx_10_9_intel.macosx_10_9_x86_64.macosx_10_10_intel.macosx_10_10_x86_64.whl", hash = "sha256:24095e882a126a0d217197b86265f821b4bb3cf9004104f67c1384a2b4b49168"}, + {file = "imageio_ffmpeg-0.4.9-py3-none-manylinux2010_x86_64.whl", hash = "sha256:2996c64af3e5489227096580269317719ea1a8121d207f2e28d6c24ebc4a253e"}, + {file = "imageio_ffmpeg-0.4.9-py3-none-manylinux2014_aarch64.whl", hash = "sha256:7eead662d2f46d748c0ab446b68f423eb63d2b54d0a8ef96f80607245540866d"}, + {file = "imageio_ffmpeg-0.4.9-py3-none-win32.whl", hash = "sha256:b6de1e18911687c538d5585d8287ab1a23624ca9dc2044fcc4607de667bcf11e"}, + {file = "imageio_ffmpeg-0.4.9-py3-none-win_amd64.whl", hash = "sha256:7e900c695c6541b1cb17feb1baacd4009b30a53a45b81c23d53a67ab13ffb766"}, +] + +[package.dependencies] +setuptools = "*" + [[package]] name = "importlib-metadata" version = "6.8.0" @@ -1085,6 +1299,25 @@ docs = ["sphinx (>=1.6.0)", "sphinx-bootstrap-theme"] flake8 = ["flake8"] tests = ["psutil", "pytest (!=3.3.0)", "pytest-cov"] +[[package]] +name = "markdown" +version = "3.5.2" +description = "Python implementation of John Gruber's Markdown." +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "Markdown-3.5.2-py3-none-any.whl", hash = "sha256:d43323865d89fc0cb9b20c75fc8ad313af307cc087e84b657d9eec768eddeadd"}, + {file = "Markdown-3.5.2.tar.gz", hash = "sha256:e1ac7b3dc550ee80e602e71c1d168002f062e49f1b11e26a36264dafd4df2ef8"}, +] + +[package.dependencies] +importlib-metadata = {version = ">=4.4", markers = "python_version < \"3.10\""} + +[package.extras] +docs = ["mdx-gh-links (>=0.2)", "mkdocs (>=1.5)", "mkdocs-gen-files", "mkdocs-literate-nav", "mkdocs-nature (>=0.6)", "mkdocs-section-index", "mkdocstrings[python]"] +testing = ["coverage", "pyyaml"] + [[package]] name = "markupsafe" version = "2.1.3" @@ -1272,6 +1505,34 @@ files = [ mcap = ">=0.0.14" protobuf = ">=3.8,<4.22.0 || >=4.25.0" +[[package]] +name = "moviepy" +version = "1.0.3" +description = "Video editing with Python" +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "moviepy-1.0.3.tar.gz", hash = "sha256:2884e35d1788077db3ff89e763c5ba7bfddbd7ae9108c9bc809e7ba58fa433f5"}, +] + +[package.dependencies] +decorator = ">=4.0.2,<5.0" +imageio = {version = ">=2.5,<3.0", markers = "python_version >= \"3.4\""} +imageio_ffmpeg = {version = ">=0.2.0", markers = "python_version >= \"3.4\""} +numpy = [ + {version = ">=1.17.3", markers = "python_version != \"2.7\""}, + {version = "*", markers = "python_version >= \"2.7\""}, +] +proglog = "<=1.0.0" +requests = ">=2.8.1,<3.0" +tqdm = ">=4.11.2,<5.0" + +[package.extras] +doc = ["Sphinx (>=1.5.2,<2.0)", "numpydoc (>=0.6.0,<1.0)", "pygame (>=1.9.3,<2.0)", "sphinx_rtd_theme (>=0.1.10b0,<1.0)"] +optional = ["matplotlib (>=2.0.0,<3.0)", "opencv-python (>=3.0,<4.0)", "scikit-image (>=0.13.0,<1.0)", "scikit-learn", "scipy (>=0.19.0,<1.5)", "youtube_dl"] +test = ["coverage (<5.0)", "coveralls (>=1.1,<2.0)", "pytest (>=3.0.0,<4.0)", "pytest-cov (>=2.5.1,<3.0)", "requests (>=2.8.1,<3.0)"] + [[package]] name = "mpmath" version = "1.3.0" @@ -1336,44 +1597,6 @@ files = [ [package.dependencies] setuptools = "*" -[[package]] -name = "numpy" -version = "1.24.4" -description = "Fundamental package for array computing in Python" -category = "main" -optional = false -python-versions = ">=3.8" -files = [ - {file = "numpy-1.24.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:c0bfb52d2169d58c1cdb8cc1f16989101639b34c7d3ce60ed70b19c63eba0b64"}, - {file = "numpy-1.24.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:ed094d4f0c177b1b8e7aa9cba7d6ceed51c0e569a5318ac0ca9a090680a6a1b1"}, - {file = "numpy-1.24.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:79fc682a374c4a8ed08b331bef9c5f582585d1048fa6d80bc6c35bc384eee9b4"}, - {file = "numpy-1.24.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7ffe43c74893dbf38c2b0a1f5428760a1a9c98285553c89e12d70a96a7f3a4d6"}, - {file = "numpy-1.24.4-cp310-cp310-win32.whl", hash = "sha256:4c21decb6ea94057331e111a5bed9a79d335658c27ce2adb580fb4d54f2ad9bc"}, - {file = "numpy-1.24.4-cp310-cp310-win_amd64.whl", hash = "sha256:b4bea75e47d9586d31e892a7401f76e909712a0fd510f58f5337bea9572c571e"}, - {file = "numpy-1.24.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:f136bab9c2cfd8da131132c2cf6cc27331dd6fae65f95f69dcd4ae3c3639c810"}, - {file = "numpy-1.24.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e2926dac25b313635e4d6cf4dc4e51c8c0ebfed60b801c799ffc4c32bf3d1254"}, - {file = "numpy-1.24.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:222e40d0e2548690405b0b3c7b21d1169117391c2e82c378467ef9ab4c8f0da7"}, - {file = "numpy-1.24.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7215847ce88a85ce39baf9e89070cb860c98fdddacbaa6c0da3ffb31b3350bd5"}, - {file = "numpy-1.24.4-cp311-cp311-win32.whl", hash = "sha256:4979217d7de511a8d57f4b4b5b2b965f707768440c17cb70fbf254c4b225238d"}, - {file = "numpy-1.24.4-cp311-cp311-win_amd64.whl", hash = "sha256:b7b1fc9864d7d39e28f41d089bfd6353cb5f27ecd9905348c24187a768c79694"}, - {file = "numpy-1.24.4-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1452241c290f3e2a312c137a9999cdbf63f78864d63c79039bda65ee86943f61"}, - {file = "numpy-1.24.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:04640dab83f7c6c85abf9cd729c5b65f1ebd0ccf9de90b270cd61935eef0197f"}, - {file = "numpy-1.24.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5425b114831d1e77e4b5d812b69d11d962e104095a5b9c3b641a218abcc050e"}, - {file = "numpy-1.24.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd80e219fd4c71fc3699fc1dadac5dcf4fd882bfc6f7ec53d30fa197b8ee22dc"}, - {file = "numpy-1.24.4-cp38-cp38-win32.whl", hash = "sha256:4602244f345453db537be5314d3983dbf5834a9701b7723ec28923e2889e0bb2"}, - {file = "numpy-1.24.4-cp38-cp38-win_amd64.whl", hash = "sha256:692f2e0f55794943c5bfff12b3f56f99af76f902fc47487bdfe97856de51a706"}, - {file = "numpy-1.24.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:2541312fbf09977f3b3ad449c4e5f4bb55d0dbf79226d7724211acc905049400"}, - {file = "numpy-1.24.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:9667575fb6d13c95f1b36aca12c5ee3356bf001b714fc354eb5465ce1609e62f"}, - {file = "numpy-1.24.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f3a86ed21e4f87050382c7bc96571755193c4c1392490744ac73d660e8f564a9"}, - {file = "numpy-1.24.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d11efb4dbecbdf22508d55e48d9c8384db795e1b7b51ea735289ff96613ff74d"}, - {file = "numpy-1.24.4-cp39-cp39-win32.whl", hash = "sha256:6620c0acd41dbcb368610bb2f4d83145674040025e5536954782467100aa8835"}, - {file = "numpy-1.24.4-cp39-cp39-win_amd64.whl", hash = "sha256:befe2bf740fd8373cf56149a5c23a0f601e82869598d41f8e188a0e9869926f8"}, - {file = "numpy-1.24.4-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:31f13e25b4e304632a4619d0e0777662c2ffea99fcae2029556b17d8ff958aef"}, - {file = "numpy-1.24.4-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:95f7ac6540e95bc440ad77f56e520da5bf877f87dca58bd095288dce8940532a"}, - {file = "numpy-1.24.4-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:e98f220aa76ca2a977fe435f5b04d7b3470c0a2e6312907b37ba6068f26787f2"}, - {file = "numpy-1.24.4.tar.gz", hash = "sha256:80f5e3a4e498641401868df4208b74581206afbee7cf7b8329daae82676d9463"}, -] - [[package]] name = "numpy" version = "1.26.3" @@ -1573,6 +1796,39 @@ files = [ {file = "nvidia_nvtx_cu12-12.1.105-py3-none-win_amd64.whl", hash = "sha256:65f4d98982b31b60026e0e6de73fbdfc09d08a96f4656dd3665ca616a11e1e82"}, ] +[[package]] +name = "oauthlib" +version = "3.2.2" +description = "A generic, spec-compliant, thorough implementation of the OAuth request-signing logic" +category = "main" +optional = false +python-versions = ">=3.6" +files = [ + {file = "oauthlib-3.2.2-py3-none-any.whl", hash = "sha256:8139f29aac13e25d502680e9e19963e83f16838d48a0d71c287fe40e7067fbca"}, + {file = "oauthlib-3.2.2.tar.gz", hash = "sha256:9859c40929662bec5d64f34d01c99e093149682a3f38915dc0655d5a633dd918"}, +] + +[package.extras] +rsa = ["cryptography (>=3.0.0)"] +signals = ["blinker (>=1.4.0)"] +signedtoken = ["cryptography (>=3.0.0)", "pyjwt (>=2.0.0,<3)"] + +[[package]] +name = "omegaconf" +version = "2.3.0" +description = "A flexible configuration library" +category = "main" +optional = false +python-versions = ">=3.6" +files = [ + {file = "omegaconf-2.3.0-py3-none-any.whl", hash = "sha256:7b4df175cdb08ba400f45cae3bdcae7ba8365db4d165fc65fd04b050ab63b46b"}, + {file = "omegaconf-2.3.0.tar.gz", hash = "sha256:d5d4b6d29955cc50ad50c46dc269bcd92c6e00f5f90d23ab5fee7bfca4ba4cc7"}, +] + +[package.dependencies] +antlr4-python3-runtime = ">=4.9.0,<4.10.0" +PyYAML = ">=5.1.0" + [[package]] name = "opencv-python" version = "4.9.0.80" @@ -1844,6 +2100,21 @@ nodeenv = ">=0.11.1" pyyaml = ">=5.1" virtualenv = ">=20.10.0" +[[package]] +name = "proglog" +version = "0.1.10" +description = "Log and progress bar manager for console, notebooks, web..." +category = "main" +optional = false +python-versions = "*" +files = [ + {file = "proglog-0.1.10-py3-none-any.whl", hash = "sha256:19d5da037e8c813da480b741e3fa71fb1ac0a5b02bf21c41577c7f327485ec50"}, + {file = "proglog-0.1.10.tar.gz", hash = "sha256:658c28c9c82e4caeb2f25f488fff9ceace22f8d69b15d0c1c86d64275e4ddab4"}, +] + +[package.dependencies] +tqdm = "*" + [[package]] name = "prompt-toolkit" version = "3.0.40" @@ -1861,23 +2132,26 @@ wcwidth = "*" [[package]] name = "protobuf" -version = "4.25.2" +version = "4.21.12" description = "" category = "main" optional = false -python-versions = ">=3.8" +python-versions = ">=3.7" files = [ - {file = "protobuf-4.25.2-cp310-abi3-win32.whl", hash = "sha256:b50c949608682b12efb0b2717f53256f03636af5f60ac0c1d900df6213910fd6"}, - {file = "protobuf-4.25.2-cp310-abi3-win_amd64.whl", hash = "sha256:8f62574857ee1de9f770baf04dde4165e30b15ad97ba03ceac65f760ff018ac9"}, - {file = "protobuf-4.25.2-cp37-abi3-macosx_10_9_universal2.whl", hash = "sha256:2db9f8fa64fbdcdc93767d3cf81e0f2aef176284071507e3ede160811502fd3d"}, - {file = "protobuf-4.25.2-cp37-abi3-manylinux2014_aarch64.whl", hash = "sha256:10894a2885b7175d3984f2be8d9850712c57d5e7587a2410720af8be56cdaf62"}, - {file = "protobuf-4.25.2-cp37-abi3-manylinux2014_x86_64.whl", hash = "sha256:fc381d1dd0516343f1440019cedf08a7405f791cd49eef4ae1ea06520bc1c020"}, - {file = "protobuf-4.25.2-cp38-cp38-win32.whl", hash = "sha256:33a1aeef4b1927431d1be780e87b641e322b88d654203a9e9d93f218ee359e61"}, - {file = "protobuf-4.25.2-cp38-cp38-win_amd64.whl", hash = "sha256:47f3de503fe7c1245f6f03bea7e8d3ec11c6c4a2ea9ef910e3221c8a15516d62"}, - {file = "protobuf-4.25.2-cp39-cp39-win32.whl", hash = "sha256:5e5c933b4c30a988b52e0b7c02641760a5ba046edc5e43d3b94a74c9fc57c1b3"}, - {file = "protobuf-4.25.2-cp39-cp39-win_amd64.whl", hash = "sha256:d66a769b8d687df9024f2985d5137a337f957a0916cf5464d1513eee96a63ff0"}, - {file = "protobuf-4.25.2-py3-none-any.whl", hash = "sha256:a8b7a98d4ce823303145bf3c1a8bdb0f2f4642a414b196f04ad9853ed0c8f830"}, - {file = "protobuf-4.25.2.tar.gz", hash = "sha256:fe599e175cb347efc8ee524bcd4b902d11f7262c0e569ececcb89995c15f0a5e"}, + {file = "protobuf-4.21.12-cp310-abi3-win32.whl", hash = "sha256:b135410244ebe777db80298297a97fbb4c862c881b4403b71bac9d4107d61fd1"}, + {file = "protobuf-4.21.12-cp310-abi3-win_amd64.whl", hash = "sha256:89f9149e4a0169cddfc44c74f230d7743002e3aa0b9472d8c28f0388102fc4c2"}, + {file = "protobuf-4.21.12-cp37-abi3-macosx_10_9_universal2.whl", hash = "sha256:299ea899484ee6f44604deb71f424234f654606b983cb496ea2a53e3c63ab791"}, + {file = "protobuf-4.21.12-cp37-abi3-manylinux2014_aarch64.whl", hash = "sha256:d1736130bce8cf131ac7957fa26880ca19227d4ad68b4888b3be0dea1f95df97"}, + {file = "protobuf-4.21.12-cp37-abi3-manylinux2014_x86_64.whl", hash = "sha256:78a28c9fa223998472886c77042e9b9afb6fe4242bd2a2a5aced88e3f4422aa7"}, + {file = "protobuf-4.21.12-cp37-cp37m-win32.whl", hash = "sha256:3d164928ff0727d97022957c2b849250ca0e64777ee31efd7d6de2e07c494717"}, + {file = "protobuf-4.21.12-cp37-cp37m-win_amd64.whl", hash = "sha256:f45460f9ee70a0ec1b6694c6e4e348ad2019275680bd68a1d9314b8c7e01e574"}, + {file = "protobuf-4.21.12-cp38-cp38-win32.whl", hash = "sha256:6ab80df09e3208f742c98443b6166bcb70d65f52cfeb67357d52032ea1ae9bec"}, + {file = "protobuf-4.21.12-cp38-cp38-win_amd64.whl", hash = "sha256:1f22ac0ca65bb70a876060d96d914dae09ac98d114294f77584b0d2644fa9c30"}, + {file = "protobuf-4.21.12-cp39-cp39-win32.whl", hash = "sha256:27f4d15021da6d2b706ddc3860fac0a5ddaba34ab679dc182b60a8bb4e1121cc"}, + {file = "protobuf-4.21.12-cp39-cp39-win_amd64.whl", hash = "sha256:237216c3326d46808a9f7c26fd1bd4b20015fb6867dc5d263a493ef9a539293b"}, + {file = "protobuf-4.21.12-py2.py3-none-any.whl", hash = "sha256:a53fd3f03e578553623272dc46ac2f189de23862e68565e83dde203d41b76fc5"}, + {file = "protobuf-4.21.12-py3-none-any.whl", hash = "sha256:b98d0148f84e3a3c569e19f52103ca1feacdac0d2df8d6533cf983d1fda28462"}, + {file = "protobuf-4.21.12.tar.gz", hash = "sha256:7cd532c4566d0e6feafecc1059d04c7915aec8e182d1cf7adee8b24ef1e2e6ab"}, ] [[package]] @@ -1936,6 +2210,33 @@ files = [ [package.extras] tests = ["pytest"] +[[package]] +name = "pyasn1" +version = "0.5.1" +description = "Pure-Python implementation of ASN.1 types and DER/BER/CER codecs (X.208)" +category = "main" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7" +files = [ + {file = "pyasn1-0.5.1-py2.py3-none-any.whl", hash = "sha256:4439847c58d40b1d0a573d07e3856e95333f1976294494c325775aeca506eb58"}, + {file = "pyasn1-0.5.1.tar.gz", hash = "sha256:6d391a96e59b23130a5cfa74d6fd7f388dbbe26cc8f1edf39fdddf08d9d6676c"}, +] + +[[package]] +name = "pyasn1-modules" +version = "0.3.0" +description = "A collection of ASN.1-based protocols modules" +category = "main" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7" +files = [ + {file = "pyasn1_modules-0.3.0-py2.py3-none-any.whl", hash = "sha256:d3ccd6ed470d9ffbc716be08bd90efbd44d0734bc9303818f7336070984a162d"}, + {file = "pyasn1_modules-0.3.0.tar.gz", hash = "sha256:5bd01446b736eb9d31512a30d46c1ac3395d676c6f3cafa4c03eb54b9925631c"}, +] + +[package.dependencies] +pyasn1 = ">=0.4.6,<0.6.0" + [[package]] name = "pycparser" version = "2.21" @@ -2322,6 +2623,40 @@ urllib3 = ">=1.21.1,<3" socks = ["PySocks (>=1.5.6,!=1.5.7)"] use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] +[[package]] +name = "requests-oauthlib" +version = "1.3.1" +description = "OAuthlib authentication support for Requests." +category = "main" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "requests-oauthlib-1.3.1.tar.gz", hash = "sha256:75beac4a47881eeb94d5ea5d6ad31ef88856affe2332b9aafb52c6452ccf0d7a"}, + {file = "requests_oauthlib-1.3.1-py2.py3-none-any.whl", hash = "sha256:2577c501a2fb8d05a304c09d090d6e47c306fef15809d102b327cf8364bddab5"}, +] + +[package.dependencies] +oauthlib = ">=3.0.0" +requests = ">=2.0.0" + +[package.extras] +rsa = ["oauthlib[signedtoken] (>=3.0.0)"] + +[[package]] +name = "rsa" +version = "4.9" +description = "Pure-Python RSA implementation" +category = "main" +optional = false +python-versions = ">=3.6,<4" +files = [ + {file = "rsa-4.9-py3-none-any.whl", hash = "sha256:90260d9058e514786967344d0ef75fa8727eed8a7d2e43ce9f4bcf1b536174f7"}, + {file = "rsa-4.9.tar.gz", hash = "sha256:e38464a49c6c85d7f1351b0126661487a7e0a14a50f1675ec50eb34d4f20ef21"}, +] + +[package.dependencies] +pyasn1 = ">=0.1.3" + [[package]] name = "scipy" version = "1.11.4" @@ -2654,6 +2989,44 @@ files = [ [package.dependencies] mpmath = ">=0.19" +[[package]] +name = "tensorboard" +version = "2.15.1" +description = "TensorBoard lets you watch Tensors Flow" +category = "main" +optional = false +python-versions = ">=3.9" +files = [ + {file = "tensorboard-2.15.1-py3-none-any.whl", hash = "sha256:c46c1d1cf13a458c429868a78b2531d8ff5f682058d69ec0840b0bc7a38f1c0f"}, +] + +[package.dependencies] +absl-py = ">=0.4" +google-auth = ">=1.6.3,<3" +google-auth-oauthlib = ">=0.5,<2" +grpcio = ">=1.48.2" +markdown = ">=2.6.8" +numpy = ">=1.12.0" +protobuf = ">=3.19.6,<4.24" +requests = ">=2.21.0,<3" +setuptools = ">=41.0.0" +six = ">1.9" +tensorboard-data-server = ">=0.7.0,<0.8.0" +werkzeug = ">=1.0.1" + +[[package]] +name = "tensorboard-data-server" +version = "0.7.2" +description = "Fast data loading for TensorBoard" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tensorboard_data_server-0.7.2-py3-none-any.whl", hash = "sha256:7e0610d205889588983836ec05dc098e80f97b7e7bbff7e994ebb78f578d0ddb"}, + {file = "tensorboard_data_server-0.7.2-py3-none-macosx_10_9_x86_64.whl", hash = "sha256:9fe5d24221b29625dbc7328b0436ca7fc1c23de4acf4d272f1180856e32f9f60"}, + {file = "tensorboard_data_server-0.7.2-py3-none-manylinux_2_31_x86_64.whl", hash = "sha256:ef687163c24185ae9754ed5650eb5bc4d84ff257aabdc33f0cc6f74d8ba54530"}, +] + [[package]] name = "tomli" version = "2.0.1" @@ -2741,6 +3114,27 @@ files = [ {file = "tornado-6.3.3.tar.gz", hash = "sha256:e7d8db41c0181c80d76c982aacc442c0783a2c54d6400fe028954201a2e032fe"}, ] +[[package]] +name = "tqdm" +version = "4.66.1" +description = "Fast, Extensible Progress Meter" +category = "main" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tqdm-4.66.1-py3-none-any.whl", hash = "sha256:d302b3c5b53d47bce91fea46679d9c3c6508cf6332229aa1e7d8653723793386"}, + {file = "tqdm-4.66.1.tar.gz", hash = "sha256:d88e651f9db8d8551a62556d3cff9e3034274ca5d66e93197cf2490e2dcb69c7"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "platform_system == \"Windows\""} + +[package.extras] +dev = ["pytest (>=6)", "pytest-cov", "pytest-timeout", "pytest-xdist"] +notebook = ["ipywidgets (>=6)"] +slack = ["slack-sdk"] +telegram = ["requests"] + [[package]] name = "traitlets" version = "5.13.0" @@ -2898,6 +3292,24 @@ files = [ {file = "wcwidth-0.2.9.tar.gz", hash = "sha256:a675d1a4a2d24ef67096a04b85b02deeecd8e226f57b5e3a72dbb9ed99d27da8"}, ] +[[package]] +name = "werkzeug" +version = "3.0.1" +description = "The comprehensive WSGI web application library." +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "werkzeug-3.0.1-py3-none-any.whl", hash = "sha256:90a285dc0e42ad56b34e696398b8122ee4c681833fb35b8334a095d82c56da10"}, + {file = "werkzeug-3.0.1.tar.gz", hash = "sha256:507e811ecea72b18a404947aded4b3390e1db8f826b494d76550ef45bb3b1dcc"}, +] + +[package.dependencies] +MarkupSafe = ">=2.1.1" + +[package.extras] +watchdog = ["watchdog (>=2.3)"] + [[package]] name = "zipp" version = "3.17.0" @@ -2979,4 +3391,4 @@ cffi = ["cffi (>=1.11)"] [metadata] lock-version = "2.0" python-versions = ">=3.9,<4.0" -content-hash = "3a6cdd7ca2420e9dbc35fb15231ffcd6579f6a7a94f5183edaecc3b7c2c38656" +content-hash = "6e13d163638d8f1e233c3b66e50973d1fa212df26b1a0a859da22a0656665edf" diff --git a/pyproject.toml b/pyproject.toml index 351eefa..146826a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -20,6 +20,9 @@ mcap-protobuf-support = "^0.4.1" scipy = "^1.11.4" pygame = "^2.5.2" opencv-python = "^4.9.0.80" +hydra-core = "^1.3.2" +tensorboard = "^2.15.1" +moviepy = "^1.0.3" [tool.poetry.group.dev.dependencies] diff --git a/scripts/configs/algo/ppo.yaml b/scripts/configs/algo/ppo.yaml new file mode 100644 index 0000000..27019b1 --- /dev/null +++ b/scripts/configs/algo/ppo.yaml @@ -0,0 +1,4 @@ +_target_: stable_baselines3.PPO +policy: "MlpPolicy" +use_sde: True +sde_sample_freq: 64 diff --git a/scripts/configs/algo/sac.yaml b/scripts/configs/algo/sac.yaml index 41faacf..5f279c7 100644 --- a/scripts/configs/algo/sac.yaml +++ b/scripts/configs/algo/sac.yaml @@ -1,5 +1,6 @@ # refer to https://stable-baselines3.readthedocs.io/en/master/modules/sac.html -_target_: furuta_gym.algos.SAC +_target_: furuta.rl.algos.SAC +# _target_: stable_baselines3.SAC policy: "MlpPolicy" learning_rate: 0.0003 buffer_size: 1_000_000 @@ -11,5 +12,6 @@ learning_starts: 500 use_sde: True use_sde_at_warmup: True sde_sample_freq: 64 -train_freq: [1, "episode"] +# train_freq: [1, "episode"] +train_freq: 1 gradient_steps: -1 diff --git a/scripts/configs/experiment/sim.yaml b/scripts/configs/experiment/sim.yaml index 84fb15e..f2aa89d 100644 --- a/scripts/configs/experiment/sim.yaml +++ b/scripts/configs/experiment/sim.yaml @@ -4,4 +4,4 @@ defaults: - override /wrappers: base_wrappers gym_id: "FurutaSim-v0" -total_timesteps: 50_000 +total_timesteps: 1_000_000 diff --git a/scripts/configs/train.yaml b/scripts/configs/train.yaml index 46822e3..f61e2a5 100644 --- a/scripts/configs/train.yaml +++ b/scripts/configs/train.yaml @@ -6,15 +6,15 @@ defaults: - wrappers: ??? - algo: sac.yaml -gym_id: ??? -total_timesteps: 1_000_000 +total_timesteps: 500_000 +n_envs: 4 # eval & early stopping evaluation: early_stopping_reward_threshold: 800 # use null for no early stopping deterministic: True n_eval_episodes: 10 - eval_freq: 500 + eval_freq: 50_000 # reproducibility seed: 1 @@ -34,4 +34,3 @@ model_artifact: null # logging save_model: False save_replay_buffer: False -# add mcap diff --git a/scripts/configs/wrappers/base_wrappers.yaml b/scripts/configs/wrappers/base_wrappers.yaml index 3bcc168..571ba59 100644 --- a/scripts/configs/wrappers/base_wrappers.yaml +++ b/scripts/configs/wrappers/base_wrappers.yaml @@ -2,11 +2,11 @@ # looks like the order is preserved but unsure if there could be edge cases? # the order of the wrappers is important TimeLimit: - module: "gym.wrappers" + module: "gymnasium.wrappers" max_episode_steps: 1000 HistoryWrapper: - module: "furuta_gym.wrappers" + module: "furuta.rl.wrappers" steps: 2 use_continuity_cost: True diff --git a/scripts/configs/wrappers/real_wrappers.yaml b/scripts/configs/wrappers/real_wrappers.yaml index 0179562..f2fd0f1 100644 --- a/scripts/configs/wrappers/real_wrappers.yaml +++ b/scripts/configs/wrappers/real_wrappers.yaml @@ -1,4 +1,5 @@ # import common wrappers between sim/real defaults: - base_wrappers -# add wrappers specific to the robot +# add wrappers specific to the robot: +# gently terminating and rate limiter diff --git a/scripts/train.py b/scripts/train.py index b1590ae..2f14f46 100644 --- a/scripts/train.py +++ b/scripts/train.py @@ -5,7 +5,7 @@ import random from pathlib import Path -import gym +import gymnasium as gym import hydra import numpy as np import stable_baselines3 @@ -16,43 +16,21 @@ EvalCallback, StopTrainingOnRewardThreshold, ) -from stable_baselines3.common.vec_env import DummyVecEnv, VecVideoRecorder - -import furuta # noqa F420 - -# TODO -# - save model/replay buffer X -# - finish setup wrappers: think about how to setup; maybe have some hardcoded wrappers? -# - load custom sim parameters X - - -def download_artifact_file(artifact_alias, filename): - """Download artifact and returns path to filename. - - :param artifact_name: wandb artifact alias - :param filename: filename in the artifact - """ - logging.info(f"loading {filename} from {artifact_alias}") - - artifact = wandb.use_artifact(artifact_alias) - artifact_dir = Path(artifact.download()) - filepath = artifact_dir / filename - - assert filepath.is_file(), f"{artifact_alias} doesn't contain {filename}" - - return filepath - - -def upload_file_to_artifacts(pth, artifact_name, artifact_type): - logging.info(f"Saving {pth} to {artifact_name}") - if not isinstance(pth, Path): - pth = Path(pth) +from stable_baselines3.common.env_checker import check_env +from stable_baselines3.common.vec_env import ( + DummyVecEnv, + SubprocVecEnv, + VecVideoRecorder, +) - assert os.path.isfile(pth), f"{pth} is not a file" +from furuta.rl.envs.furuta_real import FurutaReal +from furuta.rl.utils import ( + download_artifact_file, + seed_everything, + upload_file_to_artifacts, +) - artifact = wandb.Artifact(artifact_name, type=artifact_type) - artifact.add_file(pth) - wandb.log_artifact(artifact) +# import furuta # noqa F420 def instantiate_gym_wrapper(wrapper_name: str, wrapper_args: DictConfig, env: gym.Env) -> gym.Env: @@ -69,7 +47,6 @@ def instantiate_gym_wrapper(wrapper_name: str, wrapper_args: DictConfig, env: gy @hydra.main(version_base="1.3", config_path="configs", config_name="train.yaml") def main(cfg: DictConfig): - print(cfg) logging_level = logging.DEBUG if cfg.debug else logging.INFO logging.basicConfig(format="%(levelname)s: %(message)s", level=logging_level) @@ -85,16 +62,11 @@ def main(cfg: DictConfig): ) # setup env - env = gym.make(cfg.gym_id, **cfg.env) + env = hydra.utils.instantiate(cfg.env, _recursive_=True) + check_env(env) # seed everything - env.seed(cfg.seed) - env.action_space.seed(cfg.seed) - env.observation_space.seed(cfg.seed) - random.seed(cfg.seed) - np.random.seed(cfg.seed) - torch.manual_seed(cfg.seed) - torch.backends.cudnn.deterministic = cfg.cudnn_deterministic + seed_everything(env, cfg.seed, cfg.cudnn_deterministic) # setup wrappers for wrapper_name, wrapper_args in cfg.wrappers.items(): @@ -104,10 +76,15 @@ def main(cfg: DictConfig): # print stack trace logging.warning(e, exc_info=True) + if isinstance(env.unwrapped, FurutaReal): + vec_env = DummyVecEnv([lambda: env]) + else: + vec_env = SubprocVecEnv([lambda: copy.deepcopy(env) for _ in range(cfg.n_envs)]) + # setup algo/model verbose = 2 if cfg.debug else 0 model = hydra.utils.instantiate( - cfg.algo, env=env, tensorboard_log=f"runs/{run.id}", verbose=verbose + cfg.algo, env=vec_env, tensorboard_log=f"runs/{run.id}", verbose=verbose, _convert_="all" ) # load model/replay buffer @@ -132,11 +109,14 @@ def main(cfg: DictConfig): # use same env for eval # TODO maybe do eval even when we don't want to early stop? # would it be useful? + # accounting for vec envs + + eval_freq = max(cfg.evaluation.eval_env // cfg.n_envs, 1) eval_callback = EvalCallback( eval_env, deterministic=cfg.evaluation.deterministic, n_eval_episodes=cfg.evaluation.n_eval_episodes, - eval_freq=cfg.evaluation.eval_freq, + eval_freq=eval_freq, callback_on_new_best=callback_on_best, verbose=1, ) @@ -161,7 +141,7 @@ def main(cfg: DictConfig): video_length=video_length, ) - obs = env.reset() + obs = env.reset(options={"random_init": False}) for _ in range(video_length + 1): action, _ = model.predict(obs, deterministic=False) obs, _, done, _ = env.step(action) From 0d2179148638199748064daba2233bb7c703b6de Mon Sep 17 00:00:00 2001 From: Armand Date: Mon, 15 Jan 2024 13:53:55 +0100 Subject: [PATCH 09/44] add missing configs --- .gitignore | 2 +- scripts/configs/env/dyn/default.yaml | 1 + scripts/configs/env/sim.yaml | 11 +++++++++++ 3 files changed, 13 insertions(+), 1 deletion(-) create mode 100644 scripts/configs/env/dyn/default.yaml create mode 100644 scripts/configs/env/sim.yaml diff --git a/.gitignore b/.gitignore index 8aa8029..ebff421 100644 --- a/.gitignore +++ b/.gitignore @@ -125,7 +125,7 @@ celerybeat.pid # Environments .env .venv -env/ +# env/ bc we use env for gym env config but have no env/ dir for actual venvs venv/ ENV/ env.bak/ diff --git a/scripts/configs/env/dyn/default.yaml b/scripts/configs/env/dyn/default.yaml new file mode 100644 index 0000000..6a651dd --- /dev/null +++ b/scripts/configs/env/dyn/default.yaml @@ -0,0 +1 @@ +_target_: furuta.robot.QubeDynamics diff --git a/scripts/configs/env/sim.yaml b/scripts/configs/env/sim.yaml new file mode 100644 index 0000000..8267302 --- /dev/null +++ b/scripts/configs/env/sim.yaml @@ -0,0 +1,11 @@ +defaults: + - _self_ + - dyn: default + +_target_: furuta.rl.envs.furuta_sim.FurutaSim +control_freq: 50 +reward: "alpha" +state_limits: [3.14, 12, 50, 400] # TODO how to use PI in configs? +encoders_CPRs: null +velocity_filter: 2 +render_mode: "rgb_array" From efec34aa7af92424a440f3d464b1e1d78717f13e Mon Sep 17 00:00:00 2001 From: armandpl Date: Mon, 15 Jan 2024 16:43:20 +0100 Subject: [PATCH 10/44] fix eval freq, fix reset vec env --- scripts/train.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/scripts/train.py b/scripts/train.py index 2f14f46..4949e32 100644 --- a/scripts/train.py +++ b/scripts/train.py @@ -111,7 +111,7 @@ def main(cfg: DictConfig): # would it be useful? # accounting for vec envs - eval_freq = max(cfg.evaluation.eval_env // cfg.n_envs, 1) + eval_freq = max(cfg.evaluation.eval_freq // cfg.n_envs, 1) eval_callback = EvalCallback( eval_env, deterministic=cfg.evaluation.deterministic, @@ -141,7 +141,10 @@ def main(cfg: DictConfig): video_length=video_length, ) - obs = env.reset(options={"random_init": False}) + # vec so we can't pass reset options + # TODO random init should be an env param? + # obs = env.reset(options={"random_init": False}) + obs = env.reset() for _ in range(video_length + 1): action, _ = model.predict(obs, deterministic=False) obs, _, done, _ = env.step(action) From ccdd56b4980c913dfabe3eb3aeec197481ff09e5 Mon Sep 17 00:00:00 2001 From: Armand Date: Wed, 17 Jan 2024 17:28:09 +0100 Subject: [PATCH 11/44] fix path in lint ignores --- .github/workflows/lint.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index 4fc7fc3..1ebe13f 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -23,4 +23,4 @@ jobs: uses: py-actions/flake8@v1 with: ignore: "E203,E402,E501,F401,F841" - exclude: "logs/*,data/*,furuta_gym/logging/protobuf/*" + exclude: "logs/*,data/*,furuta/logging/protobuf/*" From be977c50709bd67d7e704073376be4e84de5f7dc Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 14:06:58 +0100 Subject: [PATCH 12/44] match github action python version to min project python --- .github/workflows/lint.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index 1ebe13f..d590039 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -18,7 +18,7 @@ jobs: - name: Set up Python environment uses: actions/setup-python@v1 with: - python-version: "3.8" + python-version: "3.9" - name: flake8 Lint uses: py-actions/flake8@v1 with: From 4057ab7269bea1628a50084a6561e482ee9edc4b Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 14:27:07 +0100 Subject: [PATCH 13/44] make reset not dependant on the velocity filter --- furuta/rl/envs/furuta_real.py | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/furuta/rl/envs/furuta_real.py b/furuta/rl/envs/furuta_real.py index b8210ac..cf32af7 100644 --- a/furuta/rl/envs/furuta_real.py +++ b/furuta/rl/envs/furuta_real.py @@ -8,7 +8,11 @@ from furuta.robot import Robot from furuta.utils import ALPHA, ALPHA_DOT, THETA, THETA_DOT, VelocityFilter -MAX_RESET_TIME = 5 # seconds +MAX_RESET_TIME = 7 # seconds +RESET_TIME = 0.5 +ALPHA_THRESH = np.deg2rad( + 2 +) # alpha should stay between -2 and 2 deg for 0.5 sec for us to consider the env reset class FurutaReal(FurutaBase): @@ -45,10 +49,16 @@ def reset( # wait for pendulum to fall back to start position reset_time = 0 - while np.abs(self._state[ALPHA_DOT]) > 0.1 and reset_time < MAX_RESET_TIME: + time_under_thresh = 0 + + while time_under_thresh < RESET_TIME and reset_time < MAX_RESET_TIME: sleep(0.01) - self._update_state(0.0) + if abs(self._state[ALPHA]) < ALPHA_THRESH: + time_under_thresh += 0.01 + else: + time_under_thresh = 0 reset_time += 0.01 + self._update_state(0.0) if reset_time >= MAX_RESET_TIME: logging.error("Reset timeout") @@ -64,6 +74,7 @@ def reset( # replace by taking webcam snapshot # and outputing rgb array? # but webcam can't record at 100hz + # or could just use the same render function!!! # def render(self, mode='human'): # raise NotImplementedError From 9e5872f7fc3db7beae19bb36b619141834eda875 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 14:30:41 +0100 Subject: [PATCH 14/44] fix control freq wrapper --- furuta/rl/wrappers.py | 17 +++++++++++++---- 1 file changed, 13 insertions(+), 4 deletions(-) diff --git a/furuta/rl/wrappers.py b/furuta/rl/wrappers.py index cecd246..9c4b475 100644 --- a/furuta/rl/wrappers.py +++ b/furuta/rl/wrappers.py @@ -18,7 +18,7 @@ class GentlyTerminating(gym.Wrapper): def step(self, action): observation, reward, terminated, truncated, info = self.env.step(action) if terminated or truncated: - logging.debug("episode done, killing motor.") + logging.info("episode done, killing motor.") self.unwrapped.robot.step(0.0) return observation, reward, terminated, truncated, info @@ -103,24 +103,33 @@ def close_mcap_writer(self): class ControlFrequency(gym.Wrapper): """Enforce a sleeping time (dt) between each step.""" - def __init__(self, env, dt): + def __init__(self, env, dt, log_freq=3000): super().__init__(env) self.dt = dt + self.last = None + + # TODO make control freq reporting better + # or maybe not, maybe we should bench speed in designated script see #55 + # average control freq over many steps + self.log_freq = log_freq + self.nb_steps = 0 def step(self, action): + self.nb_steps += 1 current = time.time() - _, _, _, _ = self.env.step(action) loop_time = 0 if self.last is not None: loop_time = current - self.last sleeping_time = self.dt - loop_time + if self.nb_steps % self.log_freq == 0: + logging.info(f"control freq: {1/loop_time}") if sleeping_time > 0: time.sleep(sleeping_time) else: logging.info("warning, loop time > dt") - obs, reward, terminated, truncated, info = self.env.step(np.array([0.0])) + obs, reward, terminated, truncated, info = self.env.step(action) self.last = time.time() if logging.root.level == logging.DEBUG: From 7692f0e795a76e15875659d8b306d392695efb63 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 14:44:20 +0100 Subject: [PATCH 15/44] handle random init state later --- furuta/rl/envs/furuta_sim.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/furuta/rl/envs/furuta_sim.py b/furuta/rl/envs/furuta_sim.py index b71e30a..304912b 100644 --- a/furuta/rl/envs/furuta_sim.py +++ b/furuta/rl/envs/furuta_sim.py @@ -31,7 +31,7 @@ def __init__( else: self.vel_filt = None - def _init_state(self, random_init: bool = True): + def _init_state(self): # TODO could also sample from state space # though we also use it as upper speed limit # the two use case are kind of conflicting @@ -91,7 +91,7 @@ def reset( ): if options is None: options = {} - self._init_state(options.get("random_init", True)) + self._init_state() obs = self.get_obs() return obs, {} From 69c3876455f6de0bc639376d7632ab95a42bf96e Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 14:45:32 +0100 Subject: [PATCH 16/44] separate state_max for safety from obs max for learning --- furuta/rl/envs/furuta_base.py | 49 ++++++++++++++++------------------- 1 file changed, 22 insertions(+), 27 deletions(-) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index 05f9e26..2da5654 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -1,5 +1,3 @@ -from collections import namedtuple -from dataclasses import dataclass from typing import Optional import gymnasium as gym @@ -32,10 +30,17 @@ def alpha_theta_reward(state): class FurutaBase(gym.Env): metadata = { "render_modes": ["rgb_array", "human"], - "render_fps": 50, # TODO should this be the same as the control freq? - } # TODO add headless mode? + "render_fps": 50, # TODO should this be the same as the control freq/sim dt? + } - def __init__(self, control_freq, reward, state_limits=None, render_mode="rgb_array"): + def __init__( + self, + control_freq, + reward, + angle_limits=[np.pi, np.pi], # used to help convergence + speed_limits=[60, 400], # used to avoid damaging the robot + render_mode="rgb_array", + ): self.render_mode = render_mode self.timing = Timing(control_freq) @@ -46,20 +51,24 @@ def __init__(self, control_freq, reward, state_limits=None, render_mode="rgb_arr self.screen_height = 400 self.screen = None self.clock = None - self.isopen = True self._reward_func = REWARDS[self.reward] act_max = np.array([1.0], dtype=np.float32) - if state_limits: - self.state_max = np.array(state_limits, dtype=np.float32) - else: - self.state_max = np.array([np.inf, np.inf, np.inf, np.inf], dtype=np.float32) + if angle_limits is None: + angle_limits = [np.inf, np.inf] + if speed_limits is None: + speed_limits = [np.inf, np.inf] - obs_max = np.array( - [1.0, 1.0, 1.0, 1.0, self.state_max[2], self.state_max[3]], dtype=np.float32 + state_limits = np.array( + [angle_limits[0], angle_limits[1], speed_limits[0], speed_limits[1]] ) + self.state_max = np.array(state_limits, dtype=np.float32) + + # max obs based on max speeds measured on the robot + # TODO need to actually measure it + obs_max = np.array([1.0, 1.0, 1.0, 1.0, 50, 50], dtype=np.float32) # Spaces self.state_space = Box( @@ -84,18 +93,6 @@ def __init__(self, control_freq, reward, state_limits=None, render_mode="rgb_arr ) def step(self, action): - # TODO this is slow, do we even need it? - # sb3 knows the env action space, probably it won't pass invalid actions - - # assert a is not None, "Action should be not None" - # assert isinstance(a, np.ndarray), "The action should be a ndarray" - # assert np.all(not np.isnan(a)), "Action NaN is not a valid action" - # assert a.ndim == 1, "The action = {a} must be 1d but the input is {a.ndim}d" - # err_msg = f"{action!r} ({type(action)}) invalid" - - # assert self.action_space.contains(action), "Action is not in action space" - # assert self._state is not None, "Call reset before using step method." - # first read the robot/sim state rwd = self._reward_func(self._state) obs = self.get_obs() @@ -107,7 +104,7 @@ def step(self, action): "motor_angle_velocity": float(self._state[THETA_DOT]), "pendulum_angle_velocity": float(self._state[ALPHA_DOT]), "reward": float(rwd), - "action": float(action), + "action": float(action[0]), } # then take action/step the sim @@ -126,7 +123,6 @@ def step(self, action): def get_obs(self): return np.float32( [ - # TODO maybe call cos, sin at once? save a bit of time np.cos(self._state[THETA]), np.sin(self._state[THETA]), np.cos(self._state[ALPHA]), @@ -241,4 +237,3 @@ def close(self): pygame.display.quit() pygame.quit() - self.isopen = False From ffbdafe84b258f47ce3b3b627f33db38cf5e7960 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 14:51:04 +0100 Subject: [PATCH 17/44] use control freq already set in the env --- furuta/rl/wrappers.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/furuta/rl/wrappers.py b/furuta/rl/wrappers.py index 9c4b475..4fccd51 100644 --- a/furuta/rl/wrappers.py +++ b/furuta/rl/wrappers.py @@ -43,7 +43,7 @@ def __init__(self, env: gym.Env, log_dir: Union[str, Path], use_sim_time: bool): self.mcap_writer = None def step(self, action): - observation, reward, done, info = self.env.step(action) + observation, reward, terminated, truncated, info = self.env.step(action) if self.use_sim_time: self.sim_time += self.unwrapped.timing.dt @@ -62,7 +62,7 @@ def step(self, action): publish_time=time_to_log, ) - return observation, reward, done, info + return observation, reward, terminated, truncated, info def reset( self, @@ -103,9 +103,9 @@ def close_mcap_writer(self): class ControlFrequency(gym.Wrapper): """Enforce a sleeping time (dt) between each step.""" - def __init__(self, env, dt, log_freq=3000): + def __init__(self, env, log_freq=3000): super().__init__(env) - self.dt = dt + self.dt = env.unwrapped.timing.dt self.last = None # TODO make control freq reporting better From 865aff2f7d32ae1ec868fd924e1b101b692c0ac8 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 14:55:43 +0100 Subject: [PATCH 18/44] setup configs for training on real robot --- scripts/configs/algo/sac.yaml | 3 +-- scripts/configs/algo/sac_real.yaml | 5 +++++ scripts/configs/env/dyn/tiny.yaml | 2 ++ scripts/configs/env/real.yaml | 5 +++++ scripts/configs/experiment/real_finetune.yaml | 10 ++++++++++ scripts/configs/wrappers/real_wrappers.yaml | 13 +++++++++++-- 6 files changed, 34 insertions(+), 4 deletions(-) create mode 100644 scripts/configs/algo/sac_real.yaml create mode 100644 scripts/configs/env/dyn/tiny.yaml create mode 100644 scripts/configs/env/real.yaml create mode 100644 scripts/configs/experiment/real_finetune.yaml diff --git a/scripts/configs/algo/sac.yaml b/scripts/configs/algo/sac.yaml index 5f279c7..1709d95 100644 --- a/scripts/configs/algo/sac.yaml +++ b/scripts/configs/algo/sac.yaml @@ -12,6 +12,5 @@ learning_starts: 500 use_sde: True use_sde_at_warmup: True sde_sample_freq: 64 -# train_freq: [1, "episode"] -train_freq: 1 +train_freq: 1 # 1 step gradient_steps: -1 diff --git a/scripts/configs/algo/sac_real.yaml b/scripts/configs/algo/sac_real.yaml new file mode 100644 index 0000000..65f7bbb --- /dev/null +++ b/scripts/configs/algo/sac_real.yaml @@ -0,0 +1,5 @@ +defaults: + - sac + +sde_sample_freq: 32 +train_freq: [1, "episode"] diff --git a/scripts/configs/env/dyn/tiny.yaml b/scripts/configs/env/dyn/tiny.yaml new file mode 100644 index 0000000..027de44 --- /dev/null +++ b/scripts/configs/env/dyn/tiny.yaml @@ -0,0 +1,2 @@ +_target_: furuta.robot.QubeDynamics +# use this file to store the sim params for the tiny version of the robot diff --git a/scripts/configs/env/real.yaml b/scripts/configs/env/real.yaml new file mode 100644 index 0000000..4001f98 --- /dev/null +++ b/scripts/configs/env/real.yaml @@ -0,0 +1,5 @@ +_target_: furuta.rl.envs.furuta_real.FurutaReal +control_freq: 50 +reward: "alpha" +state_limits: [3.14, 12, 50, 400] # TODO how to use PI in configs? +usb_device: "/dev/ttyACM0" diff --git a/scripts/configs/experiment/real_finetune.yaml b/scripts/configs/experiment/real_finetune.yaml new file mode 100644 index 0000000..f628956 --- /dev/null +++ b/scripts/configs/experiment/real_finetune.yaml @@ -0,0 +1,10 @@ +# @package _global_ +defaults: + - override /env: real + - override /wrappers: real_wrappers + - override /algo: sac_real + +total_timesteps: 150_000 +n_envs: 1 +model_artifact: "v358" +capture_video: False diff --git a/scripts/configs/wrappers/real_wrappers.yaml b/scripts/configs/wrappers/real_wrappers.yaml index f2fd0f1..3beb57f 100644 --- a/scripts/configs/wrappers/real_wrappers.yaml +++ b/scripts/configs/wrappers/real_wrappers.yaml @@ -1,5 +1,14 @@ # import common wrappers between sim/real defaults: - base_wrappers -# add wrappers specific to the robot: -# gently terminating and rate limiter + +GentlyTerminating: + module: "furuta.rl.wrappers" + +ControlFrequency: + module: "furuta.rl.wrappers" +# TODO update mcap logger to take terminated and truncated +# MCAPLogger: +# module: "furuta.rl.wrappers" +# use_sim_time: False +# log_dir: "mcap_logs/" From 47b3d75bb6d81d194f5b9c52453a86f9545b0b16 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 16:15:47 +0100 Subject: [PATCH 19/44] remove useless code --- furuta/rl/__init__.py | 3 --- 1 file changed, 3 deletions(-) diff --git a/furuta/rl/__init__.py b/furuta/rl/__init__.py index d6191b9..e69de29 100644 --- a/furuta/rl/__init__.py +++ b/furuta/rl/__init__.py @@ -1,3 +0,0 @@ -# from furuta_gym.envs.furuta_base import FurutaBase # noqa F420 -# from furuta_gym.envs.furuta_real import FurutaReal # noqa F420 -# from furuta_gym.envs.furuta_sim import FurutaSim # noqa F420 From 2d73ad0ff8e6a6cb8faa325c18a7454e8f9a736a Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 16:16:19 +0100 Subject: [PATCH 20/44] properly seed and wrap position between -pi pi when rendering --- furuta/rl/envs/furuta_base.py | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index 2da5654..af7bd79 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -37,8 +37,8 @@ def __init__( self, control_freq, reward, - angle_limits=[np.pi, np.pi], # used to help convergence - speed_limits=[60, 400], # used to avoid damaging the robot + angle_limits=[np.pi, np.pi], # used to help convergence? + speed_limits=[60, 400], # used to avoid damaging the real robot or diverging sim render_mode="rgb_array", ): self.render_mode = render_mode @@ -137,7 +137,7 @@ def reset( seed: Optional[int] = None, options: Optional[dict] = None, ): - raise NotImplementedError + super().reset(seed) def _update_state(self, a): raise NotImplementedError @@ -182,7 +182,10 @@ def render(self): l, r, t, b = -cartwidth / 2, cartwidth / 2, cartheight / 2, -cartheight / 2 axleoffset = cartheight / 4.0 - cartx = x[THETA] * scale + self.screen_width / 2.0 # MIDDLE OF CART + + # make sure theta stays between -pi and pi + theta = (x[THETA] % (2 * np.pi)) - np.pi + cartx = theta * scale + self.screen_width / 2.0 # MIDDLE OF CART carty = 100 # TOP OF CART cart_coords = [(l, b), (l, t), (r, t), (r, b)] cart_coords = [(c[0] + cartx, c[1] + carty) for c in cart_coords] From 3b018eb3302293a93bd388cb889694970ab38221 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 16:17:23 +0100 Subject: [PATCH 21/44] properly seed --- furuta/rl/envs/furuta_real.py | 6 ++++-- furuta/rl/envs/furuta_sim.py | 16 ++++++++-------- 2 files changed, 12 insertions(+), 10 deletions(-) diff --git a/furuta/rl/envs/furuta_real.py b/furuta/rl/envs/furuta_real.py index cf32af7..9344585 100644 --- a/furuta/rl/envs/furuta_real.py +++ b/furuta/rl/envs/furuta_real.py @@ -20,10 +20,11 @@ def __init__( self, control_freq=100, reward="alpha_theta", - state_limits=None, + angle_limits=None, + speed_limits=None, usb_device="/dev/ttyACM0", ): - super().__init__(control_freq, reward, state_limits) + super().__init__(control_freq, reward, angle_limits, speed_limits) self.robot = Robot(usb_device) @@ -45,6 +46,7 @@ def reset( seed: Optional[int] = None, options: Optional[dict] = None, ): + super().reset(seed=seed) logging.info("Reset env...") # wait for pendulum to fall back to start position diff --git a/furuta/rl/envs/furuta_sim.py b/furuta/rl/envs/furuta_sim.py index 304912b..d31477c 100644 --- a/furuta/rl/envs/furuta_sim.py +++ b/furuta/rl/envs/furuta_sim.py @@ -1,4 +1,4 @@ -from typing import Optional +from typing import List, Optional, Union import gymnasium as gym import numpy as np @@ -15,13 +15,14 @@ def __init__( dyn: QubeDynamics = QubeDynamics(), control_freq=50, reward="alpha", - state_limits=[np.pi, 2 * np.pi, 20, 20], - encoders_CPRs=None, + angle_limits=None, + speed_limits=None, + encoders_CPRs: Optional[List[int]] = None, velocity_filter: int = None, render_mode="rgb_array", ): - super().__init__(control_freq, reward, state_limits, render_mode) + super().__init__(control_freq, reward, angle_limits, speed_limits, render_mode) self.dyn = dyn self.encoders_CPRs = encoders_CPRs @@ -64,12 +65,12 @@ def _update_state(self, a): # do this by rounding _simulation_state[THETA/ALPHA] to the nearest multiple of 2pi/CPRs if self.encoders_CPRs: # TODO dedupe code here - theta_increment = 2 * np.pi / self.encoders_CPRs["motor_encoder_CPRs"] + theta_increment = 2 * np.pi / self.encoders_CPRs[THETA] self._state[THETA] = ( np.round(self._simulation_state[THETA] / theta_increment) * theta_increment ) - alpha_increment = 2 * np.pi / self.encoders_CPRs["pendulum_encoder_CPRs"] + alpha_increment = 2 * np.pi / self.encoders_CPRs[ALPHA] self._state[ALPHA] = ( np.round(self._simulation_state[ALPHA] / alpha_increment) * alpha_increment ) @@ -89,8 +90,7 @@ def reset( seed: Optional[int] = None, options: Optional[dict] = None, ): - if options is None: - options = {} + super().reset(seed=seed) self._init_state() obs = self.get_obs() return obs, {} From 9f95c336105021e099910567ecb9065dac8e4d4a Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 16:17:59 +0100 Subject: [PATCH 22/44] add env.reset to seed everything to comply with gymnasium --- furuta/rl/utils.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/furuta/rl/utils.py b/furuta/rl/utils.py index e8e6216..f9ae642 100644 --- a/furuta/rl/utils.py +++ b/furuta/rl/utils.py @@ -9,12 +9,14 @@ def seed_everything(env, seed, cudnn_deterministic): + # got this from cleanrl env.action_space.seed(seed) env.observation_space.seed(seed) random.seed(seed) np.random.seed(seed) torch.manual_seed(seed) torch.backends.cudnn.deterministic = cudnn_deterministic + env.reset(seed=seed) def download_artifact_file(artifact_alias, filename): From 235c61fd26544984ab5482ebdb5fc11a51fc2d49 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 16:19:40 +0100 Subject: [PATCH 23/44] clean up wrapper instantiation --- scripts/train.py | 50 +++++++++++------------------------------------- 1 file changed, 11 insertions(+), 39 deletions(-) diff --git a/scripts/train.py b/scripts/train.py index 4949e32..ae51745 100644 --- a/scripts/train.py +++ b/scripts/train.py @@ -1,15 +1,7 @@ import copy -import importlib import logging -import os -import random -from pathlib import Path -import gymnasium as gym import hydra -import numpy as np -import stable_baselines3 -import torch import wandb from omegaconf import DictConfig, OmegaConf, open_dict from stable_baselines3.common.callbacks import ( @@ -30,20 +22,6 @@ upload_file_to_artifacts, ) -# import furuta # noqa F420 - - -def instantiate_gym_wrapper(wrapper_name: str, wrapper_args: DictConfig, env: gym.Env) -> gym.Env: - """Instantiate a gym wrapper from a config.""" - module = importlib.import_module(wrapper_args.module) - wrapper_class = getattr(module, wrapper_name) - - # pop module from DictConfig wrappers args - with open_dict(wrapper_args): - wrapper_args.pop("module") - - return wrapper_class(env, **wrapper_args) - @hydra.main(version_base="1.3", config_path="configs", config_name="train.yaml") def main(cfg: DictConfig): @@ -69,15 +47,14 @@ def main(cfg: DictConfig): seed_everything(env, cfg.seed, cfg.cudnn_deterministic) # setup wrappers - for wrapper_name, wrapper_args in cfg.wrappers.items(): - try: - env = instantiate_gym_wrapper(wrapper_name, wrapper_args, env) - except Exception as e: - # print stack trace - logging.warning(e, exc_info=True) + for wrapper in cfg.wrappers: + env = hydra.utils.instantiate(wrapper, env=env) + # don't paralelize if it's the real robot if isinstance(env.unwrapped, FurutaReal): vec_env = DummyVecEnv([lambda: env]) + if cfg.n_envs > 1: + logging.warning("n_envs > 1 but using real robot, ignoring n_envs") else: vec_env = SubprocVecEnv([lambda: copy.deepcopy(env) for _ in range(cfg.n_envs)]) @@ -89,12 +66,10 @@ def main(cfg: DictConfig): # load model/replay buffer if cfg.model_artifact: - model.load(download_artifact_file(f"model:{cfg.model_artifact}", "model.zip")) + model.load(download_artifact_file(cfg.model_artifact, "model.zip")) if cfg.replay_buffer_artifact: - rb_path = download_artifact_file( - f"replay_buffer:{cfg.replay_buffer_artifact}", "buffer.pkl" - ) + rb_path = download_artifact_file(cfg.replay_buffer_artifact, "buffer.pkl") model.load_replay_buffer(rb_path) # Stop training when the model reaches the reward threshold @@ -129,7 +104,7 @@ def main(cfg: DictConfig): # only save last video if cfg.capture_video: - # TODO add headleas arg, depends on the machine + # TODO add headless arg, depends on the machine # import pyvirtualdisplay # pyvirtualdisplay.Display(visible=0, size=(1400, 900)).start() video_length = 500 @@ -141,9 +116,6 @@ def main(cfg: DictConfig): video_length=video_length, ) - # vec so we can't pass reset options - # TODO random init should be an env param? - # obs = env.reset(options={"random_init": False}) obs = env.reset() for _ in range(video_length + 1): action, _ = model.predict(obs, deterministic=False) @@ -155,15 +127,15 @@ def main(cfg: DictConfig): if cfg.save_model: logging.info("Saving model to artifacts") - model_path = f"runs/{run.id}/models/sac.zip" + model_path = f"runs/{run.id}/models/model.zip" model.save(model_path) - upload_file_to_artifacts(model_path, "sac_model", "model") + upload_file_to_artifacts(model_path, "model", "model") if cfg.save_replay_buffer: logging.info("Saving replay_buffer to artifacts") buffer_path = f"runs/{run.id}/buffers/buffer.pkl" model.save_replay_buffer(buffer_path) - upload_file_to_artifacts(buffer_path, "sac_replay_buffer", "replay buffer") + upload_file_to_artifacts(buffer_path, "replay_buffer", "replay buffer") env.close() run.finish() From 57c85c7658d56f9af71dcb825af801978d2a90ba Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 16:25:37 +0100 Subject: [PATCH 24/44] update config --- scripts/configs/env/sim.yaml | 3 ++- scripts/configs/experiment/sim.yaml | 2 +- scripts/configs/train.yaml | 2 +- scripts/configs/wrappers/base_wrappers.yaml | 12 +++--------- scripts/configs/wrappers/real_wrappers.yaml | 12 +++--------- 5 files changed, 10 insertions(+), 21 deletions(-) diff --git a/scripts/configs/env/sim.yaml b/scripts/configs/env/sim.yaml index 8267302..70c151e 100644 --- a/scripts/configs/env/sim.yaml +++ b/scripts/configs/env/sim.yaml @@ -5,7 +5,8 @@ defaults: _target_: furuta.rl.envs.furuta_sim.FurutaSim control_freq: 50 reward: "alpha" -state_limits: [3.14, 12, 50, 400] # TODO how to use PI in configs? +angle_limits: [6.3, 12] +speed_limits: [60, 400] encoders_CPRs: null velocity_filter: 2 render_mode: "rgb_array" diff --git a/scripts/configs/experiment/sim.yaml b/scripts/configs/experiment/sim.yaml index f2aa89d..5cc9093 100644 --- a/scripts/configs/experiment/sim.yaml +++ b/scripts/configs/experiment/sim.yaml @@ -4,4 +4,4 @@ defaults: - override /wrappers: base_wrappers gym_id: "FurutaSim-v0" -total_timesteps: 1_000_000 +total_timesteps: 150_000 diff --git a/scripts/configs/train.yaml b/scripts/configs/train.yaml index f61e2a5..7ea381b 100644 --- a/scripts/configs/train.yaml +++ b/scripts/configs/train.yaml @@ -32,5 +32,5 @@ replay_buffer_artifact: null model_artifact: null # logging -save_model: False +save_model: True save_replay_buffer: False diff --git a/scripts/configs/wrappers/base_wrappers.yaml b/scripts/configs/wrappers/base_wrappers.yaml index 571ba59..c8661d9 100644 --- a/scripts/configs/wrappers/base_wrappers.yaml +++ b/scripts/configs/wrappers/base_wrappers.yaml @@ -1,14 +1,8 @@ -# slightly unsure if wrappers will be applied in the order they are listed here -# looks like the order is preserved but unsure if there could be edge cases? -# the order of the wrappers is important -TimeLimit: - module: "gymnasium.wrappers" +- _target_: gymnasium.wrappers.TimeLimit max_episode_steps: 1000 -HistoryWrapper: - module: "furuta.rl.wrappers" +- _target_: furuta.rl.wrappers.HistoryWrapper steps: 2 use_continuity_cost: True -Monitor: - module: "stable_baselines3.common.monitor" +- _target_: stable_baselines3.common.monitor.Monitor diff --git a/scripts/configs/wrappers/real_wrappers.yaml b/scripts/configs/wrappers/real_wrappers.yaml index 3beb57f..4d04e82 100644 --- a/scripts/configs/wrappers/real_wrappers.yaml +++ b/scripts/configs/wrappers/real_wrappers.yaml @@ -2,13 +2,7 @@ defaults: - base_wrappers -GentlyTerminating: - module: "furuta.rl.wrappers" - -ControlFrequency: - module: "furuta.rl.wrappers" -# TODO update mcap logger to take terminated and truncated -# MCAPLogger: -# module: "furuta.rl.wrappers" +- _target_: furuta.rl.wrappers.GentlyTerminating +- _target_: furuta.rl.wrappers.ControlFrequency +# - _target_: furuta.rl.wrappers.MCAPLogger # use_sim_time: False -# log_dir: "mcap_logs/" From bb252734b2078f110eca04b9bdcfaca30e67a41e Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 16:26:15 +0100 Subject: [PATCH 25/44] make sure we can extend the wrappers config --- scripts/configs/wrappers/base_wrappers.yaml | 15 +++++++-------- scripts/configs/wrappers/real_wrappers.yaml | 7 ++++--- scripts/train.py | 2 +- 3 files changed, 12 insertions(+), 12 deletions(-) diff --git a/scripts/configs/wrappers/base_wrappers.yaml b/scripts/configs/wrappers/base_wrappers.yaml index c8661d9..96fdd0c 100644 --- a/scripts/configs/wrappers/base_wrappers.yaml +++ b/scripts/configs/wrappers/base_wrappers.yaml @@ -1,8 +1,7 @@ -- _target_: gymnasium.wrappers.TimeLimit - max_episode_steps: 1000 - -- _target_: furuta.rl.wrappers.HistoryWrapper - steps: 2 - use_continuity_cost: True - -- _target_: stable_baselines3.common.monitor.Monitor +wrappers: + - _target_: gymnasium.wrappers.TimeLimit + max_episode_steps: 1000 + - _target_: furuta.rl.wrappers.HistoryWrapper + steps: 2 + use_continuity_cost: True + - _target_: stable_baselines3.common.monitor.Monitor diff --git a/scripts/configs/wrappers/real_wrappers.yaml b/scripts/configs/wrappers/real_wrappers.yaml index 4d04e82..c448b73 100644 --- a/scripts/configs/wrappers/real_wrappers.yaml +++ b/scripts/configs/wrappers/real_wrappers.yaml @@ -1,8 +1,9 @@ # import common wrappers between sim/real defaults: - - base_wrappers + - base_wrappers@_here_ -- _target_: furuta.rl.wrappers.GentlyTerminating -- _target_: furuta.rl.wrappers.ControlFrequency +wrappers: + - _target_: furuta.rl.wrappers.GentlyTerminating + - _target_: furuta.rl.wrappers.ControlFrequency # - _target_: furuta.rl.wrappers.MCAPLogger # use_sim_time: False diff --git a/scripts/train.py b/scripts/train.py index ae51745..eb52f09 100644 --- a/scripts/train.py +++ b/scripts/train.py @@ -47,7 +47,7 @@ def main(cfg: DictConfig): seed_everything(env, cfg.seed, cfg.cudnn_deterministic) # setup wrappers - for wrapper in cfg.wrappers: + for wrapper in cfg.wrappers.wrappers: env = hydra.utils.instantiate(wrapper, env=env) # don't paralelize if it's the real robot From c3ea9cea6edf1b809dc50c8e6a0fe9c68a2929ca Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 16:34:23 +0100 Subject: [PATCH 26/44] seed is keyword arg not positional --- furuta/rl/envs/furuta_base.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index af7bd79..9bdba5d 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -137,7 +137,7 @@ def reset( seed: Optional[int] = None, options: Optional[dict] = None, ): - super().reset(seed) + super().reset(seed=seed) def _update_state(self, a): raise NotImplementedError From 920f3ab385e62a27968f9357ecd126df80074541 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 17:44:47 +0100 Subject: [PATCH 27/44] lower max obs based on simulated values --- furuta/rl/envs/furuta_base.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index 9bdba5d..ee91ddf 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -67,8 +67,10 @@ def __init__( self.state_max = np.array(state_limits, dtype=np.float32) # max obs based on max speeds measured on the robot - # TODO need to actually measure it - obs_max = np.array([1.0, 1.0, 1.0, 1.0, 50, 50], dtype=np.float32) + # in sim the speeds spike at 30 rad/s + # it's also ok if the speeds exceed theses values as we only use them for rescaling + # and it's okay if the nn sees values a little bit above 1 + obs_max = np.array([1.0, 1.0, 1.0, 1.0, 30, 30], dtype=np.float32) # Spaces self.state_space = Box( From 61c4e0be3521a7edbb3770c0a505b3b568b3dc23 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 17:45:21 +0100 Subject: [PATCH 28/44] only usb subproc vecenv when n_envs > 1 --- scripts/train.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/scripts/train.py b/scripts/train.py index eb52f09..c70398a 100644 --- a/scripts/train.py +++ b/scripts/train.py @@ -55,6 +55,8 @@ def main(cfg: DictConfig): vec_env = DummyVecEnv([lambda: env]) if cfg.n_envs > 1: logging.warning("n_envs > 1 but using real robot, ignoring n_envs") + elif cfg.n_envs == 1: + vec_env = DummyVecEnv([lambda: env]) else: vec_env = SubprocVecEnv([lambda: copy.deepcopy(env) for _ in range(cfg.n_envs)]) From fa53fdc74bc0efc2995c3534a34e85c5fd7b4c90 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 17:52:45 +0100 Subject: [PATCH 29/44] fix #25 --- furuta/logging/protobuf/pendulum_state.proto | 4 +-- furuta/logging/protobuf/pendulum_state_pb2.py | 21 ++++++------- furuta/rl/envs/furuta_base.py | 13 +------- furuta/rl/wrappers.py | 30 ++++++++++++------- 4 files changed, 33 insertions(+), 35 deletions(-) diff --git a/furuta/logging/protobuf/pendulum_state.proto b/furuta/logging/protobuf/pendulum_state.proto index 1ca63a5..69d18ce 100644 --- a/furuta/logging/protobuf/pendulum_state.proto +++ b/furuta/logging/protobuf/pendulum_state.proto @@ -6,7 +6,5 @@ message PendulumState { float motor_angle_velocity = 3; float pendulum_angle_velocity = 4; float reward = 5; - bool done = 6; - float action = 7; - float corrected_action = 8; + float action = 6; } diff --git a/furuta/logging/protobuf/pendulum_state_pb2.py b/furuta/logging/protobuf/pendulum_state_pb2.py index 3cdfa87..a1025db 100644 --- a/furuta/logging/protobuf/pendulum_state_pb2.py +++ b/furuta/logging/protobuf/pendulum_state_pb2.py @@ -1,25 +1,26 @@ +# -*- coding: utf-8 -*- # Generated by the protocol buffer compiler. DO NOT EDIT! # source: pendulum_state.proto """Generated protocol buffer code.""" +from google.protobuf.internal import builder as _builder from google.protobuf import descriptor as _descriptor from google.protobuf import descriptor_pool as _descriptor_pool from google.protobuf import symbol_database as _symbol_database -from google.protobuf.internal import builder as _builder - # @@protoc_insertion_point(imports) _sym_db = _symbol_database.Default() -DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile( - b'\n\x14pendulum_state.proto"\xc3\x01\n\rPendulumState\x12\x13\n\x0bmotor_angle\x18\x01 \x01(\x02\x12\x16\n\x0ependulum_angle\x18\x02 \x01(\x02\x12\x1c\n\x14motor_angle_velocity\x18\x03 \x01(\x02\x12\x1f\n\x17pendulum_angle_velocity\x18\x04 \x01(\x02\x12\x0e\n\x06reward\x18\x05 \x01(\x02\x12\x0c\n\x04\x64one\x18\x06 \x01(\x08\x12\x0e\n\x06\x61\x63tion\x18\x07 \x01(\x02\x12\x18\n\x10\x63orrected_action\x18\x08 \x01(\x02\x62\x06proto3' -) -_builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, globals()) -_builder.BuildTopDescriptorsAndMessages(DESCRIPTOR, "pendulum_state_pb2", globals()) + +DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x14pendulum_state.proto\"\x9b\x01\n\rPendulumState\x12\x13\n\x0bmotor_angle\x18\x01 \x01(\x02\x12\x16\n\x0ependulum_angle\x18\x02 \x01(\x02\x12\x1c\n\x14motor_angle_velocity\x18\x03 \x01(\x02\x12\x1f\n\x17pendulum_angle_velocity\x18\x04 \x01(\x02\x12\x0e\n\x06reward\x18\x05 \x01(\x02\x12\x0e\n\x06\x61\x63tion\x18\x06 \x01(\x02\x62\x06proto3') + +_globals = globals() +_builder.BuildMessageAndEnumDescriptors(DESCRIPTOR, _globals) +_builder.BuildTopDescriptorsAndMessages(DESCRIPTOR, 'pendulum_state_pb2', _globals) if _descriptor._USE_C_DESCRIPTORS == False: - DESCRIPTOR._options = None - _PENDULUMSTATE._serialized_start = 25 - _PENDULUMSTATE._serialized_end = 220 + DESCRIPTOR._options = None + _globals['_PENDULUMSTATE']._serialized_start=25 + _globals['_PENDULUMSTATE']._serialized_end=180 # @@protoc_insertion_point(module_scope) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index ee91ddf..555acf4 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -99,16 +99,6 @@ def step(self, action): rwd = self._reward_func(self._state) obs = self.get_obs() - # we use this info dict to log the state in mcap files - info = { - "motor_angle": float(self._state[THETA]), - "pendulum_angle": float(self._state[ALPHA]), - "motor_angle_velocity": float(self._state[THETA_DOT]), - "pendulum_angle_velocity": float(self._state[ALPHA_DOT]), - "reward": float(rwd), - "action": float(action[0]), - } - # then take action/step the sim self._update_state(action[0]) @@ -117,10 +107,9 @@ def step(self, action): # if terminated: # rwd -= self.reward.oob_penalty - info["terminated"] = bool(terminated) truncated = False - return obs, rwd, terminated, truncated, info + return obs, rwd, terminated, truncated, {} def get_obs(self): return np.float32( diff --git a/furuta/rl/wrappers.py b/furuta/rl/wrappers.py index 4fccd51..5326b23 100644 --- a/furuta/rl/wrappers.py +++ b/furuta/rl/wrappers.py @@ -4,12 +4,14 @@ from typing import Optional, Union import gymnasium as gym +import hydra import numpy as np import wandb from gymnasium.spaces import Box from mcap_protobuf.writer import Writer from furuta.logging.protobuf.pendulum_state_pb2 import PendulumState +from furuta.utils import ALPHA, ALPHA_DOT, THETA, THETA_DOT class GentlyTerminating(gym.Wrapper): @@ -31,12 +33,15 @@ def reset( class MCAPLogger(gym.Wrapper): - def __init__(self, env: gym.Env, log_dir: Union[str, Path], use_sim_time: bool): + def __init__(self, env: gym.Env, use_sim_time: bool, log_dir: Union[str, Path] = None): super().__init__(env) - if isinstance(log_dir, str): - log_dir = Path(log_dir) - self.log_dir = log_dir + if log_dir: + if isinstance(log_dir, str): + log_dir = Path(log_dir) + self.log_dir = log_dir + else: + self.log_dir = Path(hydra.core.hydra_config.HydraConfig.get().runtime.output_dir) self.use_sim_time = use_sim_time self.episodes = 0 @@ -50,14 +55,18 @@ def step(self, action): if self.use_sim_time: time_to_log = self.sim_time else: - time_to_log = time.time_ns() # TODO check that's the right clock - - # convert to milliseconds - time_to_log = round(time_to_log * 1e9) + time_to_log = time.time_ns() self.mcap_writer.write_message( topic="/pendulum_state", - message=PendulumState(**info), + message=PendulumState( + motor_angle=self.unwrapped._state[THETA], + pendulum_angle=self.unwrapped._state[ALPHA], + motor_angle_velocity=self.unwrapped._state[THETA_DOT], + pendulum_angle_velocity=self.unwrapped._state[ALPHA_DOT], + reward=reward, + action=float(action[0]), + ), log_time=time_to_log, publish_time=time_to_log, ) @@ -81,8 +90,9 @@ def reset( self.output_file = open(self.log_dir / fname, "wb") self.mcap_writer = Writer(self.output_file) - # TODO add metadata + # TODO add metadata? # date, control frequency, wandb run id, sim parameters, robot parameters, etc. + # or maybe we should log this to wandb? or maybe just use it for quick local debug? self.episodes += 1 From a5d02a5797111b0df49289fafdc7f7c26bd7f5f4 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 19:04:24 +0100 Subject: [PATCH 30/44] reset velocity filter at each reset --- furuta/rl/envs/furuta_real.py | 8 +++++++- furuta/rl/envs/furuta_sim.py | 9 +++++++-- 2 files changed, 14 insertions(+), 3 deletions(-) diff --git a/furuta/rl/envs/furuta_real.py b/furuta/rl/envs/furuta_real.py index 9344585..ea019c8 100644 --- a/furuta/rl/envs/furuta_real.py +++ b/furuta/rl/envs/furuta_real.py @@ -28,10 +28,13 @@ def __init__( self.robot = Robot(usb_device) - self.vel_filt = VelocityFilter(2, dt=self.timing.dt) + self._init_vel_filt() self._update_state(0.0) + def _init_vel_filt(self): + self.vel_filt = VelocityFilter(2, dt=self.timing.dt) + def _update_state(self, action): motor_angle, pendulum_angle = self.robot.step(action) @@ -70,6 +73,9 @@ def reset( logging.info("Reset done") self._update_state(0.0) + # else the first computed velocity will take into account previous episode + # and it'll be huge and wrong and will terminate the episode + self._init_vel_filt() return self.get_obs(), {} # TODO: override parent render function diff --git a/furuta/rl/envs/furuta_sim.py b/furuta/rl/envs/furuta_sim.py index d31477c..b74b3d2 100644 --- a/furuta/rl/envs/furuta_sim.py +++ b/furuta/rl/envs/furuta_sim.py @@ -27,8 +27,12 @@ def __init__( self.encoders_CPRs = encoders_CPRs - if velocity_filter: - self.vel_filt = VelocityFilter(velocity_filter, dt=self.timing.dt) + self.velocity_filter = velocity_filter + self._init_vel_filt() + + def _init_vel_filt(self): + if self.velocity_filter: + self.vel_filt = VelocityFilter(self.velocity_filter, dt=self.timing.dt) else: self.vel_filt = None @@ -93,6 +97,7 @@ def reset( super().reset(seed=seed) self._init_state() obs = self.get_obs() + self._init_vel_filt() return obs, {} From 2293209f9102c194ca11a1bdf2a81828c6c623dd Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 19:05:53 +0100 Subject: [PATCH 31/44] Revert "only usb subproc vecenv when n_envs > 1" This reverts commit 61c4e0be3521a7edbb3770c0a505b3b568b3dc23. --- scripts/train.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/scripts/train.py b/scripts/train.py index c70398a..eb52f09 100644 --- a/scripts/train.py +++ b/scripts/train.py @@ -55,8 +55,6 @@ def main(cfg: DictConfig): vec_env = DummyVecEnv([lambda: env]) if cfg.n_envs > 1: logging.warning("n_envs > 1 but using real robot, ignoring n_envs") - elif cfg.n_envs == 1: - vec_env = DummyVecEnv([lambda: env]) else: vec_env = SubprocVecEnv([lambda: copy.deepcopy(env) for _ in range(cfg.n_envs)]) From b2e113a5638acc0f38b998f3d39aafc59acb5aec Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 19:14:08 +0100 Subject: [PATCH 32/44] add simple env test, add it to ci --- .github/workflows/ci.yml | 90 ++++++++++++++++++++++++++++++++++++++ .github/workflows/lint.yml | 26 ----------- tests/test_gym_env.py | 13 ++++++ 3 files changed, 103 insertions(+), 26 deletions(-) create mode 100644 .github/workflows/ci.yml delete mode 100644 .github/workflows/lint.yml create mode 100644 tests/test_gym_env.py diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..a4addbe --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,90 @@ +name: ci + +on: + push: + branches: + - master + pull_request: + branches: + - master + +# https://jacobian.org/til/github-actions-poetry/ +jobs: + flake8-lint: + runs-on: ubuntu-latest + name: Lint + steps: + - name: Check out source repository + uses: actions/checkout@v2 + - name: Set up Python environment + uses: actions/setup-python@v1 + with: + python-version: "3.9" + - name: flake8 Lint + uses: py-actions/flake8@v1 + with: + ignore: "E203,E402,E501,F401,F841" + exclude: "logs/*,data/*,furuta/logging/protobuf/*" + + test: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + + # If you wanted to use multiple Python versions, you'd have specify a matrix in the job and + # reference the matrixe python version here. + - uses: actions/setup-python@v2 + with: + python-version: 3.9 + + # Cache the installation of Poetry itself, e.g. the next step. This prevents the workflow + # from installing Poetry every time, which can be slow. Note the use of the Poetry version + # number in the cache key, and the "-0" suffix: this allows you to invalidate the cache + # manually if/when you want to upgrade Poetry, or if something goes wrong. This could be + # mildly cleaner by using an environment variable, but I don't really care. + - name: cache poetry install + uses: actions/cache@v2 + with: + path: ~/.local + key: poetry-1.4.0-0 + + # Install Poetry. You could do this manually, or there are several actions that do this. + # `snok/install-poetry` seems to be minimal yet complete, and really just calls out to + # Poetry's default install script, which feels correct. I pin the Poetry version here + # because Poetry does occasionally change APIs between versions and I don't want my + # actions to break if it does. + # + # The key configuration value here is `virtualenvs-in-project: true`: this creates the + # venv as a `.venv` in your testing directory, which allows the next step to easily + # cache it. + - uses: snok/install-poetry@v1 + with: + version: 1.4.0 + virtualenvs-create: true + virtualenvs-in-project: true + + # Cache your dependencies (i.e. all the stuff in your `pyproject.toml`). Note the cache + # key: if you're using multiple Python versions, or multiple OSes, you'd need to include + # them in the cache key. I'm not, so it can be simple and just depend on the poetry.lock. + - name: cache deps + id: cache-deps + uses: actions/cache@v2 + with: + path: .venv + key: pydeps-${{ hashFiles('**/poetry.lock') }} + + # Install dependencies. `--no-root` means "install all dependencies but not the project + # itself", which is what you want to avoid caching _your_ code. The `if` statement + # ensures this only runs on a cache miss. + - run: poetry install --no-interaction --no-root + if: steps.cache-deps.outputs.cache-hit != 'true' + + # Now install _your_ project. This isn't necessary for many types of projects -- particularly + # things like Django apps don't need this. But it's a good idea since it fully-exercises the + # pyproject.toml and makes that if you add things like console-scripts at some point that + # they'll be installed and working. + - run: poetry install --no-interaction + + # And finally run tests. I'm using pytest and all my pytest config is in my `pyproject.toml` + # so this line is super-simple. But it could be as complex as you need. + - run: poetry run pytest diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml deleted file mode 100644 index d590039..0000000 --- a/.github/workflows/lint.yml +++ /dev/null @@ -1,26 +0,0 @@ -name: flake8 Lint - -on: - push: - branches: - - master - pull_request: - branches: - - master - -jobs: - flake8-lint: - runs-on: ubuntu-latest - name: Lint - steps: - - name: Check out source repository - uses: actions/checkout@v2 - - name: Set up Python environment - uses: actions/setup-python@v1 - with: - python-version: "3.9" - - name: flake8 Lint - uses: py-actions/flake8@v1 - with: - ignore: "E203,E402,E501,F401,F841" - exclude: "logs/*,data/*,furuta/logging/protobuf/*" diff --git a/tests/test_gym_env.py b/tests/test_gym_env.py new file mode 100644 index 0000000..2154e18 --- /dev/null +++ b/tests/test_gym_env.py @@ -0,0 +1,13 @@ +from stable_baselines3.common.env_checker import check_env + +from furuta.rl.envs.furuta_real import FurutaReal +from furuta.rl.envs.furuta_sim import FurutaSim + + +def test_sim_env(): + env = FurutaSim() + check_env(env) + + +# TODO how to check real env? +# would need to mock the robot From 3781e04e9e4e7b4d48e4eb89431ff391e86bb73a Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 19:15:14 +0100 Subject: [PATCH 33/44] tmp fix to pass CI --- furuta/controls/controllers.py | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/furuta/controls/controllers.py b/furuta/controls/controllers.py index ad3160c..f039f4c 100644 --- a/furuta/controls/controllers.py +++ b/furuta/controls/controllers.py @@ -11,11 +11,15 @@ def compute_command(self, position: float): @staticmethod def build_controller(parameters: dict): controller_type = parameters["controller_type"] - match controller_type: - case "PIDController": - return PIDController(parameters) - case _: - raise ValueError(f"Invalid controller type: {controller_type}") + # match controller_type: + # case "PIDController": + # return PIDController(parameters) + # case _: + # raise ValueError(f"Invalid controller type: {controller_type}") + if controller_type == "PIDController": + return PIDController(parameters) + else: + raise ValueError(f"Invalid controller type: {controller_type}") class PIDController(Controller): From 2cbc048b0d7347af552831b3f5c8cf5687fc18b9 Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 19:23:56 +0100 Subject: [PATCH 34/44] fix E721 --- furuta/rl/algos.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/furuta/rl/algos.py b/furuta/rl/algos.py index d06bebd..8f91ba0 100644 --- a/furuta/rl/algos.py +++ b/furuta/rl/algos.py @@ -10,7 +10,7 @@ class SAC(stable_baselines3.SAC): def __init__(self, **kwargs): # sb3 expects tuple, omegaconf returns list # so we need to convert kwarg train_freq from tuple to list - if "train_freq" in kwargs and type(kwargs["train_freq"]) == list: + if "train_freq" in kwargs and isinstance(kwargs["train_freq"], list): kwargs.update({"train_freq": tuple(kwargs["train_freq"])}) super().__init__(**kwargs) From f65e3dd0eb9757dc0d9f700cdd674513f723082d Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 22:42:46 +0100 Subject: [PATCH 35/44] update configs --- scripts/configs/algo/sac_real.yaml | 1 + scripts/configs/env/real.yaml | 3 ++- scripts/configs/env/sim.yaml | 2 +- scripts/configs/experiment/real_finetune.yaml | 2 +- scripts/configs/rl_inference.yaml | 8 ++++++++ scripts/configs/wrappers/base_wrappers.yaml | 13 ++++++------- scripts/configs/wrappers/real_wrappers.yaml | 15 ++++++++------- 7 files changed, 27 insertions(+), 17 deletions(-) create mode 100644 scripts/configs/rl_inference.yaml diff --git a/scripts/configs/algo/sac_real.yaml b/scripts/configs/algo/sac_real.yaml index 65f7bbb..2d3067b 100644 --- a/scripts/configs/algo/sac_real.yaml +++ b/scripts/configs/algo/sac_real.yaml @@ -1,5 +1,6 @@ defaults: - sac +learning_starts: 1 sde_sample_freq: 32 train_freq: [1, "episode"] diff --git a/scripts/configs/env/real.yaml b/scripts/configs/env/real.yaml index 4001f98..a35001b 100644 --- a/scripts/configs/env/real.yaml +++ b/scripts/configs/env/real.yaml @@ -1,5 +1,6 @@ _target_: furuta.rl.envs.furuta_real.FurutaReal control_freq: 50 reward: "alpha" -state_limits: [3.14, 12, 50, 400] # TODO how to use PI in configs? +angle_limits: [3.14, 6.3] +speed_limits: null # [60, 400] usb_device: "/dev/ttyACM0" diff --git a/scripts/configs/env/sim.yaml b/scripts/configs/env/sim.yaml index 70c151e..77f6cb1 100644 --- a/scripts/configs/env/sim.yaml +++ b/scripts/configs/env/sim.yaml @@ -5,7 +5,7 @@ defaults: _target_: furuta.rl.envs.furuta_sim.FurutaSim control_freq: 50 reward: "alpha" -angle_limits: [6.3, 12] +angle_limits: [3.14, 12] speed_limits: [60, 400] encoders_CPRs: null velocity_filter: 2 diff --git a/scripts/configs/experiment/real_finetune.yaml b/scripts/configs/experiment/real_finetune.yaml index f628956..36b356b 100644 --- a/scripts/configs/experiment/real_finetune.yaml +++ b/scripts/configs/experiment/real_finetune.yaml @@ -6,5 +6,5 @@ defaults: total_timesteps: 150_000 n_envs: 1 -model_artifact: "v358" +model_artifact: "model:v7" capture_video: False diff --git a/scripts/configs/rl_inference.yaml b/scripts/configs/rl_inference.yaml new file mode 100644 index 0000000..c8b4559 --- /dev/null +++ b/scripts/configs/rl_inference.yaml @@ -0,0 +1,8 @@ +defaults: + - _self_ + - env: real + - wrappers: real_wrappers + +model_artifact: "model:latest" +render: False +nb_episodes: 1 diff --git a/scripts/configs/wrappers/base_wrappers.yaml b/scripts/configs/wrappers/base_wrappers.yaml index 96fdd0c..747d90d 100644 --- a/scripts/configs/wrappers/base_wrappers.yaml +++ b/scripts/configs/wrappers/base_wrappers.yaml @@ -1,7 +1,6 @@ -wrappers: - - _target_: gymnasium.wrappers.TimeLimit - max_episode_steps: 1000 - - _target_: furuta.rl.wrappers.HistoryWrapper - steps: 2 - use_continuity_cost: True - - _target_: stable_baselines3.common.monitor.Monitor +- _target_: gymnasium.wrappers.TimeLimit + max_episode_steps: 1000 +- _target_: furuta.rl.wrappers.HistoryWrapper + steps: 2 + use_continuity_cost: True +- _target_: stable_baselines3.common.monitor.Monitor diff --git a/scripts/configs/wrappers/real_wrappers.yaml b/scripts/configs/wrappers/real_wrappers.yaml index c448b73..8f23307 100644 --- a/scripts/configs/wrappers/real_wrappers.yaml +++ b/scripts/configs/wrappers/real_wrappers.yaml @@ -1,9 +1,10 @@ -# import common wrappers between sim/real -defaults: - - base_wrappers@_here_ - -wrappers: - - _target_: furuta.rl.wrappers.GentlyTerminating - - _target_: furuta.rl.wrappers.ControlFrequency +- _target_: gymnasium.wrappers.TimeLimit + max_episode_steps: 1000 +- _target_: furuta.rl.wrappers.HistoryWrapper + steps: 2 + use_continuity_cost: True +- _target_: stable_baselines3.common.monitor.Monitor +- _target_: furuta.rl.wrappers.GentlyTerminating +- _target_: furuta.rl.wrappers.ControlFrequency # - _target_: furuta.rl.wrappers.MCAPLogger # use_sim_time: False From cc6bbb3eb347d18f96688f68cdf99065bdcfc22b Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 22:43:18 +0100 Subject: [PATCH 36/44] change the way we config wrappers --- scripts/train.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/scripts/train.py b/scripts/train.py index eb52f09..620ac76 100644 --- a/scripts/train.py +++ b/scripts/train.py @@ -2,7 +2,6 @@ import logging import hydra -import wandb from omegaconf import DictConfig, OmegaConf, open_dict from stable_baselines3.common.callbacks import ( EvalCallback, @@ -15,6 +14,7 @@ VecVideoRecorder, ) +import wandb from furuta.rl.envs.furuta_real import FurutaReal from furuta.rl.utils import ( download_artifact_file, @@ -47,7 +47,7 @@ def main(cfg: DictConfig): seed_everything(env, cfg.seed, cfg.cudnn_deterministic) # setup wrappers - for wrapper in cfg.wrappers.wrappers: + for wrapper in cfg.wrappers: env = hydra.utils.instantiate(wrapper, env=env) # don't paralelize if it's the real robot From ffe6e9aa0abf86e7caf35ea5ab1e1abbd043960d Mon Sep 17 00:00:00 2001 From: Armand Date: Thu, 18 Jan 2024 22:54:32 +0100 Subject: [PATCH 37/44] setup sweep on safety limits --- scripts/configs/sweeps/bounds.yaml | 16 ++++++++++++++++ scripts/configs/sweeps/encoders_simulation.yaml | 6 ++---- scripts/configs/train.yaml | 4 ++-- 3 files changed, 20 insertions(+), 6 deletions(-) create mode 100644 scripts/configs/sweeps/bounds.yaml diff --git a/scripts/configs/sweeps/bounds.yaml b/scripts/configs/sweeps/bounds.yaml new file mode 100644 index 0000000..fc6670d --- /dev/null +++ b/scripts/configs/sweeps/bounds.yaml @@ -0,0 +1,16 @@ +program: train.py +method: grid +parameters: + env.angle_limits: + values: [null, [3.14, 6.3], [3.14, 400], [400, 6.3], [12, 6.3]] + env.speed_limits: + values: [null, [30, 400], [60, 400], [120, 400]] + seed: + values: [1, 2, 3] + +command: + - ${env} + - python + - ${program} + - ${args_no_hyphens} + - +experiment=sim diff --git a/scripts/configs/sweeps/encoders_simulation.yaml b/scripts/configs/sweeps/encoders_simulation.yaml index a6427e6..a71d70a 100644 --- a/scripts/configs/sweeps/encoders_simulation.yaml +++ b/scripts/configs/sweeps/encoders_simulation.yaml @@ -1,10 +1,8 @@ program: train.py method: grid parameters: - env: - values: ["sim", "sim_encoders", "sim_precise_encoders"] - env.velocity_filter: - values: [null, 2, 3, 4, 10] + env.encoders_CPRs: + values: [[100, 4000.0], [211.2, 8192.0], [400.0, 16000.0]] seed: values: [1, 2, 3, 4] diff --git a/scripts/configs/train.yaml b/scripts/configs/train.yaml index 7ea381b..d28df52 100644 --- a/scripts/configs/train.yaml +++ b/scripts/configs/train.yaml @@ -7,7 +7,7 @@ defaults: - algo: sac.yaml total_timesteps: 500_000 -n_envs: 4 +n_envs: 12 # eval & early stopping evaluation: @@ -25,7 +25,7 @@ debug: True capture_video: True wandb: entity: null - project: furuta + project: furuta2 # resume training: load replay buffer and/or model replay_buffer_artifact: null From 013468dbf55f5e01534ff87f50dff490ee65bc27 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 21 Jan 2024 12:43:42 +0100 Subject: [PATCH 38/44] remove unecessary try catch --- furuta/rl/envs/furuta_base.py | 8 +------- 1 file changed, 1 insertion(+), 7 deletions(-) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index 555acf4..e783859 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -8,13 +8,7 @@ def alpha_reward(state): - try: - rwd = (1 + -np.cos(state[ALPHA])) / 2 - except Exception as e: # TODO why would this fail? - print(e) - print(state) - - return rwd + return (1 + -np.cos(state[ALPHA])) / 2 def alpha_theta_reward(state): From 39fec9ccb992bf7161910647d8def6ccc49d2b25 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 21 Jan 2024 12:44:19 +0100 Subject: [PATCH 39/44] remove unecessary line --- furuta/rl/envs/furuta_base.py | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index e783859..8518aac 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -55,10 +55,9 @@ def __init__( if speed_limits is None: speed_limits = [np.inf, np.inf] - state_limits = np.array( - [angle_limits[0], angle_limits[1], speed_limits[0], speed_limits[1]] + self.state_max = np.array( + [angle_limits[0], angle_limits[1], speed_limits[0], speed_limits[1]], dtype=np.float32 ) - self.state_max = np.array(state_limits, dtype=np.float32) # max obs based on max speeds measured on the robot # in sim the speeds spike at 30 rad/s From b30f1f873c93444b854cc7b7199aebecfbfee89e Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 21 Jan 2024 12:45:39 +0100 Subject: [PATCH 40/44] describe obs, set the bounds slightly higer --- furuta/rl/envs/furuta_base.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index 8518aac..fda806a 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -60,10 +60,12 @@ def __init__( ) # max obs based on max speeds measured on the robot - # in sim the speeds spike at 30 rad/s + # in sim the speeds spike at 30 rad/s when trained + # selected 50 rad/s to be safe bc its probably higher during training # it's also ok if the speeds exceed theses values as we only use them for rescaling # and it's okay if the nn sees values a little bit above 1 - obs_max = np.array([1.0, 1.0, 1.0, 1.0, 30, 30], dtype=np.float32) + # obs is [cos(th), sin(th), cos(al), sin(al), th_d, al_d)] + obs_max = np.array([1.0, 1.0, 1.0, 1.0, 50, 50], dtype=np.float32) # Spaces self.state_space = Box( From 4d02dc2a085928b6f4bd885f63f2543e720990c5 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 21 Jan 2024 12:48:29 +0100 Subject: [PATCH 41/44] add link to where render was copy pasted from --- furuta/rl/envs/furuta_base.py | 1 + 1 file changed, 1 insertion(+) diff --git a/furuta/rl/envs/furuta_base.py b/furuta/rl/envs/furuta_base.py index fda806a..db1d4af 100644 --- a/furuta/rl/envs/furuta_base.py +++ b/furuta/rl/envs/furuta_base.py @@ -129,6 +129,7 @@ def _update_state(self, a): raise NotImplementedError def render(self): + # https://github.com/Farama-Foundation/Gymnasium/blob/6baf8708bfb08e37ce3027b529193169eaa230fd/gymnasium/envs/classic_control/cartpole.py#L229 if self.render_mode is None: assert self.spec is not None gym.logger.warn( From 360b6a5270765eb63e378ca26f14622707d69b1c Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 21 Jan 2024 12:52:04 +0100 Subject: [PATCH 42/44] pygame should work headless without a virtual display --- scripts/train.py | 3 --- 1 file changed, 3 deletions(-) diff --git a/scripts/train.py b/scripts/train.py index 620ac76..5e97ce0 100644 --- a/scripts/train.py +++ b/scripts/train.py @@ -104,9 +104,6 @@ def main(cfg: DictConfig): # only save last video if cfg.capture_video: - # TODO add headless arg, depends on the machine - # import pyvirtualdisplay - # pyvirtualdisplay.Display(visible=0, size=(1400, 900)).start() video_length = 500 env = DummyVecEnv([lambda: env]) env = VecVideoRecorder( From 525be80618c87927e19edb2e805428a542cd9e9d Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 21 Jan 2024 12:52:38 +0100 Subject: [PATCH 43/44] CPRs are actually floats --- furuta/rl/envs/furuta_sim.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/furuta/rl/envs/furuta_sim.py b/furuta/rl/envs/furuta_sim.py index b74b3d2..01d0231 100644 --- a/furuta/rl/envs/furuta_sim.py +++ b/furuta/rl/envs/furuta_sim.py @@ -17,7 +17,7 @@ def __init__( reward="alpha", angle_limits=None, speed_limits=None, - encoders_CPRs: Optional[List[int]] = None, + encoders_CPRs: Optional[List[float]] = None, velocity_filter: int = None, render_mode="rgb_array", ): From d8a07de5e20593f6f3443495017777b622b3bfb2 Mon Sep 17 00:00:00 2001 From: Armand Date: Sun, 21 Jan 2024 12:54:11 +0100 Subject: [PATCH 44/44] add back comments describing the sim params --- furuta/robot.py | 22 ++++++++++++---------- 1 file changed, 12 insertions(+), 10 deletions(-) diff --git a/furuta/robot.py b/furuta/robot.py index e17fad8..b63c3db 100644 --- a/furuta/robot.py +++ b/furuta/robot.py @@ -37,15 +37,18 @@ class QubeDynamics: def __init__( self, g=9.81, - Rm=8.4, - V=12.0, - km=0.042, - Mr=0.095, - Lr=0.085, - Dr=5e-6, - Mp=0.024, - Lp=0.129, - Dp=1e-6, + # Motor + Rm=8.4, # reskstance (rated voltage/stall current) + V=12.0, # nominal voltage + km=0.042, # back-emf constant (V-s/rad) = (rated voltage / no load speed) + # Rotary Arm + Mr=0.095, # mass (kg) + Lr=0.085, # length (m) + Dr=5e-6, # viscous damping (N-m-s/rad), original: 0.0015 + # Pendulum Link + Mp=0.024, # mass (kg) + Lp=0.129, # length (m) + Dp=1e-6, # viscous damping (N-m-s/rad), original: 0.0005 ): # Gravity self.g = g @@ -54,7 +57,6 @@ def __init__( self.Rm = Rm self.V = V - # back-emf constant (V-s/rad) self.km = km # Rotary arm