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.
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: lumera
3
- Version: 0.9.6
3
+ Version: 0.9.8
4
4
  Summary: SDK for building on Lumera platform
5
5
  Requires-Python: >=3.11
6
6
  Requires-Dist: requests
@@ -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} environment variable not set (checked environment and {ENV_PATH})"
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__ = ["upload", "upload_file", "download_url", "list_files", "UploadResult"]
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
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: lumera
3
- Version: 0.9.6
3
+ Version: 0.9.8
4
4
  Summary: SDK for building on Lumera platform
5
5
  Requires-Python: >=3.11
6
6
  Requires-Dist: requests
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "lumera"
3
- version = "0.9.6"
3
+ version = "0.9.8"
4
4
  description = "SDK for building on Lumera platform"
5
5
  requires-python = ">=3.11"
6
6
  dependencies = [
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