Compare commits

...

8 Commits

Author SHA1 Message Date
jfm
036910f06f Fixed missing fields 2025-10-30 10:15:01 +01:00
jfm
7f91f330ac Testing expires in client 2025-10-30 09:53:38 +01:00
jfm
15de00fae5 Setup 2025-10-30 09:18:51 +01:00
jfm
122ecdc2ad Version bump 2025-10-30 09:16:46 +01:00
jfm
a3e414ad00 Fixed url handling 2025-10-30 09:14:27 +01:00
jfm
1f9b108dd0 Python 3.13 for home assistant 2025-10-30 07:50:04 +01:00
jfm
027a16a769 Readded setup.py 2025-10-30 07:09:27 +01:00
jfm
40f23cf64b Readded setup.py 2025-10-30 07:09:12 +01:00
8 changed files with 144 additions and 40 deletions

View File

@@ -1,9 +1,9 @@
[project]
name = "rapt-cloud-api"
version = "0.2.2"
version = "0.2.4"
description = "Python bindings for the Rapt.io API"
readme = "README.md"
requires-python = ">=3.14"
requires-python = ">=3.13"
dependencies = [
"aiohttp>=3.13.1",
"loguru>=0.7.3",

23
setup.py Normal file
View File

@@ -0,0 +1,23 @@
from setuptools import find_packages, setup
with open("README.md", "r", encoding="utf-8") as fh:
long_description = fh.read()
setup(
name='rapt-cloud-api',
version='0.2.4',
author='Jesper Fussing Mørk',
author_email='jfmo@moerks.dk',
description='Client Library for the rapt.io api',
long_description=long_description,
long_description_content_type="text/markdown",
url='https://gitea.moerks.dk/jfm/rapt-cloud-api',
project_urls={
"Bug Tracker": "https://gitea.moerks.dk/jfm/rapt-cloud-api/issues",
},
license='MIT',
package_dir={"": "src"},
packages=find_packages(where="src"),
install_requires=["loguru", "pydantic", "aiohttp"]
)

View File

@@ -51,8 +51,6 @@ class Client():
raise HttpClientException(response.reason, response.status)
async def get_auth_headers(self):
if self.expires is not None:
logger.trace("DATE COMP: " + str(self.expires <= datetime.now()))
if self.expires is None or self.expires <= datetime.now():
await self.get_jwt_token()
@@ -61,7 +59,7 @@ class Client():
async def get_json(self, url, parameters):
headers = await self.get_auth_headers()
logger.trace("HEADERS: " + str(headers))
async with self.session.get(url, headers=headers, params=parameters) as response:
async with self.session.get(self.host+url, headers=headers, params=parameters) as response:
response_json = await response.json()
logger.trace(response.request_info.url)
if response.status == 200:

View File

@@ -80,11 +80,11 @@ class ProfileStepModel(RaptBaseModel):
class ProfileSessionModel(RaptBaseModel):
name: Optional[str]
description: Optional[str]
profile_id: Optional[UUID] = Field(alias="profileId")
description: Optional[str] = None
profile_id: Optional[UUID] = Field(alias="profileId", default=None)
# profile: ProfileModel
brewzilla_id: Optional[UUID] = Field(alias="brewZillaId")
fermentation_chamger_id: Optional[UUID] = Field(alias="fermentationChamberId")
brewzilla_id: Optional[UUID] = Field(alias="brewZillaId", default=None)
fermentation_chamger_id: Optional[UUID] = Field(alias="fermentationChamberId", default=None)
hydrometer_id: Optional[UUID] = Field(alias="hydrometerId")
still_id: Optional[UUID] = Field(alias="stillId")
temperature_controller_id: Optional[UUID] = Field(alias="temperatureControllerId")
@@ -113,22 +113,22 @@ class ProfileModel(RaptBaseModel):
class ProfileSessionStatusModel(RaptBaseModel):
name: Optional[str]
description: Optional[str]
profile_id: Optional[UUID] = Field(alias="profileId")
profile: ProfileModel
brewzilla_id: Optional[UUID] = Field(alias="brewZillaId")
fermentation_chamger_id: Optional[UUID] = Field(alias="fermentationChamberId")
hydrometer_id: Optional[UUID] = Field(alias="hydrometerId")
still_id: Optional[UUID] = Field(alias="stillId")
temperature_controller_id: Optional[UUID] = Field(alias="temperatureControllerId")
start_date: Optional[datetime] = Field(alias="startDate")
end_date: Optional[datetime] = Field(alias="endDate")
original_gravity: Optional[float] = Field(alias="originalGravity")
final_gravity: Optional[float] = Field(alias="finalGravity")
yeast_id: Optional[UUID] = Field(alias="yeastId")
yeast: YeastModel
sent_alerts: List[Optional[ProfileAlertModel]] = Field(alias="sentAlerts")
estimated_end_date: Optional[datetime] = Field(alias="estimatedEndDate")
profile_length: Optional[float] = Field(alias="profileLength")
current_profile_time: Optional[float] = Field(alias="currentProfileTime")
remaining_profile_time: Optional[float] = Field(alias="remainingProfileTime")
description: Optional[str] = None
profile_id: Optional[UUID] = Field(alias="profileId", default=None)
profile: Optional[ProfileModel] = None
brewzilla_id: Optional[UUID] = Field(alias="brewZillaId", default=None)
fermentation_chamger_id: Optional[UUID] = Field(alias="fermentationChamberId", default=None)
hydrometer_id: Optional[UUID] = Field(alias="hydrometerId", default=None)
still_id: Optional[UUID] = Field(alias="stillId", default=None)
temperature_controller_id: Optional[UUID] = Field(alias="temperatureControllerId", default=None)
start_date: Optional[datetime] = Field(alias="startDate", default=None)
end_date: Optional[datetime] = Field(alias="endDate", default=None)
original_gravity: Optional[float] = Field(alias="originalGravity", default=None)
final_gravity: Optional[float] = Field(alias="finalGravity", default=None)
yeast_id: Optional[UUID] = Field(alias="yeastId", default=None)
yeast: Optional[YeastModel] = None
sent_alerts: List[Optional[ProfileAlertModel]] = Field(alias="sentAlerts", default=[])
estimated_end_date: Optional[datetime] = Field(alias="estimatedEndDate", default=None)
profile_length: Optional[float] = Field(alias="profileLength", default=None)
current_profile_time: Optional[float] = Field(alias="currentProfileTime", default=None)
remaining_profile_time: Optional[float] = Field(alias="remainingProfileTime", default=None)

View File

@@ -0,0 +1,48 @@
[
{
"telemetry": [
{
"temperature": 24.6875,
"gravity": 1018.35,
"gravityVelocity": 0.13616,
"battery": 100,
"version": "20250319_055542_b3788ba",
"id": "88a30cdb-3a3d-4ed1-8db1-fe82f53eb127",
"rowKey": "2516404896183910585",
"createdOn": "2025-10-30T07:59:41.6089414+00:00",
"macAddress": "ac-15-18-df-84-94",
"rssi": -83
}
],
"temperature": 24.6875,
"gravity": 1018.35,
"gravityVelocity": 0.13616,
"battery": 100,
"name": "Hegnsgården Yellow",
"macAddress": "ac-15-18-df-84-94",
"deviceType": "Hydrometer",
"active": false,
"disabled": false,
"lastActivityTime": "2025-10-30T07:59:41.6089414+00:00",
"rssi": -83,
"firmwareVersion": "20250319_055542_b3788ba",
"isLatestFirmware": false,
"activeProfileSession": {
"name": "Batch 1",
"hydrometerId": "2aa3b02c-78de-4715-8f5b-61bf7c3d1b62",
"startDate": "2025-10-26T07:05:43.257+00:00",
"modifiedOn": "2025-10-26T07:06:01.4699396+00:00",
"modifiedBy": "00000000-0000-0000-0000-000000000000",
"id": "3782039d-c829-465e-bb27-aff7bd50c2bb",
"deleted": false,
"createdOn": "2025-10-26T07:06:01.468437+00:00",
"createdBy": "475ebc33-8e90-4be5-2424-08ddfbe49482"
},
"modifiedOn": "2025-10-30T07:59:41.61736+00:00",
"modifiedBy": "00000000-0000-0000-0000-000000000000",
"id": "2aa3b02c-78de-4715-8f5b-61bf7c3d1b62",
"deleted": false,
"createdOn": "2025-10-07T09:44:01.1515163+00:00",
"createdBy": "475ebc33-8e90-4be5-2424-08ddfbe49482"
}
]

24
tests/test_client.py Normal file
View File

@@ -0,0 +1,24 @@
import pytest
import json
from aiohttp import web
from aioresponses import aioresponses
from rapt.client import Client
@pytest.fixture
async def client(aiohttp_client):
app = web.Application()
# session = await aiohttp_client(app)
yield Client("test", "test", None)
async def test_get_hydrometers(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"), repeat=1)
headers1 = await client.get_auth_headers()
headers2 = await client.get_auth_headers()
assert headers1 == headers2
def json_reader(path):
with open(path) as f:
return json.load(f)

View File

@@ -15,7 +15,7 @@ async def client(aiohttp_client):
async def test_get_fermentation_chambers(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/fermentationchambers/getfermentationchambers", payload=json_reader("./tests/json/get_fermentation_chambers_response.json"))
responses.get("https://api.rapt.io/api/fermentationchambers/getfermentationchambers", payload=json_reader("./tests/json/get_fermentation_chambers_response.json"))
fermentation_chamber = FermentationChamber(client)
ferms = await fermentation_chamber.get_fermentation_chambers()
@@ -25,7 +25,7 @@ async def test_get_fermentation_chambers(client):
async def test_get_fermentation_chambers_500(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/fermentationchambers/getfermentationchambers", status=500)
responses.get("https://api.rapt.io/api/fermentationchambers/getfermentationchambers", status=500)
fermentation_chamber = FermentationChamber(client)
ferms = await fermentation_chamber.get_fermentation_chambers()
@@ -34,7 +34,7 @@ async def test_get_fermentation_chambers_500(client):
async def test_get_fermentation_chamber(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/fermentationchambers/getfermentationchamber", payload=json_reader("./tests/json/get_fermentation_chamber_response.json"))
responses.get("https://api.rapt.io/api/fermentationchambers/getfermentationchamber", payload=json_reader("./tests/json/get_fermentation_chamber_response.json"))
fermentation_chamber = FermentationChamber(client)
ferms = await fermentation_chamber.get_fermentation_chamber("")
@@ -43,7 +43,7 @@ async def test_get_fermentation_chamber(client):
async def test_get_fermentation_chamber_500(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/fermentationchambers/getfermentationchamber", status=500)
responses.get("https://api.rapt.io/api/fermentationchambers/getfermentationchamber", status=500)
fermentation_chamber = FermentationChamber(client)
ferms = await fermentation_chamber.get_fermentation_chamber("")
@@ -52,7 +52,7 @@ async def test_get_fermentation_chamber_500(client):
async def test_get_telemetry(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/fermentationchambers/gettelemetry", payload=json_reader("./tests/json/get_fermentation_chamber_telemetry_response.json"))
responses.get("https://api.rapt.io/api/fermentationchambers/gettelemetry", payload=json_reader("./tests/json/get_fermentation_chamber_telemetry_response.json"))
fermentation_chamber = FermentationChamber(client)
telemetry = await fermentation_chamber.get_telemetry("", "", "", "")
@@ -62,7 +62,7 @@ async def test_get_telemetry(client):
async def test_get_telemetry_500(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/fermentationchambers/gettelemetry", status=500)
responses.get("https://api.rapt.io/api/fermentationchambers/gettelemetry", status=500)
fermentation_chamber = FermentationChamber(client)
telemetry = await fermentation_chamber.get_telemetry("", "", "", "")

View File

@@ -15,17 +15,28 @@ async def client(aiohttp_client):
async def test_get_hydrometers(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/hydrometers/gethydrometers", payload=json_reader("./tests/json/get_hydrometers_response.json"))
responses.get("https://api.rapt.io/api/hydrometers/gethydrometers", payload=json_reader("./tests/json/get_hydrometers_response.json"))
hydrometer = Hydrometer(client)
hydros = await hydrometer.get_hydrometers()
assert hydros is not None
assert len(hydros) == 1
async def test_get_hydrometers_missing_fields(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("https://api.rapt.io/api/hydrometers/gethydrometers", payload=json_reader("./tests/json/get_hydrometers_response_missing_fields.json"))
hydrometer = Hydrometer(client)
hydros = await hydrometer.get_hydrometers()
assert hydros is not None
assert len(hydros) == 1
async def test_get_hydrometers_500(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/hydrometers/gethydrometers", status=500)
responses.get("https://api.rapt.io/api/hydrometers/gethydrometers", status=500)
hydrometer = Hydrometer(client)
hydros = await hydrometer.get_hydrometers()
@@ -34,7 +45,7 @@ async def test_get_hydrometers_500(client):
async def test_get_hydrometer(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/hydrometers/gethydrometer", payload=json_reader("./tests/json/get_hydrometer_response.json"))
responses.get("https://api.rapt.io/api/hydrometers/gethydrometer", payload=json_reader("./tests/json/get_hydrometer_response.json"))
hydrometer = Hydrometer(client)
hydros = await hydrometer.get_hydrometer("")
@@ -43,7 +54,7 @@ async def test_get_hydrometer(client):
async def test_get_hydrometer_500(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/hydrometers/gethydrometer", status=500)
responses.get("https://api.rapt.io/api/hydrometers/gethydrometer", status=500)
hydrometer = Hydrometer(client)
hydros = await hydrometer.get_hydrometer("")
@@ -52,7 +63,7 @@ async def test_get_hydrometer_500(client):
async def test_get_telemetry(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/hydrometers/gettelemetry", payload=json_reader("./tests/json/get_hydrometer_telemetry_response.json"))
responses.get("https://api.rapt.io/api/hydrometers/gettelemetry", payload=json_reader("./tests/json/get_hydrometer_telemetry_response.json"))
hydrometer = Hydrometer(client)
telemetry = await hydrometer.get_telemetry("", "", "", "")
@@ -61,7 +72,7 @@ async def test_get_telemetry(client):
async def test_get_telemetry_500(client):
with aioresponses() as responses:
responses.post("https://id.rapt.io/connect/token", payload=json_reader("./tests/json/token_response.json"))
responses.get("/api/hydrometers/gettelemetry", status=500)
responses.get("https://api.rapt.io/api/hydrometers/gettelemetry", status=500)
hydrometer = Hydrometer(client)
telemetry = await hydrometer.get_telemetry("", "", "", "")