Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature/78 ruff docstring formatting #109

Merged
merged 1 commit into from
Dec 28, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .ruff.toml
Original file line number Diff line number Diff line change
@@ -1 +1 @@
ignore-init-module-imports = true
ignore-init-module-imports = true
63 changes: 32 additions & 31 deletions pyproject.toml
Original file line number Diff line number Diff line change
@@ -1,36 +1,37 @@
[build-system]
requires = ["setuptools", "setuptools-scm"]
build-backend = "setuptools.build_meta"
requires = ["setuptools", "setuptools-scm"]
build-backend = "setuptools.build_meta"

[project]
name = "bsk_rl"
version = "0.0.0"
authors = [
{name = "Adam Herrmann", email = "adam.herrmann@colorado.edu"},
{name = "Mark Stephenson", email = "mark.a.stephenson@colorado.edu"},
]
description = "RL environments and tools for spacecraft autonomy research, built on Basilisk. Developed by the AVS Lab."
readme = "README.md"
requires-python = ">=3.9.0"
license = {text = "MIT"}
dependencies = [
"deap==1.3.3",
"Deprecated",
"gymnasium",
"matplotlib",
"numpy",
"pandas",
"pettingzoo",
"pytest",
"pytest-cov",
"pytest-repeat",
"requests",
"scikit-learn",
"scipy",
"stable-baselines3",
"tensorflow",
"torch",
]
name = "bsk_rl"
version = "0.0.0"
authors = [
{ name = "Adam Herrmann", email = "adam.herrmann@colorado.edu" },
{ name = "Mark Stephenson", email = "mark.a.stephenson@colorado.edu" },
]
description = "RL environments and tools for spacecraft autonomy research, built on Basilisk. Developed by the AVS Lab."
readme = "README.md"
requires-python = ">=3.9.0"
license = { text = "MIT" }
dependencies = [
"deap==1.3.3",
"Deprecated",
"gymnasium",
"matplotlib",
"numpy",
"pandas",
"pettingzoo",
"pytest",
"pytest-cov",
"pytest-repeat",
"requests",
"ruff>=0.1.9",
"scikit-learn",
"scipy",
"stable-baselines3",
"tensorflow",
"torch",
]

[project.scripts]
finish_install = "bsk_rl.finish_install:pck_install"
finish_install = "bsk_rl.finish_install:pck_install"
11 changes: 11 additions & 0 deletions src/bsk_rl/envs/general_satellite_tasking/.ruff.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
extend-safe-fixes = ["D"]

[lint.pydocstyle]
convention = "google"

[lint]
select = ["D", "W", "D401", "D404"]
ignore = ["D104"]

[lint.pycodestyle]
max-doc-length = 88
108 changes: 60 additions & 48 deletions src/bsk_rl/envs/general_satellite_tasking/gym_env.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
"""General Satellite Tasking is a framework for satellite tasking RL environments."""

import functools
import logging
import os
Expand Down Expand Up @@ -30,6 +32,28 @@


class GeneralSatelliteTasking(Env, Generic[SatObs, SatAct]):
"""A Gymnasium environment adaptable to a wide range satellite tasking problems.

These problems involve satellite(s) being tasked to complete tasks and maintain
aliveness. These tasks often include rewards for data collection. The environment
can be configured for any collection of satellites, including heterogenous
constellations. Other configurable aspects are environment features (e.g.
imaging targets), data collection and recording, and intersatellite
communication of data.

The state space is a tuple containing the state of each satellite. Actions are
assigned as a tuple of actions, one per satellite.

The preferred method of instantiating this environment is to make the
"GeneralSatelliteTasking-v1" environment and pass a kwargs dict with the
environment configuration. In some cases (e.g. the multiprocessed Gymnasium
vector environment), it is necessary for compatibility to instead register a new
environment using the GeneralSatelliteTasking class and a kwargs dict. See
examples/general_satellite_tasking for examples of environment configuration.

New environments should be built using this framework.
"""

