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

Add Pterodactyl integration #141197

Merged
merged 17 commits into from
Mar 25, 2025
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
43 changes: 18 additions & 25 deletions homeassistant/components/pterodactyl/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,10 +24,6 @@
"""Raised when no data can be fechted from the server."""


class PterodactylNotInitializedError(Exception):
"""Raised when APIs are used although server instance is not initialized yet."""


@dataclass
class PterodactylData:
"""Data for the Pterodactyl server."""
Expand Down Expand Up @@ -74,9 +70,9 @@
PterodactylApiError,
) as error:
raise PterodactylConnectionError(error) from error
except Exception as error:
_LOGGER.exception("Unexpected exception occurred during initialization")
raise PterodactylConnectionError(error) from error

Check warning on line 75 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L73-L75

Added lines #L73 - L75 were not covered by tests
else:
servers = paginated_response.collect()
for server in servers:
Expand All @@ -86,23 +82,29 @@

async def async_get_data(self) -> dict[str, PterodactylData]:
"""Update the data from all Pterodactyl servers."""
data = {}

Check warning on line 85 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L85

Added line #L85 was not covered by tests

if self.pterodactyl is None:
raise PterodactylNotInitializedError(
"Pterodactyl API is not initialized yet"
)

try:
for identifier in self.identifiers:
for identifier in self.identifiers:
try:
server = await self.hass.async_add_executor_job(

Check warning on line 89 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L87-L89

Added lines #L87 - L89 were not covered by tests
self.pterodactyl.client.servers.get_server, identifier
self.pterodactyl.client.servers.get_server, # type: ignore[union-attr]
identifier,
)
utilization = await self.hass.async_add_executor_job(

Check warning on line 93 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L93

Added line #L93 was not covered by tests
self.pterodactyl.client.servers.get_server_utilization, identifier
self.pterodactyl.client.servers.get_server_utilization, # type: ignore[union-attr]
identifier,
)

except (

Check warning on line 97 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L97

Added line #L97 was not covered by tests
PydactylError,
BadRequestError,
PterodactylApiError,
) as error:
raise PterodactylConnectionError(error) from error
except Exception as error:
_LOGGER.exception("Unexpected exception occurred during data update")
raise PterodactylConnectionError(error) from error

Check warning on line 105 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L102-L105

Added lines #L102 - L105 were not covered by tests
else:
data[identifier] = PterodactylData(

Check warning on line 107 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L107

Added line #L107 was not covered by tests
name=server["name"],
uuid=server["uuid"],
identifier=identifier,
Expand All @@ -115,15 +117,6 @@
uptime=utilization["resources"]["uptime"],
)

_LOGGER.debug("%s", data[identifier])

Check warning on line 120 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L120

Added line #L120 was not covered by tests
except (
PydactylError,
BadRequestError,
PterodactylApiError,
) as error:
raise PterodactylConnectionError(error) from error
except Exception as error:
_LOGGER.exception("Unexpected exception occurred during data update")
raise PterodactylConnectionError(error) from error
else:
return data

return data

Check warning on line 122 in homeassistant/components/pterodactyl/api.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/api.py#L122

Added line #L122 was not covered by tests
23 changes: 8 additions & 15 deletions homeassistant/components/pterodactyl/binary_sensor.py
Original file line number Diff line number Diff line change
@@ -1,20 +1,20 @@
"""Binary sensor platform of the Pterodactyl integration."""

from homeassistant.components.binary_sensor import (

Check warning on line 3 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L3

Added line #L3 was not covered by tests
BinarySensorDeviceClass,
BinarySensorEntity,
BinarySensorEntityDescription,
)
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback

Check warning on line 9 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L8-L9

Added lines #L8 - L9 were not covered by tests

from .coordinator import PterodactylConfigEntry, PterodactylCoordinator
from .entity import PterodactylEntity

Check warning on line 12 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L11-L12

Added lines #L11 - L12 were not covered by tests

KEY_STATUS = "status"

Check warning on line 14 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L14

Added line #L14 was not covered by tests


BINARY_SENSOR_DESCRIPTIONS = [

Check warning on line 17 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L17

Added line #L17 was not covered by tests
BinarySensorEntityDescription(
key=KEY_STATUS,
translation_key=KEY_STATUS,
Expand All @@ -23,32 +23,30 @@
]

# Coordinator is used to centralize the data updates.
PARALLEL_UPDATES = 0

Check warning on line 26 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L26

Added line #L26 was not covered by tests


async def async_setup_entry(

Check warning on line 29 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L29

Added line #L29 was not covered by tests
hass: HomeAssistant,
config_entry: PterodactylConfigEntry,
async_add_entities: AddConfigEntryEntitiesCallback,
) -> None:
"""Set up the Pterodactyl binary sensor platform."""
coordinator = config_entry.runtime_data

Check warning on line 35 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L35

Added line #L35 was not covered by tests

async_add_entities(

Check warning on line 37 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L37

Added line #L37 was not covered by tests
[
PterodactylBinarySensorEntity(
coordinator, identifier, description, config_entry
)
for identifier in coordinator.api.identifiers
for description in BINARY_SENSOR_DESCRIPTIONS
]
PterodactylBinarySensorEntity(
coordinator, identifier, description, config_entry
)
for identifier in coordinator.api.identifiers
for description in BINARY_SENSOR_DESCRIPTIONS
)


class PterodactylBinarySensorEntity(PterodactylEntity, BinarySensorEntity):

Check warning on line 46 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L46

Added line #L46 was not covered by tests
"""Representation of a Pterodactyl binary sensor base entity."""

def __init__(

Check warning on line 49 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L49

Added line #L49 was not covered by tests
self,
coordinator: PterodactylCoordinator,
identifier: str,
Expand All @@ -56,18 +54,13 @@
config_entry: PterodactylConfigEntry,
) -> None:
"""Initialize binary sensor base entity."""
super().__init__(coordinator, identifier, config_entry)
self.entity_description = description
self.identifier = identifier
self._attr_unique_id = f"{config_entry.entry_id}-{identifier}-{description.key}"
self._attr_is_on = False

@property
def available(self) -> bool:
"""Return binary sensor availability."""
return super().available and self.identifier in self.coordinator.data
self._attr_unique_id = (

Check warning on line 59 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L57-L59

Added lines #L57 - L59 were not covered by tests
f"{self.coordinator.data[self.identifier].uuid}-{description.key}"
)

@property
def is_on(self) -> bool:

Check warning on line 64 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L63-L64

Added lines #L63 - L64 were not covered by tests
"""Return binary sensor state."""
return self.coordinator.data[self.identifier].state == "running"

Check warning on line 66 in homeassistant/components/pterodactyl/binary_sensor.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/binary_sensor.py#L66

Added line #L66 was not covered by tests
14 changes: 7 additions & 7 deletions homeassistant/components/pterodactyl/coordinator.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,6 @@
PterodactylConfigurationError,
PterodactylConnectionError,
PterodactylData,
PterodactylNotInitializedError,
)

