Asynchronous Python client for the Hager TJA470 Intercom API
Project description
aiotja470_intercom
An asynchronous, stateless Python client for the Hager TJA470 Intercom API.
This library was heavily designed to act as the underlying foundation for Home Assistant integrations. It relies purely on dependency injection for network requests, correctly manages raw session cookies over IP addresses, and provides an end-to-end command-line interface for local testing.
⚠️ Disclaimer: This is an unofficial library. It is not affiliated with, endorsed by, or supported by Hager. Use it at your own risk.
Installation
pip install aiotja470_intercom
CLI Usage
This package bundles a highly useful tja470 binary that acts as a local test bench. It replicates exact Home Assistant session storage to ~/.tja470_config.json, including caching the aiohttp cookie jars!
1. Pair a new device
You only need to run this once. It verifies credentials, registers a UUID as a mobile client, and saves your configuration and JSESSIONID cookies.
# If you don't provide a --uuid, a random one will be generated and registered.
tja470 pair --host 192.168.1.100 --username user@example.com --password my_password
2. View Connection Status
Verifies the current session, fetches the latest provisioning data (including SIP credentials and RTSP URLs), and lists all extensions.
tja470 status
3. Run Intercom Commands
Run commands utilizing the cached session cookies without re-authenticating!
# Open the door
tja470 run --open-door
# Open the door at a specific camera position index (e.g. 0)
# (Cycles the camera feed to position 0 first, then triggers release)
tja470 run --open-door-at 0
# Switch camera
tja470 run --switch-camera
# View raw provisioning data
tja470 run --provisioning
Debugging
You can attach --debug to any command to enable verbose HTTP tracing. It will print the exact requests, headers, sent cookies, and raw JSON response content!
tja470 --debug status
Python API Usage
The client is completely decoupled from the network request layer via a Runner protocol. This allows you to easily inject a mock runner during tests.
Basic Setup
import asyncio
import aiohttp
from aiotja470_intercom import TJA470IntercomClient, AiohttpRunner
async def main():
# The AiohttpRunner automatically initializes an aiohttp.CookieJar(unsafe=True)
# under the hood so it correctly caches cookies from local IP addresses.
runner = AiohttpRunner()
client = TJA470IntercomClient(
host="192.168.1.100",
username="user@example.com",
password="your_password",
runner=runner
)
try:
# Verify the manifest
manifest = await client.get_manifest()
print(f"Firmware: {manifest.fw}")
# Execute commands
await client.open_door(door_id=1)
finally:
await runner.close()
if __name__ == "__main__":
asyncio.run(main())
Session & Cookie Management
For Home Assistant integrations, you do not want to spam the TJA-470 with Basic Auth headers on every request.
The client is hard-coded with a cookie-first fallback loop. It attempts the request utilizing the cached cookies first. If the intercom rejects it (e.g. cookie expired), it automatically catches the 401 Unauthorized, re-authenticates using your credentials to get a fresh session cookie, and seamlessly retries the request!
# Extract all cookies to persist across reboots (e.g. in HA ConfigEntry.data)
cookies_dict = client.get_cookies()
# Later, on integration startup, re-inject the cookies
client.set_cookies(cookies_dict)
Advanced API Features
# 1. Fetching available devices for pairing
# Retrieves all Unassigned 'MOBILE_CLIENT' slots in the Hager UI
devices = await client.get_free_devices()
device_id = devices[0].id
# 2. Registering a UUID to an open device slot
await client.set_uid(device_id, "your-uuid-string")
# 3. Fetching the Provisioning Info
# This returns all the juicy details (SIP passwords, RTSP urls, etc.)
prov = await client.get_provisioning("your-uuid-string")
print(f"SIP Password: {prov.sip_info.sip_password}")
print(f"RTSP Stream: {prov.rtsp_video_url}")
for ext in prov.called_elements:
print(f"Known device: {ext.name} (SIP: {ext.sip_id})")
# 4. Opening the door
await client.open_door(door_id=1)
# 5. Switching camera feeds
await client.switch_camera("your-uuid-string")
# 6. Switching to a specific camera position and opening the door
await client.open_door_at_position("your-uuid-string", position=0)
SIP & Audio Stream API
The library includes an asynchronous wrapper around pyVoIP to handle SIP registration, make/receive VoIP calls, and stream audio.
To avoid blocking the asyncio event loop (a critical requirement for Home Assistant integrations), all blocking pyVoIP I/O operations are run in an executor thread, and callbacks are dispatched thread-safely back to the main loop.
Basic SIP Setup
from aiotja470_intercom import TJA470SipPhone, TJA470SipCall, PhoneStatus
# Initialize the SIP Phone client using provisioning details
sip_phone = TJA470SipPhone(
host="192.168.1.100", # Intercom host IP
sip_id="6004", # SIP Extension username from provisioning
sip_password="your_password", # SIP password from provisioning
local_ip="192.168.1.50", # Local IP of the system running the client
sip_port=5060, # Local SIP port to bind
)
# Register callbacks for incoming calls and status changes
async def on_incoming_call(call: TJA470SipCall):
print(f"Incoming call from: {call.caller}")
# Answer the call
await call.answer()
# Process audio stream asynchronously
async for frame in call.audio_stream(frame_size=320, convert_16bit=True):
# frame is a 20ms chunk of 16-bit linear PCM audio at 8000Hz (mono)
# Process the audio or pipe it to a websocket/media player
pass
async def on_registration_status(status: PhoneStatus):
print(f"SIP Registration Status: {status}")
sip_phone.register_incoming_call_callback(on_incoming_call)
sip_phone.register_registration_state_callback(on_registration_status)
# Start the SIP client (initiates non-blocking registration)
await sip_phone.start()
# Later, initiate an outgoing call
# call = await sip_phone.call("6000")
# Stop the SIP client and unregister
# await sip_phone.stop()
Audio Processing
The library provides helper methods for reading and writing audio in standard formats (8-bit or 16-bit PCM at 8000Hz):
await call.read_audio_16bit(length=320): Reads standard 16-bit linear PCM audio.await call.write_audio_16bit(data): Converts and transmits standard 16-bit linear PCM audio.call.audio_stream(frame_size=320, convert_16bit=True): Async generator helper to continuously consume incoming audio.
Exception Handling
The library provides native Home Assistant style typed exceptions:
TJA470Error: Base class for all library errors.TJA470ConnectionError: Raised on timeouts, unreachable host, or DNS failures.TJA470AuthError: Raised on401 Unauthorizedor403 Forbidden.TJA470ResponseError: Raised when the intercom returns invalid JSON or unexpected schema data.TJA470SipError: Raised during SIP or audio streaming operations.
Project details
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 aiotja470_intercom-0.1.6.tar.gz.
File metadata
- Download URL: aiotja470_intercom-0.1.6.tar.gz
- Upload date:
- Size: 23.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
78323fc939ec682d539a0ad33917871a318168a944765163cdcaafc0ffcd5739
|
|
| MD5 |
6489116e8424295823eec69d47bce033
|
|
| BLAKE2b-256 |
3b3a917c1fafedd0537afc0038143fde50a17166362e15f38736ed267ed696c3
|
File details
Details for the file aiotja470_intercom-0.1.6-py3-none-any.whl.
File metadata
- Download URL: aiotja470_intercom-0.1.6-py3-none-any.whl
- Upload date:
- Size: 20.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
9ae0a096a177c92c9f9d3a25eb138d757f07706414ed97a207e4a03745e2e237
|
|
| MD5 |
fb28f4564b5af514df7f409fe356ce23
|
|
| BLAKE2b-256 |
b1303457f1789011fc273beaeab0328bfdcc1c4557f7a52470080b035103c55a
|