apexdevkit 1.8.4__tar.gz → 1.8.6__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.
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/PKG-INFO +1 -1
- apexdevkit-1.8.6/apexdevkit/repository/sqlite.py +101 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/pyproject.toml +1 -1
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/LICENSE +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/README.md +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/__init__.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/annotation/__init__.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/annotation/deprecate.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/error.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/__init__.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/builder.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/dependable.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/docs.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/resource.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/response.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/router.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/schema.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/fastapi/service.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/formatter.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/http/__init__.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/http/fake.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/http/fluent.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/http/httpx.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/http/json.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/http/url.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/py.typed +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/repository/__init__.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/repository/base.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/repository/connector.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/repository/database.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/repository/in_memory.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/repository/interface.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/testing/__init__.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/testing/database.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/testing/fake.py +0 -0
- {apexdevkit-1.8.4 → apexdevkit-1.8.6}/apexdevkit/testing/rest.py +0 -0
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
from dataclasses import dataclass
|
|
4
|
+
from sqlite3 import IntegrityError
|
|
5
|
+
from typing import Any, Callable, Generic, Iterator, TypeVar
|
|
6
|
+
|
|
7
|
+
from apexdevkit.error import DoesNotExistError, ExistsError
|
|
8
|
+
from apexdevkit.repository import Database, DatabaseCommand
|
|
9
|
+
|
|
10
|
+
ItemT = TypeVar("ItemT")
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
@dataclass
|
|
14
|
+
class SqliteRepository(Generic[ItemT]):
|
|
15
|
+
db: Database
|
|
16
|
+
table: SqlTable[ItemT]
|
|
17
|
+
duplicate_criteria: Callable[[ItemT], str] = lambda i: "Unknown"
|
|
18
|
+
|
|
19
|
+
def __iter__(self) -> Iterator[ItemT]:
|
|
20
|
+
for raw in self.db.execute(self.table.select_all()).fetch_all():
|
|
21
|
+
yield self.table.load(raw)
|
|
22
|
+
|
|
23
|
+
def __len__(self) -> int:
|
|
24
|
+
raw = self.db.execute(self.table.count_all()).fetch_one()
|
|
25
|
+
|
|
26
|
+
try:
|
|
27
|
+
return int(raw["n_items"])
|
|
28
|
+
except KeyError:
|
|
29
|
+
raise UnknownError(raw)
|
|
30
|
+
|
|
31
|
+
def create(self, item: ItemT) -> ItemT:
|
|
32
|
+
try:
|
|
33
|
+
return self.table.load(self.db.execute(self.table.insert(item)).fetch_one())
|
|
34
|
+
except IntegrityError: # pragma: no cover
|
|
35
|
+
ExistsError(item).with_duplicate(self.duplicate_criteria).fire()
|
|
36
|
+
return item
|
|
37
|
+
|
|
38
|
+
def create_many(self, items: list[ItemT]) -> list[ItemT]:
|
|
39
|
+
result = []
|
|
40
|
+
for item in items:
|
|
41
|
+
try:
|
|
42
|
+
result.append(
|
|
43
|
+
self.table.load(
|
|
44
|
+
self.db.execute(self.table.insert(item)).fetch_one()
|
|
45
|
+
)
|
|
46
|
+
)
|
|
47
|
+
except IntegrityError: # pragma: no cover
|
|
48
|
+
ExistsError(item).with_duplicate(self.duplicate_criteria).fire()
|
|
49
|
+
return result
|
|
50
|
+
|
|
51
|
+
def read(self, item_id: str) -> ItemT:
|
|
52
|
+
raw = self.db.execute(self.table.select(item_id)).fetch_one()
|
|
53
|
+
|
|
54
|
+
if not raw:
|
|
55
|
+
raise DoesNotExistError(item_id)
|
|
56
|
+
|
|
57
|
+
return self.table.load(raw)
|
|
58
|
+
|
|
59
|
+
def update(self, item: ItemT) -> None:
|
|
60
|
+
self.db.execute(self.table.update(item)).fetch_none()
|
|
61
|
+
|
|
62
|
+
def update_many(self, items: list[ItemT]) -> None:
|
|
63
|
+
for item in items:
|
|
64
|
+
self.update(item)
|
|
65
|
+
|
|
66
|
+
def delete(self, item_id: str) -> None:
|
|
67
|
+
self.db.execute(self.table.delete(item_id)).fetch_none()
|
|
68
|
+
|
|
69
|
+
def delete_all(self) -> None:
|
|
70
|
+
self.db.execute(self.table.delete_all()).fetch_none()
|
|
71
|
+
|
|
72
|
+
|
|
73
|
+
class SqlTable(Generic[ItemT]): # pragma: no cover
|
|
74
|
+
def count_all(self) -> DatabaseCommand:
|
|
75
|
+
raise NotImplementedError("Not implemented")
|
|
76
|
+
|
|
77
|
+
def insert(self, item: ItemT) -> DatabaseCommand:
|
|
78
|
+
raise NotImplementedError("Not implemented")
|
|
79
|
+
|
|
80
|
+
def select(self, item_id: str) -> DatabaseCommand:
|
|
81
|
+
raise NotImplementedError("Not implemented")
|
|
82
|
+
|
|
83
|
+
def select_all(self) -> DatabaseCommand:
|
|
84
|
+
raise NotImplementedError("Not implemented")
|
|
85
|
+
|
|
86
|
+
def update(self, item: ItemT) -> DatabaseCommand:
|
|
87
|
+
raise NotImplementedError("Not implemented")
|
|
88
|
+
|
|
89
|
+
def delete(self, item_id: str) -> DatabaseCommand:
|
|
90
|
+
raise NotImplementedError("Not implemented")
|
|
91
|
+
|
|
92
|
+
def delete_all(self) -> DatabaseCommand:
|
|
93
|
+
raise NotImplementedError("Not implemented")
|
|
94
|
+
|
|
95
|
+
def load(self, data: dict[str, Any]) -> ItemT:
|
|
96
|
+
raise NotImplementedError("Not implemented")
|
|
97
|
+
|
|
98
|
+
|
|
99
|
+
@dataclass
|
|
100
|
+
class UnknownError(Exception):
|
|
101
|
+
raw: dict[str, Any]
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|