Habiticalib 0.3.2__py3-none-any.whl → 0.3.4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
habiticalib/__init__.py CHANGED
@@ -16,6 +16,7 @@ from .types import (
16
16
  ContentData,
17
17
  Direction,
18
18
  Frequency,
19
+ HabiticaCastSkillResponse,
19
20
  HabiticaClass,
20
21
  HabiticaClassSystemResponse,
21
22
  HabiticaContentResponse,
@@ -32,12 +33,13 @@ from .types import (
32
33
  HabiticaTaskOrderResponse,
33
34
  HabiticaTaskResponse,
34
35
  HabiticaTasksResponse,
35
- HabiticaUserAnonymizedrResponse,
36
+ HabiticaUserAnonymizedResponse,
36
37
  HabiticaUserExport,
37
38
  HabiticaUserResponse,
38
39
  Language,
39
40
  LoginData,
40
41
  QuestData,
42
+ Repeat,
41
43
  ScoreData,
42
44
  Skill,
43
45
  StatsUser,
@@ -62,6 +64,7 @@ __all__ = [
62
64
  "Direction",
63
65
  "Frequency",
64
66
  "Habitica",
67
+ "HabiticaCastSkillResponse",
65
68
  "HabiticaClass",
66
69
  "HabiticaClassSystemResponse",
67
70
  "HabiticaContentResponse",
@@ -79,7 +82,7 @@ __all__ = [
79
82
  "HabiticaTaskOrderResponse",
80
83
  "HabiticaTaskResponse",
81
84
  "HabiticaTasksResponse",
82
- "HabiticaUserAnonymizedrResponse",
85
+ "HabiticaUserAnonymizedResponse",
83
86
  "HabiticaUserExport",
84
87
  "HabiticaUserResponse",
85
88
  "Language",
@@ -87,6 +90,7 @@ __all__ = [
87
90
  "NotAuthorizedError",
88
91
  "NotFoundError",
89
92
  "QuestData",
93
+ "Repeat",
90
94
  "ScoreData",
91
95
  "Skill",
92
96
  "StatsUser",
habiticalib/const.py CHANGED
@@ -1,18 +1,48 @@
1
1
  """Constants for Habiticalib."""
2
2
 
3
- __version__ = "0.3.2"
3
+ __version__ = "0.3.4"
4
4
 
5
5
  DEFAULT_URL = "https://habitica.com/"
6
6
  ASSETS_URL = "https://habitica-assets.s3.amazonaws.com/mobileApp/images/"
7
7
 
8
8
  DEVELOPER_ID = "4c4ca53f-c059-4ffa-966e-9d29dd405daf"
9
9
 
10
- BACKER_ONLY_GEAR = {
10
+ # Assets that doesn't follow name conventions like 2019 Kickstarter gear
11
+ # https://github.com/HabitRPG/habitica/blob/develop/website/client/src/assets/css/sprites.css
12
+ SPECIAL_ASSETS = {
11
13
  "armor_special_ks2019": "BackerOnly-Equip-MythicGryphonArmor.gif",
14
+ "back_special_heroicAureole": "back_special_heroicAureole.gif",
15
+ "background_airship": "background_airship.gif",
16
+ "background_clocktower": "background_clocktower.gif",
17
+ "background_steamworks": "background_steamworks.gif",
18
+ "broad_armor_special_0": "BackerOnly-Equip-ShadeArmor.gif",
19
+ "broad_armor_special_1": "ContributorOnly-Equip-CrystalArmor.gif",
20
+ "broad_armor_special_ks2019": "BackerOnly-Equip-MythicGryphonArmor.gif",
12
21
  "eyewear_special_ks2019": "BackerOnly-Equip-MythicGryphonVisor.gif",
22
+ "head_special_0": "BackerOnly-Equip-ShadeHelmet.gif",
23
+ "head_special_1": "ContributorOnly-Equip-CrystalHelmet.gif",
13
24
  "head_special_ks2019": "BackerOnly-Equip-MythicGryphonHelm.gif",
25
+ "Mount_Body_Gryphon-Gryphatrice": "BackerOnly-Mount-Body-Gryphatrice.gif",
26
+ "Mount_Head_Gryphon-Gryphatrice": "BackerOnly-Mount-Head-Gryphatrice.gif",
27
+ "Pet-Gryphatrice-Jubilant": "Pet-Gryphatrice-Jubilant.gif",
28
+ "Pet-Gryphon-Gryphatrice": "BackerOnly-Pet-Gryphatrice.gif",
29
+ "Pet-Wolf-Cerberus": "BackerOnly-Pet-CerberusPup.gif",
30
+ "shield_special_0": "BackerOnly-Shield-TormentedSkull.gif",
14
31
  "shield_special_ks2019": "BackerOnly-Equip-MythicGryphonShield.gif",
32
+ "slim_armor_special_0": "BackerOnly-Equip-ShadeArmor.gif",
33
+ "slim_armor_special_1": "ContributorOnly-Equip-CrystalArmor.gif",
34
+ "slim_armor_special_ks2019": "BackerOnly-Equip-MythicGryphonArmor.gif",
35
+ "weapon_special_0": "BackerOnly-Weapon-DarkSoulsBlade.gif",
36
+ "weapon_special_critical": "weapon_special_critical.gif",
15
37
  "weapon_special_ks2019": "BackerOnly-Equip-MythicGryphonGlaive.gif",
16
38
  }
17
39
 
40
+ SPECIAL_ASSETS_OFFSET = {
41
+ "head_special_0": (-3, -18),
42
+ "weapon_special_0": (-3, -18),
43
+ "weapon_special_critical": (-12, 12),
44
+ "weapon_special_1": (-12, 0),
45
+ "head_special_1": (0, 3),
46
+ }
47
+
18
48
  PAGE_LIMIT = 60
habiticalib/ha.py ADDED
@@ -0,0 +1,11 @@
1
+ """Constants for Home Assistant."""
2
+
3
+ HP = ""
4
+ XP = ""
5
+ MP = ""
6
+ GP = ""
7
+ WARRIOR = ""
8
+ ROGUE = ""
9
+ WIZARD = ""
10
+ HEALER = ""
11
+ FOOD = ""
habiticalib/lib.py CHANGED
@@ -6,6 +6,7 @@ import asyncio
6
6
  from http import HTTPStatus
7
7
  from io import BytesIO
8
8
  import logging
9
+ from operator import add
9
10
  from typing import IO, TYPE_CHECKING, Any, Self
10
11
 
11
12
  from aiohttp import ClientError, ClientResponseError, ClientSession
@@ -13,7 +14,13 @@ from habitipy.aio import HabitipyAsync # type: ignore[import-untyped]
13
14
  from PIL import Image
14
15
  from yarl import URL
15
16
 
16
- from .const import ASSETS_URL, BACKER_ONLY_GEAR, DEFAULT_URL, PAGE_LIMIT
17
+ from .const import (
18
+ ASSETS_URL,
19
+ DEFAULT_URL,
20
+ PAGE_LIMIT,
21
+ SPECIAL_ASSETS,
22
+ SPECIAL_ASSETS_OFFSET,
23
+ )
17
24
  from .exceptions import (
18
25
  BadRequestError,
19
26
  NotAuthorizedError,
@@ -30,6 +37,7 @@ from .helpers import (
30
37
  from .types import (
31
38
  Attributes,
32
39
  Direction,
40
+ HabiticaCastSkillResponse,
33
41
  HabiticaClass,
34
42
  HabiticaClassSystemResponse,
35
43
  HabiticaContentResponse,
@@ -46,7 +54,7 @@ from .types import (
46
54
  HabiticaTaskOrderResponse,
47
55
  HabiticaTaskResponse,
48
56
  HabiticaTasksResponse,
49
- HabiticaUserAnonymizedrResponse,
57
+ HabiticaUserAnonymizedResponse,
50
58
  HabiticaUserExport,
51
59
  HabiticaUserResponse,
52
60
  Language,
@@ -112,7 +120,7 @@ class Habitica:
112
120
  async def _request(self, method: str, url: URL, **kwargs) -> str:
113
121
  """Handle API request."""
114
122
  async with self._session.request(
115
- method,
123
+ method.upper(),
116
124
  url,
117
125
  headers=self._headers,
118
126
  **kwargs,
@@ -254,7 +262,7 @@ class Habitica:
254
262
 
255
263
  async def get_user_anonymized(
256
264
  self,
257
- ) -> HabiticaUserAnonymizedrResponse:
265
+ ) -> HabiticaUserAnonymizedResponse:
258
266
  """Get the authenticated user's anonymized profile.
259
267
 
260
268
  This method retrieves the user's profile data while excluding sensitive
@@ -291,7 +299,7 @@ class Habitica:
291
299
 
292
300
  url = url / "anonymized"
293
301
 
294
- return HabiticaUserAnonymizedrResponse.from_json(
302
+ return HabiticaUserAnonymizedResponse.from_json(
295
303
  await self._request("get", url=url),
296
304
  )
297
305
 
@@ -851,7 +859,7 @@ class Habitica:
851
859
  self,
852
860
  skill: Skill,
853
861
  target_id: UUID | None = None,
854
- ) -> HabiticaUserResponse:
862
+ ) -> HabiticaCastSkillResponse:
855
863
  """Cast a skill (spell) in Habitica, optionally targeting a specific user, task or party.
856
864
 
857
865
  Parameters
@@ -864,8 +872,9 @@ class Habitica:
864
872
 
865
873
  Returns
866
874
  -------
867
- HabiticaStatResponse
868
- A response object containing the user's updated stats and a success message.
875
+ HabiticaCastSkillResponse
876
+ A response object containing the user's updated data, updated task data if
877
+ a task was targeted, and a success message.
869
878
 
870
879
  Raises
871
880
  ------
@@ -887,7 +896,7 @@ class Habitica:
887
896
 
888
897
  if target_id:
889
898
  params.update({"targetId": str(target_id)})
890
- return HabiticaUserResponse.from_json(
899
+ return HabiticaCastSkillResponse.from_json(
891
900
  await self._request("post", url=url, params=params),
892
901
  )
893
902
 
@@ -1806,9 +1815,9 @@ class Habitica:
1806
1815
  -------
1807
1816
  None
1808
1817
  """
1809
- url = URL(ASSETS_URL) / f"{asset}"
1810
- if not url.suffix:
1811
- url = url.with_suffix(".png")
1818
+
1819
+ url = URL(ASSETS_URL) / SPECIAL_ASSETS.get(asset, f"{asset}.png")
1820
+
1812
1821
  try:
1813
1822
  if not (asset_data := self._assets_cache.get(asset)):
1814
1823
  async with self._session.get(url) as r:
@@ -1817,18 +1826,20 @@ class Habitica:
1817
1826
  self._cache_asset(asset, asset_data)
1818
1827
  except ClientResponseError as e:
1819
1828
  _LOGGER.exception(
1820
- "Failed to load %s.png due to error [%s]: %s",
1829
+ "Failed to load %s due to error [%s]: %s",
1821
1830
  asset,
1822
1831
  e.status,
1823
1832
  e.message,
1824
1833
  )
1825
1834
  except ClientError:
1826
1835
  _LOGGER.exception(
1827
- "Failed to load %s.png due to a request error",
1836
+ "Failed to load %s due to a request error",
1828
1837
  asset,
1829
1838
  )
1830
1839
  else:
1831
1840
  fetched_image = Image.open(asset_data).convert("RGBA")
1841
+ if offset := SPECIAL_ASSETS_OFFSET.get(asset):
1842
+ position = tuple(map(add, position, offset))
1832
1843
  image.paste(fetched_image, position, fetched_image)
1833
1844
 
1834
1845
  async def generate_avatar( # noqa: PLR0912, PLR0915
@@ -1893,11 +1904,8 @@ class Habitica:
1893
1904
  )
1894
1905
  gear = getattr(gear_set, gear_type)
1895
1906
  if gear and gear != f"{gear_type}_base_0":
1896
- # 2019 Kickstarter gear doesn't follow name conventions
1897
- if special_ks2019 := BACKER_ONLY_GEAR.get(gear):
1898
- gear = special_ks2019
1899
1907
  # armor has slim and broad size options
1900
- elif gear_type == "armor":
1908
+ if gear_type == "armor":
1901
1909
  gear = f"{preferences.size}_{gear}"
1902
1910
  await self.paste_image(image, gear, (24, mount_offset_y))
1903
1911
 
@@ -1926,7 +1934,7 @@ class Habitica:
1926
1934
  ):
1927
1935
  if stats.buffs.spookySparkles:
1928
1936
  await self.paste_image(image, "ghost", (24, mount_offset_y))
1929
- if stats.buffs.shinySeed:
1937
+ if stats.buffs.snowball:
1930
1938
  await self.paste_image(
1931
1939
  image,
1932
1940
  f"avatar_snowball_{stats.Class}",
@@ -1959,7 +1967,7 @@ class Habitica:
1959
1967
  await self.paste_image(
1960
1968
  image,
1961
1969
  f"chair_{preferences.chair}",
1962
- (24, 0),
1970
+ (24, mount_offset_y),
1963
1971
  )
1964
1972
 
1965
1973
  # Fetch and paste the back accessory
@@ -1968,7 +1976,7 @@ class Habitica:
1968
1976
  # Fetch and paste the skin
1969
1977
  await self.paste_image(
1970
1978
  image,
1971
- f"skin_{preferences.skin}{"_sleep" if preferences.sleep else ""}",
1979
+ f"skin_{preferences.skin}{'_sleep' if preferences.sleep else ''}",
1972
1980
  (24, mount_offset_y),
1973
1981
  )
1974
1982
 
habiticalib/types.py CHANGED
@@ -903,6 +903,7 @@ class Task(TypedDict("Task", {"type": NotRequired[TaskType]}), total=True):
903
903
  weeksOfMonth: NotRequired[list[int]]
904
904
  completed: NotRequired[bool]
905
905
  streak: NotRequired[int]
906
+ value: NotRequired[float]
906
907
 
907
908
 
908
909
  @dataclass(kw_only=True)
@@ -914,7 +915,7 @@ class TaskData:
914
915
  Type: TaskType | None = field(default=None, metadata=field_options(alias="type"))
915
916
  text: str | None = None
916
917
  notes: str | None = None
917
- tags: list[UUID] | None = None
918
+ tags: list[UUID] = field(default_factory=list)
918
919
  value: float | None = None
919
920
  priority: TaskPriority | None = None
920
921
  attribute: Attributes | None = None
@@ -929,7 +930,7 @@ class TaskData:
929
930
  counterUp: int | None = None
930
931
  counterDown: int | None = None
931
932
  frequency: Frequency | None = None
932
- history: list[EntryHistory] | None = None
933
+ history: list[EntryHistory] = field(default_factory=list)
933
934
  alias: str | None = None
934
935
  everyX: int | None = None
935
936
  startDate: datetime | None = None
@@ -1059,7 +1060,7 @@ class UserAnonymizedData:
1059
1060
 
1060
1061
 
1061
1062
  @dataclass(kw_only=True)
1062
- class HabiticaUserAnonymizedrResponse(DataClassORJSONMixin):
1063
+ class HabiticaUserAnonymizedResponse(DataClassORJSONMixin):
1063
1064
  """Representation of a anonymized user data export."""
1064
1065
 
1065
1066
  data: UserAnonymizedData
@@ -1656,3 +1657,29 @@ class HabiticaContentResponse(HabiticaResponse):
1656
1657
  """Representation of a content response."""
1657
1658
 
1658
1659
  data: ContentData
1660
+
1661
+
1662
+ @dataclass
1663
+ class PartyMember:
1664
+ """Party member data."""
1665
+
1666
+ stats: StatsUser
1667
+ achievements: AchievementsUser
1668
+ items: ItemsUser
1669
+ profile: ProfileUser
1670
+
1671
+
1672
+ @dataclass
1673
+ class UserTasks:
1674
+ """User and tasks data."""
1675
+
1676
+ user: UserData
1677
+ partyMembers: list[PartyMember] = field(default_factory=list)
1678
+ task: TaskData | None = None
1679
+
1680
+
1681
+ @dataclass
1682
+ class HabiticaCastSkillResponse(HabiticaResponse):
1683
+ """Representation of a cast skill response."""
1684
+
1685
+ data: UserTasks
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: Habiticalib
3
- Version: 0.3.2
3
+ Version: 0.3.4
4
4
  Summary: Asynchronous Python client library for the Habitica API
5
5
  Project-URL: Documentation, https://tr4nt0r.github.io/habiticalib/
6
6
  Project-URL: Source, https://github.com/tr4nt0r/habiticalib
@@ -0,0 +1,12 @@
1
+ habiticalib/__init__.py,sha256=YDYsEGMIxZftYU9YkNC0tyvJrkxn2lXV8WcVQoOXyvA,2415
2
+ habiticalib/const.py,sha256=HP5xKY09-GES72btrXbI0XUFSrCBCqXdn2JPwfOR0vs,2308
3
+ habiticalib/exceptions.py,sha256=oVFCGbHkVn0UpIKIPZPzXfvzs9US4R05ebdEn6cOpqM,1350
4
+ habiticalib/ha.py,sha256=rSzrs7ixLJH3ZtOFlcuKV2t1ZndT0VpuPhDsGqorkOs,20757
5
+ habiticalib/helpers.py,sha256=IRZLYWkDVLI0iVBgBMmvZ6L83KCUl-CnzGhUR_tP6Fg,4576
6
+ habiticalib/lib.py,sha256=YxnlRwCsSlmrdvVu06TKRa6SgnQj2fh2LY8JM7HRqq8,73610
7
+ habiticalib/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
8
+ habiticalib/types.py,sha256=GS3GOlSY31SrJKcuci__drYewhJpi4jfPmAHapPijrg,43359
9
+ habiticalib-0.3.4.dist-info/METADATA,sha256=9DSGPmVo5IqO1hR1Z47oPP1xMiEsiCZpdMT3AGvs448,4207
10
+ habiticalib-0.3.4.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
11
+ habiticalib-0.3.4.dist-info/licenses/LICENSE,sha256=oIinIOSJ49l1iVIRI3XGXFWt6SF7a83kEFBAY8ORwNI,1084
12
+ habiticalib-0.3.4.dist-info/RECORD,,
@@ -1,11 +0,0 @@
1
- habiticalib/__init__.py,sha256=4R9cHRWgKBakoUfouQ8x-5Q1psHnFayj88-exKCLYnE,2327
2
- habiticalib/const.py,sha256=oVV6SvHxkvrLWD609IAcplC1Jjq9qho9DsXkv6VUJag,624
3
- habiticalib/exceptions.py,sha256=oVFCGbHkVn0UpIKIPZPzXfvzs9US4R05ebdEn6cOpqM,1350
4
- habiticalib/helpers.py,sha256=IRZLYWkDVLI0iVBgBMmvZ6L83KCUl-CnzGhUR_tP6Fg,4576
5
- habiticalib/lib.py,sha256=sH0L33VspxC6gVcpsYcryCXlViwwIVTc3zuTZ3V9wnU,73527
6
- habiticalib/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
7
- habiticalib/types.py,sha256=YcMdZsB0jXL2_t7eYuD3-84qEl18arK1gRWBCA5quLQ,42818
8
- habiticalib-0.3.2.dist-info/METADATA,sha256=_tuam7yUeLxcpy5QhmXvRfAz1X2bknmKLTAU83S3Hj8,4207
9
- habiticalib-0.3.2.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
10
- habiticalib-0.3.2.dist-info/licenses/LICENSE,sha256=oIinIOSJ49l1iVIRI3XGXFWt6SF7a83kEFBAY8ORwNI,1084
11
- habiticalib-0.3.2.dist-info/RECORD,,