Pyectool provides Python bindings for interacting with the Embedded Controller (EC) on ChromeOS and Framework devices.
It is extracted from and based on Dustin Howett's ectool
and exposes EC control functions directly to Python via a native C++ extension built with pybind11
.
Pyectool also provides a simple way to build the original ectool
CLI tool, or to build libectool
—a standalone C library that wrap most of ectool’s functionality, making it reusable in C/C++ projects or accessible from other languages. Both the CLI binary and the library are built automatically during installation.
- Python-native interface to low-level EC functionality via
pybind11
- Supports fan duty control, temperature reading, AC power status, and more.
- Designed for hardware monitoring, thermal management, and fan control tooling.
- Bundles the native
ectool
CLI andlibectool
C library alongside the Python package:pyectool/bin/ectool
(ectool CLI)pyectool/lib/libectool.a
(libectool static library)pyectool/include/libectool.h
(libectool C header)
Install system dependencies:
sudo apt update
sudo apt install -y libusb-1.0-0-dev libftdi1-dev pkg-config
Create and activate a virtual environment:
python3 -m venv ~/.venv/pyectool
source ~/.venv/pyectool/bin/activate
Install from PyPI:
pip install pyectool
Test installation (requires sudo
to access the EC):
sudo env "PATH=$PATH" python -c "from pyectool import ECController; ec = ECController(); print(ec.is_on_ac())"
This will print True
if the system is on AC power, or False
if on battery.
Clone the repository:
git clone https://github.com/CCExtractor/libectool
cd libectool
Create and activate a virtual environment:
python3 -m venv ~/.venv/pyectool
source ~/.venv/pyectool/bin/activate
Install the package:
sudo pip install .
After installation, do not run Python from inside the libectool/
directory. It contains a pyectool/
folder that may shadow the installed package.
Instead, test from a different directory:
cd ..
sudo env "PATH=$PATH" python -c "from pyectool import ECController; ec = ECController(); print(ec.is_on_ac())"
'env "PATH=$PATH"' preserve venv path and ensures the correct Python from your virtual environment is used even with sudo
.
from pyectool import ECController
ec = ECController()
Method | Description |
---|---|
ec.is_on_ac() -> bool |
Returns True if the system is on AC power, else False . |
ec.get_num_fans() -> int |
Returns the number of fan devices detected. |
ec.enable_fan_auto_ctrl(fan_idx: int) -> None |
Enables automatic fan control for a specific fan. |
ec.enable_all_fans_auto_ctrl() -> None |
Enables automatic control for all fans. |
ec.set_fan_duty(percent: int, fan_idx: int) -> None |
Sets fan duty (speed) as a percentage for a specific fan. |
ec.set_all_fans_duty(percent: int) -> None |
Sets the same duty percentage for all fans. |
ec.set_fan_rpm(target_rpm: int, fan_idx: int) -> None |
Sets a specific RPM target for a specific fan. |
ec.set_all_fans_rpm(target_rpm: int) -> None |
Sets the same RPM target for all fans. |
ec.get_fan_rpm(fan_idx: int) -> int |
Returns current RPM of a specific fan. |
ec.get_all_fans_rpm() -> list[int] |
Returns a list of current RPM values for all fans. |
ec.get_num_temp_sensors() -> int |
Returns the total number of temperature sensors detected. |
ec.get_temp(sensor_idx: int) -> int |
Returns the temperature (in °C) for the given sensor index. |
ec.get_all_temps() -> list[int] |
Returns a list of all sensor temperatures (in °C). |
ec.get_max_temp() -> int |
Returns the highest temperature across all sensors. |
ec.get_max_non_battery_temp() -> int |
Returns the highest temperature excluding battery-related sensors. |
ec.get_temp_info(sensor_idx: int) -> ECTempInfo |
Returns detailed info for a sensor, including name, type, and thresholds. |
Returned by get_temp_info()
, acts like a dict
with:
sensor_name
: strsensor_type
: inttemp
: inttemp_fan_off
: inttemp_fan_max
: int
BSD 3-Clause License
See the LICENSE
file for full terms.