Carrier Api Wrapper
Project description
carrier_api
Async Python client for Carrier Infinity and Bryant Evolution HVAC systems.
This package wraps the Carrier Infinity GraphQL API, exposes typed model objects for system profile, status, configuration, and energy data, and includes helper methods for common thermostat updates. It is intended for integrations and local automation code that need a lightweight API client rather than a full application.
The client is unofficial and depends on Carrier's private web and mobile API behavior. Carrier can change that API without notice.
Requirements
- Python 3.14 or newer
- A Carrier or Bryant account that works in the official app
- Network access to Carrier's cloud API
Installation
Install the published package in a virtual environment:
python3.14 -m venv .venv
.venv/bin/python -m pip install carrier_api
Live Smoke Test
The live smoke test connects to a real Carrier account, prints loaded system state, starts websocket listeners, and sends one sample manual activity update to the first available zone.
scripts/live_smoke_test
Use it only when you want to exercise the live Carrier API. It can change thermostat settings.
To run non-interactively, put credentials in a local env file and pass it with --credentials-file. Supported keys are CARRIER_USERNAME/CARRIER_PASSWORD.
scripts/live_smoke_test --credentials-file scripts/carrier_api.env --output-file /private/tmp/carrier_api.txt --schema-output-file /private/tmp/carrier_api_schema.json --read-only
Use --output-file when you want a full transcript for later API debugging or fixture updates. Use --schema-output-file to also capture the authenticated GraphQL introspection schema without temporarily editing the client. Use --read-only to skip the sample thermostat mutation while still loading systems, capturing schema, and listening for websocket messages.
Basic Usage
import asyncio
from carrier_api import ApiConnectionGraphql
async def main() -> None:
api = ApiConnectionGraphql(
username="your-account@example.com",
password="your-password",
)
try:
systems = await api.load_data()
for system in systems:
print(system.profile.name)
print(system.status.mode)
print(system.as_dict())
finally:
await api.cleanup()
asyncio.run(main())
load_data() authenticates when needed and returns a list of System objects. Each System includes:
profile: system identity and location metadatastatus: current mode, temperature, humidity, zone, and equipment stateconfig: configured zones, activities, schedules, and holdsenergy: reported energy usage data
Model objects provide as_dict() for structured serialization. Their string and repr forms are intended for readable debugging output.
System also exposes HVAC capability helpers:
supports_heat()supports_cool()supports_fan()supported_hvac_capabilities()
For zone-level profile resolution, call ConfigZone.current_status_activity(status_zone) with matching zones from the same system. This uses Carrier's reported current activity type from live status data. Use ConfigZone.current_scheduled_activity() when you want the activity implied by schedule and hold configuration instead.
Deprecated compatibility aliases:
StatusZone.current_activityis an alias forStatusZone.current_status_activity_type. It returns Carrier's live status-derived activity type/name.ConfigZone.current_activity()is an alias forConfigZone.current_scheduled_activity(). It returns the schedule/configuration-derived activity.
To get both activity sources together, use ConfigZone.as_dict(status_zone)["current_activity"] or System.as_dict()["config"]["zones"][...]["current_activity"]. That object has from_schedule and from_status keys. System.as_dict() performs the status/config pairing for the aggregate system.
System.as_dict() includes supported_hvac_capabilities and passes status-zone context into config serialization so each zone dictionary includes current_activity.from_schedule and current_activity.from_status.
Updating Thermostat Settings
Mutation helpers send Carrier configuration updates and then request websocket reconciliation when a websocket manager is available.
import asyncio
from carrier_api import ActivityTypes, ApiConnectionGraphql, FanModes, SystemModes
async def main() -> None:
api = ApiConnectionGraphql(
username="your-account@example.com",
password="your-password",
)
try:
systems = await api.load_data()
system = systems[0]
zone = system.config.zones[0]
await api.set_config_mode(system.profile.serial, SystemModes.AUTO)
await api.update_fan(
system_serial=system.profile.serial,
zone_id=zone.api_id,
activity_type=ActivityTypes.HOME,
fan_mode=FanModes.LOW,
)
await api.set_config_manual_activity(
system_serial=system.profile.serial,
zone_id=zone.api_id,
heat_set_point="68",
cool_set_point="74",
fan_mode=FanModes.LOW,
)
finally:
await api.cleanup()
asyncio.run(main())
Available update helpers include:
set_config_mode(...)set_config_heat_humidity(...)set_heat_source(...)set_humidifier(...)update_fan(...)set_config_hold(...)resume_schedule(...)set_config_manual_activity(...)
These methods can change real HVAC settings. Validate the selected system, zone, mode, and set points before calling them from automation.
Realtime Updates
ApiWebsocket manages Carrier realtime messages. WebsocketDataUpdater can merge incoming websocket messages into the System objects returned by load_data().
import asyncio
from carrier_api import ApiConnectionGraphql, WebsocketDataUpdater
async def main() -> None:
api = ApiConnectionGraphql(
username="your-account@example.com",
password="your-password",
)
try:
systems = await api.load_data()
updater = WebsocketDataUpdater(systems)
if api.api_websocket is None:
raise RuntimeError("websocket manager was not initialized")
api.api_websocket.callback_add(updater.message_handler)
await api.api_websocket.create_task_listener()
finally:
await api.cleanup()
asyncio.run(main())
Callbacks receive the raw websocket message text. Register WebsocketDataUpdater.message_handler first when later callbacks need to read the updated in-memory system state.
Development
Clone this repository and install the development environment:
git clone https://github.com/dahlb/carrier_api.git
cd carrier_api
scripts/setup
Use the repository scripts so commands run through the local .venv.
scripts/setup
scripts/test
scripts/lint
The underlying commands are:
./.venv/bin/pytest
./.venv/bin/prek run --all-files
./.venv/bin/mypy .
Add or update deterministic pytest coverage for behavior changes. Fixture data for GraphQL and websocket responses lives under tests/graphql and
tests/messages.
Updating the Captured Schema
To refresh captured GraphQL schema data, run the live smoke test with --schema-output-file:
scripts/live_smoke_test --credentials-file scripts/carrier_api.env --schema-output-file /private/tmp/carrier_api_schema.json --read-only
The command authenticates with Carrier, runs the GraphQL introspection query, and writes the schema JSON to the path you provide. Add --output-file /private/tmp/carrier_api.txt when you also want the full live smoke-test transcript.
Contributing
See CONTRIBUTING.md for contribution guidance. Please open bugs and feature requests in the issue tracker.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file carrier_api-3.3.0.tar.gz.
File metadata
- Download URL: carrier_api-3.3.0.tar.gz
- Upload date:
- Size: 60.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e841d099a5431f14cde254178565d684853ea668a5770025fafab3c01d859cdb
|
|
| MD5 |
5515d40af03c0d5c108a346d8a89b45f
|
|
| BLAKE2b-256 |
865b0c2d830cab3ab488f5f2c8dd1475302ad69c1c43001958d350a5c6bb5e45
|
File details
Details for the file carrier_api-3.3.0-py3-none-any.whl.
File metadata
- Download URL: carrier_api-3.3.0-py3-none-any.whl
- Upload date:
- Size: 37.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2cfa47c8d14c4c188aa7a436fdbd688c0cc4c584f3ba297611f96240eee58c64
|
|
| MD5 |
58a5ff123f8bdc813043f23b8a7eece1
|
|
| BLAKE2b-256 |
c85457f26af1e59e1d38de15f9ca8d3113bb2ad99b751033047ab0452ea0610a
|