lumera 0.9.6__tar.gz → 0.9.8__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.
- {lumera-0.9.6 → lumera-0.9.8}/PKG-INFO +1 -1
- {lumera-0.9.6 → lumera-0.9.8}/lumera/_utils.py +38 -1
- {lumera-0.9.6 → lumera-0.9.8}/lumera/storage.py +82 -1
- {lumera-0.9.6 → lumera-0.9.8}/lumera.egg-info/PKG-INFO +1 -1
- {lumera-0.9.6 → lumera-0.9.8}/pyproject.toml +1 -1
- {lumera-0.9.6 → lumera-0.9.8}/lumera/__init__.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/automations.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/exceptions.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/files.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/google.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/integrations/__init__.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/integrations/google.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/llm.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/locks.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/pb.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/sdk.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera/webhooks.py +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera.egg-info/SOURCES.txt +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera.egg-info/dependency_links.txt +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera.egg-info/requires.txt +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/lumera.egg-info/top_level.txt +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/setup.cfg +0 -0
- {lumera-0.9.6 → lumera-0.9.8}/tests/test_sdk.py +0 -0
|
@@ -19,6 +19,8 @@ from dotenv import load_dotenv
|
|
|
19
19
|
TOKEN_ENV = "LUMERA_TOKEN"
|
|
20
20
|
BASE_URL_ENV = "LUMERA_BASE_URL"
|
|
21
21
|
ENV_PATH = "/root/.env"
|
|
22
|
+
LOCAL_CREDS_PATH = ".lumera/credentials.json"
|
|
23
|
+
GLOBAL_CREDS_PATH = os.path.join(os.path.expanduser("~"), ".config", "lumera", "credentials.json")
|
|
22
24
|
|
|
23
25
|
load_dotenv(override=False)
|
|
24
26
|
load_dotenv(ENV_PATH, override=False)
|
|
@@ -56,12 +58,47 @@ def _get_session() -> requests.Session:
|
|
|
56
58
|
return _http_session
|
|
57
59
|
|
|
58
60
|
|
|
61
|
+
def _read_token_from_creds_file(path: str) -> str | None:
|
|
62
|
+
"""Read token from a credentials.json file."""
|
|
63
|
+
try:
|
|
64
|
+
if os.path.exists(path):
|
|
65
|
+
with open(path, "r") as f:
|
|
66
|
+
creds = json.load(f)
|
|
67
|
+
return creds.get("token")
|
|
68
|
+
except (json.JSONDecodeError, IOError):
|
|
69
|
+
pass
|
|
70
|
+
return None
|
|
71
|
+
|
|
72
|
+
|
|
59
73
|
def get_lumera_token() -> str:
|
|
74
|
+
"""Get Lumera API token from environment or credentials files.
|
|
75
|
+
|
|
76
|
+
Priority:
|
|
77
|
+
1. LUMERA_TOKEN environment variable
|
|
78
|
+
2. .lumera/credentials.json (project-local, from CLI login --local)
|
|
79
|
+
3. ~/.config/lumera/credentials.json (global, from CLI login)
|
|
80
|
+
4. /root/.env (legacy, for automations in sandbox)
|
|
81
|
+
"""
|
|
82
|
+
# 1. Check environment variable (highest priority)
|
|
60
83
|
token = os.getenv(TOKEN_ENV)
|
|
61
84
|
if token:
|
|
62
85
|
return token
|
|
86
|
+
|
|
87
|
+
# 2. Check project-local credentials
|
|
88
|
+
token = _read_token_from_creds_file(LOCAL_CREDS_PATH)
|
|
89
|
+
if token:
|
|
90
|
+
return token
|
|
91
|
+
|
|
92
|
+
# 3. Check global credentials
|
|
93
|
+
token = _read_token_from_creds_file(GLOBAL_CREDS_PATH)
|
|
94
|
+
if token:
|
|
95
|
+
return token
|
|
96
|
+
|
|
97
|
+
# 4. /root/.env is already loaded via dotenv at module load time
|
|
98
|
+
# so if we get here, no token was found anywhere
|
|
63
99
|
raise RuntimeError(
|
|
64
|
-
f"{TOKEN_ENV}
|
|
100
|
+
f"{TOKEN_ENV} not found. Checked: environment, {LOCAL_CREDS_PATH}, {GLOBAL_CREDS_PATH}, {ENV_PATH}. "
|
|
101
|
+
f"Run `lumera login` to authenticate."
|
|
65
102
|
)
|
|
66
103
|
|
|
67
104
|
|
|
@@ -23,7 +23,15 @@ Example:
|
|
|
23
23
|
>>> print(result["url"])
|
|
24
24
|
"""
|
|
25
25
|
|
|
26
|
-
__all__ = [
|
|
26
|
+
__all__ = [
|
|
27
|
+
"upload",
|
|
28
|
+
"upload_file",
|
|
29
|
+
"download_url",
|
|
30
|
+
"list_files",
|
|
31
|
+
"UploadResult",
|
|
32
|
+
"get_download_url",
|
|
33
|
+
"download",
|
|
34
|
+
]
|
|
27
35
|
|
|
28
36
|
import mimetypes
|
|
29
37
|
import os
|
|
@@ -268,3 +276,76 @@ def list_files(prefix: str | None = None) -> list[dict[str, Any]]:
|
|
|
268
276
|
files = [f for f in files if f.get("name", "").startswith(prefix)]
|
|
269
277
|
|
|
270
278
|
return files
|
|
279
|
+
|
|
280
|
+
|
|
281
|
+
def get_download_url(object_key: str) -> str:
|
|
282
|
+
"""Get a presigned download URL for a file by its object_key.
|
|
283
|
+
|
|
284
|
+
Use this to get download URLs for files stored in lumera_file fields on records.
|
|
285
|
+
The object_key is found in the file descriptor (e.g., record["file"]["object_key"]).
|
|
286
|
+
|
|
287
|
+
Args:
|
|
288
|
+
object_key: The storage object key from a lumera_file field descriptor
|
|
289
|
+
|
|
290
|
+
Returns:
|
|
291
|
+
Presigned download URL (valid for ~15 minutes)
|
|
292
|
+
|
|
293
|
+
Raises:
|
|
294
|
+
ValueError: If object_key is empty
|
|
295
|
+
requests.HTTPError: If the file doesn't exist or request fails
|
|
296
|
+
|
|
297
|
+
Example:
|
|
298
|
+
>>> record = pb.get("documents", "rec_123")
|
|
299
|
+
>>> url = storage.get_download_url(record["file"]["object_key"])
|
|
300
|
+
>>> # Use url to download the file
|
|
301
|
+
"""
|
|
302
|
+
if not object_key or not object_key.strip():
|
|
303
|
+
raise ValueError("object_key is required and cannot be empty")
|
|
304
|
+
|
|
305
|
+
token = get_lumera_token()
|
|
306
|
+
headers = {"Authorization": f"token {token}", "Content-Type": "application/json"}
|
|
307
|
+
|
|
308
|
+
resp = requests.post(
|
|
309
|
+
f"{API_BASE}/pb/uploads/download",
|
|
310
|
+
json={"object_key": object_key.strip()},
|
|
311
|
+
headers=headers,
|
|
312
|
+
timeout=30,
|
|
313
|
+
)
|
|
314
|
+
resp.raise_for_status()
|
|
315
|
+
|
|
316
|
+
data = resp.json()
|
|
317
|
+
url = data.get("download_url")
|
|
318
|
+
if not url:
|
|
319
|
+
raise RuntimeError("uploads/download response missing download_url")
|
|
320
|
+
|
|
321
|
+
return url
|
|
322
|
+
|
|
323
|
+
|
|
324
|
+
def download(object_key: str) -> bytes:
|
|
325
|
+
"""Download file content by its object_key.
|
|
326
|
+
|
|
327
|
+
Use this to download files stored in lumera_file fields on records.
|
|
328
|
+
The object_key is found in the file descriptor (e.g., record["file"]["object_key"]).
|
|
329
|
+
|
|
330
|
+
Args:
|
|
331
|
+
object_key: The storage object key from a lumera_file field descriptor
|
|
332
|
+
|
|
333
|
+
Returns:
|
|
334
|
+
File content as bytes
|
|
335
|
+
|
|
336
|
+
Raises:
|
|
337
|
+
ValueError: If object_key is empty
|
|
338
|
+
requests.HTTPError: If the file doesn't exist or download fails
|
|
339
|
+
|
|
340
|
+
Example:
|
|
341
|
+
>>> record = pb.get("documents", "rec_123")
|
|
342
|
+
>>> content = storage.download(record["file"]["object_key"])
|
|
343
|
+
>>> with open("local_copy.pdf", "wb") as f:
|
|
344
|
+
... f.write(content)
|
|
345
|
+
"""
|
|
346
|
+
url = get_download_url(object_key)
|
|
347
|
+
|
|
348
|
+
response = requests.get(url, timeout=300)
|
|
349
|
+
response.raise_for_status()
|
|
350
|
+
|
|
351
|
+
return response.content
|
|
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
|