canonicalwebteam.store-api 6.9.0__tar.gz → 7.0.0__tar.gz
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.
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/PKG-INFO +4 -3
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/store_api/devicegw.py +0 -1
- canonicalwebteam_store_api-7.0.0/canonicalwebteam/stores_web_redis/__init__.py +0 -0
- canonicalwebteam_store_api-7.0.0/canonicalwebteam/stores_web_redis/utility.py +87 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/pyproject.toml +4 -2
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/LICENSE +0 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/README.md +0 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/__init__.py +0 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/exceptions.py +0 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/retry_utils.py +0 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/store_api/__init__.py +0 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/store_api/base.py +0 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/store_api/dashboard.py +0 -0
- {canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/store_api/publishergw.py +0 -0
|
@@ -1,22 +1,23 @@
|
|
|
1
1
|
Metadata-Version: 2.3
|
|
2
2
|
Name: canonicalwebteam.store-api
|
|
3
|
-
Version:
|
|
3
|
+
Version: 7.0.0
|
|
4
4
|
Summary:
|
|
5
5
|
License: LGPL-3.0
|
|
6
6
|
Author: Canonical Web Team
|
|
7
7
|
Author-email: webteam@canonical.com
|
|
8
|
-
Requires-Python: >=3.
|
|
8
|
+
Requires-Python: >=3.9,<4.0
|
|
9
9
|
Classifier: License :: OSI Approved :: GNU Lesser General Public License v3 (LGPLv3)
|
|
10
10
|
Classifier: Programming Language :: Python :: 3
|
|
11
|
-
Classifier: Programming Language :: Python :: 3.8
|
|
12
11
|
Classifier: Programming Language :: Python :: 3.9
|
|
13
12
|
Classifier: Programming Language :: Python :: 3.10
|
|
14
13
|
Classifier: Programming Language :: Python :: 3.11
|
|
15
14
|
Classifier: Programming Language :: Python :: 3.12
|
|
16
15
|
Classifier: Programming Language :: Python :: 3.13
|
|
16
|
+
Requires-Dist: cachetools (>=6.2.0,<7.0.0)
|
|
17
17
|
Requires-Dist: coverage (>=7.0,<8.0)
|
|
18
18
|
Requires-Dist: mypy (>=1.14.1,<2.0.0)
|
|
19
19
|
Requires-Dist: pymacaroons (==0.13.0)
|
|
20
|
+
Requires-Dist: redis (>=6.4.0,<7.0.0)
|
|
20
21
|
Requires-Dist: requests (>=2.32.2,<3.0.0)
|
|
21
22
|
Description-Content-Type: text/markdown
|
|
22
23
|
|
|
@@ -132,7 +132,6 @@ class DeviceGW(Base):
|
|
|
132
132
|
if featured:
|
|
133
133
|
params["featured"] = featured
|
|
134
134
|
response = self.session.get(url, params=params, headers=headers)
|
|
135
|
-
# pprint(response.json())
|
|
136
135
|
return self.process_response(response)
|
|
137
136
|
|
|
138
137
|
def get_all_items(self, size: int, api_version: int = 1) -> dict:
|
|
File without changes
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import os
|
|
2
|
+
from cachetools import TTLCache
|
|
3
|
+
import redis
|
|
4
|
+
import json
|
|
5
|
+
import logging
|
|
6
|
+
from typing import Optional, Any
|
|
7
|
+
|
|
8
|
+
logger = logging.getLogger(__name__)
|
|
9
|
+
|
|
10
|
+
host = os.getenv("REDIS_DB_HOSTNAME", "localhost")
|
|
11
|
+
port = os.getenv("REDIS_DB_PORT", "6379")
|
|
12
|
+
password = os.getenv("REDIS_DB_PASSWORD", None)
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
class RedisCache:
|
|
16
|
+
def __init__(self, namespace: str, maxsize: int, ttl: int = 300):
|
|
17
|
+
self.namespace = namespace
|
|
18
|
+
self.fallback = TTLCache(maxsize=maxsize, ttl=ttl)
|
|
19
|
+
try:
|
|
20
|
+
self.client = redis.Redis(
|
|
21
|
+
host=host,
|
|
22
|
+
port=int(port),
|
|
23
|
+
password=password,
|
|
24
|
+
decode_responses=True,
|
|
25
|
+
)
|
|
26
|
+
self.client.ping()
|
|
27
|
+
self.redis_available = True
|
|
28
|
+
except redis.RedisError as e:
|
|
29
|
+
logger.warning("Redis unavailable: %s", e)
|
|
30
|
+
self.redis_available = False
|
|
31
|
+
|
|
32
|
+
def _build_key(self, key: str) -> str:
|
|
33
|
+
return f"{self.namespace}:{key}"
|
|
34
|
+
|
|
35
|
+
def _serialize(self, value: Any) -> str:
|
|
36
|
+
if isinstance(value, str):
|
|
37
|
+
return value
|
|
38
|
+
try:
|
|
39
|
+
return json.dumps(value)
|
|
40
|
+
except (TypeError, ValueError) as e:
|
|
41
|
+
logger.error("Serialization error: %s", e)
|
|
42
|
+
raise
|
|
43
|
+
|
|
44
|
+
def _deserialize(
|
|
45
|
+
self, value: Optional[str], expected_type: type = str
|
|
46
|
+
) -> Any:
|
|
47
|
+
if value is None:
|
|
48
|
+
return None
|
|
49
|
+
if expected_type is str:
|
|
50
|
+
return value
|
|
51
|
+
try:
|
|
52
|
+
return json.loads(value)
|
|
53
|
+
except (TypeError, ValueError) as e:
|
|
54
|
+
logger.error("Deserialization error: %s", e)
|
|
55
|
+
raise
|
|
56
|
+
|
|
57
|
+
def get(self, key: str, expected_type: type = str) -> Any:
|
|
58
|
+
if self.redis_available:
|
|
59
|
+
full_key = self._build_key(key)
|
|
60
|
+
try:
|
|
61
|
+
value = self.client.get(full_key)
|
|
62
|
+
return self._deserialize(value, expected_type)
|
|
63
|
+
except redis.RedisError as e:
|
|
64
|
+
logger.error("Redis get error: %s", e)
|
|
65
|
+
value = self.fallback.get(key)
|
|
66
|
+
return value
|
|
67
|
+
|
|
68
|
+
def set(self, key: str, value: Any, ttl=300):
|
|
69
|
+
if self.redis_available:
|
|
70
|
+
full_key = self._build_key(key)
|
|
71
|
+
try:
|
|
72
|
+
serialized = self._serialize(value)
|
|
73
|
+
self.client.setex(full_key, ttl, serialized)
|
|
74
|
+
return
|
|
75
|
+
except redis.RedisError as e:
|
|
76
|
+
logger.error("Redis set error: %s", e)
|
|
77
|
+
self.fallback[key] = value
|
|
78
|
+
|
|
79
|
+
def delete(self, key: str):
|
|
80
|
+
if self.redis_available:
|
|
81
|
+
full_key = self._build_key(key)
|
|
82
|
+
try:
|
|
83
|
+
self.client.delete(full_key)
|
|
84
|
+
except redis.RedisError as e:
|
|
85
|
+
logger.error("Redis delete error: %s", e)
|
|
86
|
+
else:
|
|
87
|
+
self.fallback.pop(key, None)
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
[tool.poetry]
|
|
2
2
|
name = 'canonicalwebteam.store-api'
|
|
3
|
-
version = '
|
|
3
|
+
version = '7.0.0'
|
|
4
4
|
description = ''
|
|
5
5
|
authors = ['Canonical Web Team <webteam@canonical.com>']
|
|
6
6
|
license = 'LGPL-3.0'
|
|
@@ -10,11 +10,13 @@ readme = 'README.md'
|
|
|
10
10
|
include = 'canonicalwebteam'
|
|
11
11
|
|
|
12
12
|
[tool.poetry.dependencies]
|
|
13
|
-
python = '^3.
|
|
13
|
+
python = '^3.9'
|
|
14
14
|
coverage = '^7.0'
|
|
15
15
|
pymacaroons = '0.13.0'
|
|
16
16
|
requests = '^2.32.2'
|
|
17
17
|
mypy = "^1.14.1"
|
|
18
|
+
redis = "^6.4.0"
|
|
19
|
+
cachetools = "^6.2.0"
|
|
18
20
|
|
|
19
21
|
[tool.poetry.dev-dependencies]
|
|
20
22
|
vcrpy-unittest = '^0.1.7'
|
|
File without changes
|
|
File without changes
|
{canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/__init__.py
RENAMED
|
File without changes
|
{canonicalwebteam_store_api-6.9.0 → canonicalwebteam_store_api-7.0.0}/canonicalwebteam/exceptions.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|