SCAN_INTERVAL = timedelta(seconds=60)
Expand Down Expand Up @@ -49,19 +48,20 @@

async def _async_setup(self) -> None:
"""Set up the Pterodactyl data coordinator."""
self.api = PterodactylAPI(
hass=self.hass,
host=self.config_entry.data[CONF_URL],
api_key=self.config_entry.data[CONF_API_KEY],
)

try:
self.api = PterodactylAPI(
hass=self.hass,
host=self.config_entry.data[CONF_URL],
api_key=self.config_entry.data[CONF_API_KEY],
)
await self.api.async_init()
except PterodactylConfigurationError as error:
raise ConfigEntryNotReady(error) from error

Check warning on line 60 in homeassistant/components/pterodactyl/coordinator.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/coordinator.py#L60

Added line #L60 was not covered by tests

async def _async_update_data(self) -> dict[str, PterodactylData]:
"""Get updated data from the Pterodactyl server."""
try:
return await self.api.async_get_data()
except (PterodactylNotInitializedError, PterodactylConnectionError) as error:
except PterodactylConnectionError as error:
raise UpdateFailed(error) from error

Check warning on line 67 in homeassistant/components/pterodactyl/coordinator.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/coordinator.py#L64-L67

Added lines #L64 - L67 were not covered by tests
6 changes: 6 additions & 0 deletions homeassistant/components/pterodactyl/entity.py
Original file line number Diff line number Diff line change
@@ -1,31 +1,32 @@
"""Base entity for the Pterodactyl integration."""

from homeassistant.config_entries import ConfigEntry
from homeassistant.const import CONF_URL
from homeassistant.helpers.device_registry import DeviceInfo
from homeassistant.helpers.update_coordinator import CoordinatorEntity

Check warning on line 6 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L3-L6

Added lines #L3 - L6 were not covered by tests

from .const import DOMAIN
from .coordinator import PterodactylCoordinator

Check warning on line 9 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L8-L9

Added lines #L8 - L9 were not covered by tests

MANUFACTURER = "Pterodactyl"

Check warning on line 11 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L11

Added line #L11 was not covered by tests


class PterodactylEntity(CoordinatorEntity[PterodactylCoordinator]):

Check warning on line 14 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L14

Added line #L14 was not covered by tests
"""Representation of a Pterodactyl base entity."""

_attr_has_entity_name = True

Check warning on line 17 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L17

Added line #L17 was not covered by tests

def __init__(

Check warning on line 19 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L19

Added line #L19 was not covered by tests
self,
coordinator: PterodactylCoordinator,
identifier: str,
config_entry: ConfigEntry,
) -> None:
"""Initialize base entity."""
super().__init__(coordinator)

Check warning on line 26 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L26

Added line #L26 was not covered by tests

self.identifier = identifier
self._attr_device_info = DeviceInfo(

Check warning on line 29 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L28-L29

Added lines #L28 - L29 were not covered by tests
identifiers={(DOMAIN, identifier)},
manufacturer=MANUFACTURER,
name=coordinator.data[identifier].name,
Expand All @@ -33,3 +34,8 @@
model_id=coordinator.data[identifier].uuid,
configuration_url=f"{config_entry.data[CONF_URL]}/server/{identifier}",
)

@property
def available(self) -> bool:

Check warning on line 39 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L38-L39

Added lines #L38 - L39 were not covered by tests
"""Return binary sensor availability."""
return super().available and self.identifier in self.coordinator.data

Check warning on line 41 in homeassistant/components/pterodactyl/entity.py

View check run for this annotation

Codecov / codecov/patch

homeassistant/components/pterodactyl/entity.py#L41

Added line #L41 was not covered by tests
4 changes: 2 additions & 2 deletions homeassistant/components/pterodactyl/strings.json
Original file line number Diff line number Diff line change
Expand Up @@ -8,15 +8,15 @@
},
"data_description": {
"url": "The IP address or hostname of your Pterodactyl server, starting with either 'http://' or 'https://', optionally including the port at the end.",
"api_key": "The client (account) API key for accessing your Pterodactyl server."
"api_key": "The account API key for accessing your Pterodactyl server."
}
}
},
"error": {
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]"
},
"abort": {
"already_configured": "[%key:common::config_flow::abort::already_configured_device%]"
"already_configured": "[%key:common::config_flow::abort::already_configured_service%]"
}
},
"entity": {
Expand Down
Loading