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

Added the Active Disturbance Rejection Control (ADRC) Algorithm #12648

Open
wants to merge 11 commits into
base: master
Choose a base branch
from
Empty file added control_algorithms/__init__.py
Empty file.
62 changes: 62 additions & 0 deletions control_algorithms/adrc.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
"""
Active Disturbance Rejection Control (ADRC) is a robust control strategy
that estimates and compensates for disturbances in real-time without needing
an explicit mathematical model of the system.

It consists of:
1. Tracking Differentiator (TD) - Smooths the reference signal
2. Extended State Observer (ESO) - Estimates system states and disturbances
3. Nonlinear State Error Feedback (NLSEF) - Generates the control signal

Refer - https://en.wikipedia.org/wiki/Active_disturbance_rejection_control
"""


class ADRC:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Optional: This might be simpler as a dataclass. https://docs.python.org/3/library/dataclasses.html

def __init__(self, error_correction: float, disturbance: float, acceleration: float, target: float = 0.0):

Check failure on line 16 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (ARG002)

control_algorithms/adrc.py:16:90: ARG002 Unused method argument: `target`

Check failure on line 16 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (E501)

control_algorithms/adrc.py:16:89: E501 Line too long (110 > 88)

Check failure on line 16 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (ARG002)

control_algorithms/adrc.py:16:69: ARG002 Unused method argument: `acceleration`

Check failure on line 16 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (ARG002)

control_algorithms/adrc.py:16:49: ARG002 Unused method argument: `disturbance`

Check failure on line 16 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (ARG002)

control_algorithms/adrc.py:16:24: ARG002 Unused method argument: `error_correction`

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please provide return type hint for the function: __init__. If the function does not return a value, please provide the type hint as: def function() -> None:

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please provide return type hint for the function: __init__. If the function does not return a value, please provide the type hint as: def function() -> None:

"""
Initialize the ADRC controller.

:param beta1: Gain for error correction in ESO
:param beta2: Gain for disturbance estimation in ESO
:param beta3: Gain for acceleration estimation in ESO
:param setpoint: Desired target value
"""
self.beta1 = beta1

Check failure on line 25 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (F821)

control_algorithms/adrc.py:25:22: F821 Undefined name `beta1`
self.beta2 = beta2

Check failure on line 26 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (F821)

control_algorithms/adrc.py:26:22: F821 Undefined name `beta2`
self.beta3 = beta3

Check failure on line 27 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (F821)

control_algorithms/adrc.py:27:22: F821 Undefined name `beta3`
self.setpoint = setpoint

Check failure on line 28 in control_algorithms/adrc.py

View workflow job for this annotation

GitHub Actions / ruff

Ruff (F821)

control_algorithms/adrc.py:28:25: F821 Undefined name `setpoint`

self.system_output = 0.0 # Estimated system output
self.system_velocity = 0.0 # Estimated system velocity
self.total_disturbance = 0.0 # Estimated total disturbance

def compute(self, measured_value: float, dt: float) -> float:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
def compute(self, measured_value: float, dt: float) -> float:
def control_output(self, measured_value: float, dt: float) -> float:

or get_control_output() or calculate_control_output()

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As there is no test file in this pull request nor any test function or class in the file control_algorithms/adrc.py, please provide doctest for the function compute

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As there is no test file in this pull request nor any test function or class in the file control_algorithms/adrc.py, please provide doctest for the function compute

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As there is no test file in this pull request nor any test function or class in the file control_algorithms/adrc.py, please provide doctest for the function compute

"""
Compute the control signal based on error estimation and disturbance rejection.

:param measured_value: The current process variable
:param dt: Time difference since the last update
:return: Control output
"""

# Extended State Observer (ESO) Update
self.z1 += dt * (self.z2 - self.beta1 * (self.z1 - measured_value))
self.z2 += dt * (self.z3 - self.beta2 * (self.z1 - measured_value))
self.z3 -= self.beta3 * (self.z1 - measured_value)

# Control Law (Nonlinear State Error Feedback - NLSEF)
control_output = self.z2 - self.z3
return control_output

def reset(self):

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As there is no test file in this pull request nor any test function or class in the file control_algorithms/adrc.py, please provide doctest for the function reset

Please provide return type hint for the function: reset. If the function does not return a value, please provide the type hint as: def function() -> None:

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As there is no test file in this pull request nor any test function or class in the file control_algorithms/adrc.py, please provide doctest for the function reset

Please provide return type hint for the function: reset. If the function does not return a value, please provide the type hint as: def function() -> None:

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As there is no test file in this pull request nor any test function or class in the file control_algorithms/adrc.py, please provide doctest for the function reset

Please provide return type hint for the function: reset. If the function does not return a value, please provide the type hint as: def function() -> None:

"""Reset the estimated states."""
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As discussed in CONTRIBUTING.md, all methods need one or more doctests.

Suggested change
"""Reset the estimated states."""
"""Reset the estimated states.
>>> adrc = ADRC(0.0, 0.0, 0.0)
>>> adrc.z1, adrc.z2, adrc.z3 = 1.1, 2.2, 3.3
>>> adrc.reset()
>>> adrc.z1, adrc.z2, adrc.z3
(0.0, 0.0, 0.0)
"""

self.z1 = 0.0
self.z2 = 0.0
self.z3 = 0.0


if __name__ == "__main__":
import doctest

doctest.testmod()
Loading