def __init__(
self,
satellites: Union[Satellite, list[Satellite]],
Expand All @@ -47,25 +71,7 @@ def __init__(
log_dir: Optional[str] = None,
render_mode=None,
) -> None:
"""A Gymnasium environment adaptable to a wide range satellite tasking problems
that involve satellite(s) being tasked to complete tasks and maintain aliveness.
These tasks often include rewards for data collection. The environment can be
configured for any collection of satellites, including heterogenous
constellations. Other configurable aspects are environment features (e.g.
imaging targets), data collection and recording, and intersatellite
communication of data.

The state space is a tuple containing the state of each satellite. Actions are
assigned as a tuple of actions, one per satellite.

The preferred method of instantiating this environment is to make the
"GeneralSatelliteTasking-v1" environment and pass a kwargs dict with the
environment configuration. In some cases (e.g. the multiprocessed Gymnasium
vector environment), it is necessary for compatibility to instead register a new
environment using the GeneralSatelliteTasking class and a kwargs dict. See
examples/general_satellite_tasking for examples of environment configuration.

New environments should be built using this framework.
"""Construct the GeneralSatelliteTasking environment.

Args:
satellites: Satellites(s) to be simulated.
Expand Down Expand Up @@ -206,10 +212,11 @@ def reset(
return observation, info

def delete_simulator(self):
"""Delete Basilisk objects. Only self.simulator contains strong references to
BSK models, so deleting it will delete all Basilisk objects. Enable debug-level
logging to verify that the simulator, FSW, dynamics, and environment models are
all deleted on reset.
"""Delete Basilisk objects.

Only self.simulator contains strong references to BSK models, so deleting it
will delete all Basilisk objects. Enable debug-level logging to verify that the
simulator, FSW, dynamics, and environment models are all deleted on reset.
"""
try:
del self.simulator
Expand Down Expand Up @@ -264,7 +271,7 @@ def _get_truncated(self) -> bool:

@property
def action_space(self) -> spaces.Space[MultiSatAct]:
"""Compose satellite action spaces
"""Compose satellite action spaces.

Returns:
Joint action space
Expand All @@ -273,8 +280,9 @@ def action_space(self) -> spaces.Space[MultiSatAct]:

@property
def observation_space(self) -> spaces.Space[MultiSatObs]:
"""Compose satellite observation spaces. Note: calls reset(), which can be
expensive, to determine observation size.
"""Compose satellite observation spaces.

Note: calls reset(), which can be expensive, to determine observation size.

Returns:
Joint observation space
Expand Down Expand Up @@ -319,10 +327,10 @@ def _step(self, actions: MultiSatAct) -> None:
def step(
self, actions: MultiSatAct
) -> tuple[MultiSatObs, float, bool, bool, dict[str, Any]]:
"""Propagate the simulation, update information, and get rewards
"""Propagate the simulation, update information, and get rewards.

Args:
Joint action for satellites
actions: Joint action for satellites

Returns:
observation, reward, terminated, truncated, info
Expand All @@ -343,22 +351,24 @@ def step(
return observation, reward, terminated, truncated, info

def render(self) -> None: # pragma: no cover
"""No rendering implemented"""
"""No rendering implemented."""
return None

def close(self) -> None:
"""Try to cleanly delete everything"""
"""Try to cleanly delete everything."""
if self.simulator is not None:
del self.simulator


class SingleSatelliteTasking(GeneralSatelliteTasking, Generic[SatObs, SatAct]):
"""A special case of the GeneralSatelliteTasking for one satellite. For
compatibility with standard training APIs, actions and observations are directly
"""A special case of the GeneralSatelliteTasking for one satellite.

For compatibility with standard training APIs, actions and observations are directly
exposed for the single satellite and are not wrapped in a tuple.
"""

def __init__(self, *args, **kwargs) -> None:
"""Construct the SingleSatelliteTasking environment."""
super().__init__(*args, **kwargs)
if not len(self.satellites) == 1:
raise ValueError(
Expand All @@ -367,21 +377,22 @@ def __init__(self, *args, **kwargs) -> None:

@property
def action_space(self) -> spaces.Space[SatAct]:
"""Return the single satellite action space"""
"""Return the single satellite action space."""
return self.satellite.action_space

@property
def observation_space(self) -> spaces.Box:
"""Return the single satellite observation space"""
"""Return the single satellite observation space."""
super().observation_space
return self.satellite.observation_space

@property
def satellite(self) -> Satellite:
"""Satellite being tasked."""
return self.satellites[0]

def step(self, action) -> tuple[Any, float, bool, bool, dict[str, Any]]:
"""Task the satellite with a single action"""
"""Task the satellite with a single action."""
return super().step([action])

def _get_obs(self) -> Any:
Expand All @@ -396,12 +407,13 @@ class MultiagentSatelliteTasking(
def reset(
self, seed: int | None = None, options=None
) -> tuple[MultiSatObs, dict[str, Any]]:
"""Reset the environment and return PettingZoo Parallel API format."""
self.newly_dead = []
return super().reset(seed, options)

@property
def agents(self) -> list[AgentID]:
"""Agents currently in the environment"""
"""Agents currently in the environment."""
truncated = super()._get_truncated()
return [
satellite.id
Expand All @@ -411,7 +423,7 @@ def agents(self) -> list[AgentID]:

@property
def num_agents(self) -> int:
"""Number of agents currently in the environment"""
"""Number of agents currently in the environment."""
return len(self.agents)

@property
Expand All @@ -421,7 +433,7 @@ def possible_agents(self) -> list[AgentID]:

@property
def max_num_agents(self) -> int:
"""Maximum number of agents possible in the environment"""
"""Maximum number of agents possible in the environment."""
return len(self.possible_agents)

@property
Expand All @@ -431,40 +443,40 @@ def previously_dead(self) -> list[AgentID]:

@property
def observation_spaces(self) -> dict[AgentID, spaces.Box]:
"""Return the observation space for each agent"""
"""Return the observation space for each agent."""
return {
agent: obs_space
for agent, obs_space in zip(self.possible_agents, super().observation_space)
}

@functools.lru_cache(maxsize=None)
def observation_space(self, agent: AgentID) -> spaces.Space[SatObs]:
"""Return the observation space for a certain agent"""
"""Return the observation space for a certain agent."""
return self.observation_spaces[agent]

@property
def action_spaces(self) -> dict[AgentID, spaces.Space[SatAct]]:
"""Return the action space for each agent"""
"""Return the action space for each agent."""
return {
agent: act_space
for agent, act_space in zip(self.possible_agents, super().action_space)
}

@functools.lru_cache(maxsize=None)
def action_space(self, agent: AgentID) -> spaces.Space[SatAct]:
"""Return the action space for a certain agent"""
"""Return the action space for a certain agent."""
return self.action_spaces[agent]

def _get_obs(self) -> dict[AgentID, SatObs]:
"""Format the observation per the PettingZoo Parallel API"""
"""Format the observation per the PettingZoo Parallel API."""
return {
agent: satellite.get_obs()
for agent, satellite in zip(self.possible_agents, self.satellites)
if agent not in self.previously_dead
}

def _get_reward(self) -> dict[AgentID, float]:
"""Format the reward per the PettingZoo Parallel API"""
"""Format the reward per the PettingZoo Parallel API."""
reward = deepcopy(self.reward_dict)
for agent, satellite in zip(self.possible_agents, self.satellites):
if not satellite.is_alive():
Expand All @@ -478,7 +490,7 @@ def _get_reward(self) -> dict[AgentID, float]:
return reward

def _get_terminated(self) -> dict[AgentID, bool]:
"""Format terminations per the PettingZoo Parallel API"""
"""Format terminations per the PettingZoo Parallel API."""
if self.terminate_on_time_limit and super()._get_truncated():
return {
agent: True
Expand All @@ -493,7 +505,7 @@ def _get_terminated(self) -> dict[AgentID, bool]:
}

def _get_truncated(self) -> dict[AgentID, bool]:
"""Format truncations per the PettingZoo Parallel API"""
"""Format truncations per the PettingZoo Parallel API."""
truncated = super()._get_truncated()
return {
agent: truncated
Expand All @@ -502,7 +514,7 @@ def _get_truncated(self) -> dict[AgentID, bool]:
}

def _get_info(self) -> dict[AgentID, dict]:
"""Format info per the PettingZoo Parallel API"""
"""Format info per the PettingZoo Parallel API."""
info = super()._get_info()
for agent in self.possible_agents:
if agent in self.previously_dead:
Expand All @@ -519,7 +531,7 @@ def step(
dict[AgentID, bool],
dict[AgentID, dict],
]:
"""Step the environment and return PettingZoo Parallel API format"""
"""Step the environment and return PettingZoo Parallel API format."""
logger.info("=== STARTING STEP ===")

previous_alive = self.agents
Expand Down
Loading