kgmodule-utils 0.4.0__tar.gz → 0.4.2__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.
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/PKG-INFO +53 -13
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/README.md +52 -12
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/pyproject.toml +1 -1
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/__init__.py +3 -1
- kgmodule_utils-0.4.2/src/kg_utils/retrieval/__init__.py +5 -0
- kgmodule_utils-0.4.2/src/kg_utils/retrieval/hits.py +75 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/synthesis/__init__.py +8 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/synthesis/_config.py +2 -2
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/synthesis/_image.py +27 -4
- kgmodule_utils-0.4.2/src/kg_utils/synthesis/factory.py +97 -0
- kgmodule_utils-0.4.2/src/kg_utils/worker/__init__.py +17 -0
- kgmodule_utils-0.4.2/src/kg_utils/worker/client.py +196 -0
- kgmodule_utils-0.4.2/src/kg_utils/worker/ops.py +72 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/LICENSE +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/embed.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/embedder.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/extractor.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/module.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/pipeline.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/py.typed +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/semantic.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/snapshots/__init__.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/snapshots/manager.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/snapshots/models.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/specs.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/store.py +0 -0
- {kgmodule_utils-0.4.0 → kgmodule_utils-0.4.2}/src/kg_utils/synthesis/_text.py +0 -0
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.4
|
|
2
2
|
Name: kgmodule-utils
|
|
3
|
-
Version: 0.4.
|
|
3
|
+
Version: 0.4.2
|
|
4
4
|
Summary: Shared types, graph store, semantic index, and pipeline base for the KGModule SDK
|
|
5
5
|
License: Elastic-2.0
|
|
6
6
|
License-File: LICENSE
|
|
@@ -35,7 +35,7 @@ Description-Content-Type: text/markdown
|
|
|
35
35
|
|
|
36
36
|
[](https://www.python.org/)
|
|
37
37
|
[](https://www.elastic.co/licensing/elastic-license)
|
|
38
|
-
[](https://github.com/Flux-Frontiers/KG_utils/releases)
|
|
39
39
|
[](https://github.com/Flux-Frontiers/KG_utils/actions/workflows/ci.yml)
|
|
40
40
|
[](https://python-poetry.org/)
|
|
41
41
|
|
|
@@ -67,6 +67,7 @@ Every KGModule implementation — [PyCodeKG](https://github.com/Flux-Frontiers/p
|
|
|
67
67
|
- **`kg_utils.embedder`** — `get_embedder()`, `wrap_embedder()`, `load_sentence_transformer()` factory functions
|
|
68
68
|
- **`kg_utils.embed`** — `Embedder` protocol, `DEFAULT_MODEL`, `KNOWN_MODELS`, `resolve_model_path()`
|
|
69
69
|
- **`kg_utils.snapshots`** — `Snapshot`, `SnapshotManager`, `SnapshotManifest` for temporal metric tracking
|
|
70
|
+
- **`kg_utils.synthesis`** — Unified text + image synthesis: oMLX, Ollama, and OpenAI text backends; mflux-local, mflux-serve, and DALL-E image backends; all env-var configurable
|
|
70
71
|
|
|
71
72
|
---
|
|
72
73
|
|
|
@@ -86,11 +87,23 @@ pip install kgmodule-utils
|
|
|
86
87
|
pip install 'kgmodule-utils[semantic]'
|
|
87
88
|
```
|
|
88
89
|
|
|
90
|
+
### With text + image synthesis (oMLX / Ollama / OpenAI / mflux-serve)
|
|
91
|
+
|
|
92
|
+
```bash
|
|
93
|
+
pip install 'kgmodule-utils[synthesis]'
|
|
94
|
+
```
|
|
95
|
+
|
|
96
|
+
### With local mflux image generation (Apple Silicon, includes synthesis)
|
|
97
|
+
|
|
98
|
+
```bash
|
|
99
|
+
pip install 'kgmodule-utils[synthesis-mflux]'
|
|
100
|
+
```
|
|
101
|
+
|
|
89
102
|
### In a Poetry project
|
|
90
103
|
|
|
91
104
|
```toml
|
|
92
105
|
[tool.poetry.dependencies]
|
|
93
|
-
kgmodule-utils = { version = ">=0.
|
|
106
|
+
kgmodule-utils = { version = ">=0.4.0", extras = ["semantic", "synthesis"] }
|
|
94
107
|
```
|
|
95
108
|
|
|
96
109
|
---
|
|
@@ -217,6 +230,23 @@ delta = mgr.diff_snapshots(snaps[-1]["key"], snaps[0]["key"])
|
|
|
217
230
|
| `SnapshotManager` | Capture, persist, load, list, diff, and prune snapshots |
|
|
218
231
|
| `SnapshotManifest` | Fast-lookup index with format versioning |
|
|
219
232
|
|
|
233
|
+
### `kg_utils.synthesis`
|
|
234
|
+
|
|
235
|
+
> Full reference: [docs/synthesis.md](docs/synthesis.md)
|
|
236
|
+
|
|
237
|
+
| Class / function | Description |
|
|
238
|
+
|---|---|
|
|
239
|
+
| `TextBackend` | Enum: `omlx` \| `ollama` \| `openai` |
|
|
240
|
+
| `ImageBackend` | Enum: `mflux-local` \| `mflux-serve` \| `openai` |
|
|
241
|
+
| `TextConfig` | Backend config dataclass with `resolved_endpoint()` / `resolved_model()` |
|
|
242
|
+
| `ImageConfig` | Backend config dataclass with `resolved_server_url()` / `resolved_model()` |
|
|
243
|
+
| `TextSynthesizer` | `list_models()`, `synthesize_rag()`, `rewrite_for_image()` |
|
|
244
|
+
| `ImageSynthesizer` | `generate()` → PIL Image, `generate_b64()` → base64 PNG |
|
|
245
|
+
| `text_config_from_env()` | Build `TextConfig` from `SYNTH_*` env vars |
|
|
246
|
+
| `image_config_from_env()` | Build `ImageConfig` from `IMAGE_*` env vars |
|
|
247
|
+
| `text_synthesizer_from_env()` | Convenience: config + synthesizer in one call |
|
|
248
|
+
| `image_synthesizer_from_env()` | Convenience: config + synthesizer in one call |
|
|
249
|
+
|
|
220
250
|
---
|
|
221
251
|
|
|
222
252
|
## Project Structure
|
|
@@ -224,6 +254,8 @@ delta = mgr.diff_snapshots(snaps[-1]["key"], snaps[0]["key"])
|
|
|
224
254
|
```
|
|
225
255
|
KG_utils/
|
|
226
256
|
├── pyproject.toml
|
|
257
|
+
├── docs/
|
|
258
|
+
│ └── synthesis.md # Synthesis sub-package reference
|
|
227
259
|
├── src/
|
|
228
260
|
│ └── kg_utils/
|
|
229
261
|
│ ├── __init__.py
|
|
@@ -235,17 +267,25 @@ KG_utils/
|
|
|
235
267
|
│ ├── module.py # Re-export shim
|
|
236
268
|
│ ├── embed.py # Embedder protocol, model registry
|
|
237
269
|
│ ├── embedder.py # SentenceTransformerEmbedder factory functions
|
|
238
|
-
│
|
|
239
|
-
│
|
|
240
|
-
│
|
|
241
|
-
│
|
|
270
|
+
│ ├── snapshots/
|
|
271
|
+
│ │ ├── __init__.py
|
|
272
|
+
│ │ ├── models.py # Snapshot, SnapshotManifest, PruneResult
|
|
273
|
+
│ │ └── manager.py # SnapshotManager
|
|
274
|
+
│ └── synthesis/
|
|
275
|
+
│ ├── __init__.py # Public API + factory functions
|
|
276
|
+
│ ├── _config.py # TextBackend, ImageBackend, TextConfig, ImageConfig, env factories
|
|
277
|
+
│ ├── _text.py # TextSynthesizer
|
|
278
|
+
│ └── _image.py # ImageSynthesizer
|
|
242
279
|
└── tests/
|
|
243
|
-
├── test_store.py
|
|
244
|
-
├── test_pipeline_utils.py
|
|
245
|
-
├── test_pipeline_module.py
|
|
246
|
-
├── test_types.py
|
|
247
|
-
├── test_snapshots.py
|
|
248
|
-
|
|
280
|
+
├── test_store.py # GraphStore unit tests
|
|
281
|
+
├── test_pipeline_utils.py # Pipeline utility function tests
|
|
282
|
+
├── test_pipeline_module.py # End-to-end integration tests (--integration)
|
|
283
|
+
├── test_types.py # Spec dataclass and KGExtractor tests
|
|
284
|
+
├── test_snapshots.py # Snapshot lifecycle tests
|
|
285
|
+
├── test_integration.py # Cross-module integration tests
|
|
286
|
+
├── test_synthesis_config.py # Config defaults and env-var priority chains (44 tests)
|
|
287
|
+
├── test_synthesis_text.py # TextSynthesizer with mocked openai client (38 tests)
|
|
288
|
+
└── test_synthesis_image.py # ImageSynthesizer with mocked backends (34 tests)
|
|
249
289
|
```
|
|
250
290
|
|
|
251
291
|
---
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
|
|
2
2
|
[](https://www.python.org/)
|
|
3
3
|
[](https://www.elastic.co/licensing/elastic-license)
|
|
4
|
-
[](https://github.com/Flux-Frontiers/KG_utils/releases)
|
|
5
5
|
[](https://github.com/Flux-Frontiers/KG_utils/actions/workflows/ci.yml)
|
|
6
6
|
[](https://python-poetry.org/)
|
|
7
7
|
|
|
@@ -33,6 +33,7 @@ Every KGModule implementation — [PyCodeKG](https://github.com/Flux-Frontiers/p
|
|
|
33
33
|
- **`kg_utils.embedder`** — `get_embedder()`, `wrap_embedder()`, `load_sentence_transformer()` factory functions
|
|
34
34
|
- **`kg_utils.embed`** — `Embedder` protocol, `DEFAULT_MODEL`, `KNOWN_MODELS`, `resolve_model_path()`
|
|
35
35
|
- **`kg_utils.snapshots`** — `Snapshot`, `SnapshotManager`, `SnapshotManifest` for temporal metric tracking
|
|
36
|
+
- **`kg_utils.synthesis`** — Unified text + image synthesis: oMLX, Ollama, and OpenAI text backends; mflux-local, mflux-serve, and DALL-E image backends; all env-var configurable
|
|
36
37
|
|
|
37
38
|
---
|
|
38
39
|
|
|
@@ -52,11 +53,23 @@ pip install kgmodule-utils
|
|
|
52
53
|
pip install 'kgmodule-utils[semantic]'
|
|
53
54
|
```
|
|
54
55
|
|
|
56
|
+
### With text + image synthesis (oMLX / Ollama / OpenAI / mflux-serve)
|
|
57
|
+
|
|
58
|
+
```bash
|
|
59
|
+
pip install 'kgmodule-utils[synthesis]'
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
### With local mflux image generation (Apple Silicon, includes synthesis)
|
|
63
|
+
|
|
64
|
+
```bash
|
|
65
|
+
pip install 'kgmodule-utils[synthesis-mflux]'
|
|
66
|
+
```
|
|
67
|
+
|
|
55
68
|
### In a Poetry project
|
|
56
69
|
|
|
57
70
|
```toml
|
|
58
71
|
[tool.poetry.dependencies]
|
|
59
|
-
kgmodule-utils = { version = ">=0.
|
|
72
|
+
kgmodule-utils = { version = ">=0.4.0", extras = ["semantic", "synthesis"] }
|
|
60
73
|
```
|
|
61
74
|
|
|
62
75
|
---
|
|
@@ -183,6 +196,23 @@ delta = mgr.diff_snapshots(snaps[-1]["key"], snaps[0]["key"])
|
|
|
183
196
|
| `SnapshotManager` | Capture, persist, load, list, diff, and prune snapshots |
|
|
184
197
|
| `SnapshotManifest` | Fast-lookup index with format versioning |
|
|
185
198
|
|
|
199
|
+
### `kg_utils.synthesis`
|
|
200
|
+
|
|
201
|
+
> Full reference: [docs/synthesis.md](docs/synthesis.md)
|
|
202
|
+
|
|
203
|
+
| Class / function | Description |
|
|
204
|
+
|---|---|
|
|
205
|
+
| `TextBackend` | Enum: `omlx` \| `ollama` \| `openai` |
|
|
206
|
+
| `ImageBackend` | Enum: `mflux-local` \| `mflux-serve` \| `openai` |
|
|
207
|
+
| `TextConfig` | Backend config dataclass with `resolved_endpoint()` / `resolved_model()` |
|
|
208
|
+
| `ImageConfig` | Backend config dataclass with `resolved_server_url()` / `resolved_model()` |
|
|
209
|
+
| `TextSynthesizer` | `list_models()`, `synthesize_rag()`, `rewrite_for_image()` |
|
|
210
|
+
| `ImageSynthesizer` | `generate()` → PIL Image, `generate_b64()` → base64 PNG |
|
|
211
|
+
| `text_config_from_env()` | Build `TextConfig` from `SYNTH_*` env vars |
|
|
212
|
+
| `image_config_from_env()` | Build `ImageConfig` from `IMAGE_*` env vars |
|
|
213
|
+
| `text_synthesizer_from_env()` | Convenience: config + synthesizer in one call |
|
|
214
|
+
| `image_synthesizer_from_env()` | Convenience: config + synthesizer in one call |
|
|
215
|
+
|
|
186
216
|
---
|
|
187
217
|
|
|
188
218
|
## Project Structure
|
|
@@ -190,6 +220,8 @@ delta = mgr.diff_snapshots(snaps[-1]["key"], snaps[0]["key"])
|
|
|
190
220
|
```
|
|
191
221
|
KG_utils/
|
|
192
222
|
├── pyproject.toml
|
|
223
|
+
├── docs/
|
|
224
|
+
│ └── synthesis.md # Synthesis sub-package reference
|
|
193
225
|
├── src/
|
|
194
226
|
│ └── kg_utils/
|
|
195
227
|
│ ├── __init__.py
|
|
@@ -201,17 +233,25 @@ KG_utils/
|
|
|
201
233
|
│ ├── module.py # Re-export shim
|
|
202
234
|
│ ├── embed.py # Embedder protocol, model registry
|
|
203
235
|
│ ├── embedder.py # SentenceTransformerEmbedder factory functions
|
|
204
|
-
│
|
|
205
|
-
│
|
|
206
|
-
│
|
|
207
|
-
│
|
|
236
|
+
│ ├── snapshots/
|
|
237
|
+
│ │ ├── __init__.py
|
|
238
|
+
│ │ ├── models.py # Snapshot, SnapshotManifest, PruneResult
|
|
239
|
+
│ │ └── manager.py # SnapshotManager
|
|
240
|
+
│ └── synthesis/
|
|
241
|
+
│ ├── __init__.py # Public API + factory functions
|
|
242
|
+
│ ├── _config.py # TextBackend, ImageBackend, TextConfig, ImageConfig, env factories
|
|
243
|
+
│ ├── _text.py # TextSynthesizer
|
|
244
|
+
│ └── _image.py # ImageSynthesizer
|
|
208
245
|
└── tests/
|
|
209
|
-
├── test_store.py
|
|
210
|
-
├── test_pipeline_utils.py
|
|
211
|
-
├── test_pipeline_module.py
|
|
212
|
-
├── test_types.py
|
|
213
|
-
├── test_snapshots.py
|
|
214
|
-
|
|
246
|
+
├── test_store.py # GraphStore unit tests
|
|
247
|
+
├── test_pipeline_utils.py # Pipeline utility function tests
|
|
248
|
+
├── test_pipeline_module.py # End-to-end integration tests (--integration)
|
|
249
|
+
├── test_types.py # Spec dataclass and KGExtractor tests
|
|
250
|
+
├── test_snapshots.py # Snapshot lifecycle tests
|
|
251
|
+
├── test_integration.py # Cross-module integration tests
|
|
252
|
+
├── test_synthesis_config.py # Config defaults and env-var priority chains (44 tests)
|
|
253
|
+
├── test_synthesis_text.py # TextSynthesizer with mocked openai client (38 tests)
|
|
254
|
+
└── test_synthesis_image.py # ImageSynthesizer with mocked backends (34 tests)
|
|
215
255
|
```
|
|
216
256
|
|
|
217
257
|
---
|
|
@@ -10,7 +10,7 @@ build-backend = "poetry.core.masonry.api"
|
|
|
10
10
|
|
|
11
11
|
[project]
|
|
12
12
|
name = "kgmodule-utils"
|
|
13
|
-
version = "0.4.
|
|
13
|
+
version = "0.4.2"
|
|
14
14
|
description = "Shared types, graph store, semantic index, and pipeline base for the KGModule SDK"
|
|
15
15
|
readme = "README.md"
|
|
16
16
|
license = { text = "Elastic-2.0" }
|
|
@@ -14,6 +14,8 @@ Sub-packages / modules:
|
|
|
14
14
|
kg_utils.synthesis — Unified text + image synthesis: TextSynthesizer, ImageSynthesizer.
|
|
15
15
|
Backends: omlx | ollama | openai (text);
|
|
16
16
|
mflux-local | mflux-serve | openai (image).
|
|
17
|
+
kg_utils.worker — RunPod worker protocol helpers and WorkerClient for /runsync calls.
|
|
18
|
+
kg_utils.retrieval — Shared retrieval helpers: hit_to_dict, attach_content_by_sqlite.
|
|
17
19
|
|
|
18
20
|
Optional extras
|
|
19
21
|
---------------
|
|
@@ -22,4 +24,4 @@ Optional extras
|
|
|
22
24
|
pip install 'kgmodule-utils[synthesis-mflux]' # + mflux (Apple Silicon local gen)
|
|
23
25
|
"""
|
|
24
26
|
|
|
25
|
-
__version__ = "0.4.
|
|
27
|
+
__version__ = "0.4.2"
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
# © 2026 Eric G. Suchanek, PhD — Flux-Frontiers · SPDX-License-Identifier: Elastic-2.0
|
|
2
|
+
"""Hit serialization and content hydration helpers for KG retrieval responses."""
|
|
3
|
+
|
|
4
|
+
from __future__ import annotations
|
|
5
|
+
|
|
6
|
+
import sqlite3
|
|
7
|
+
from collections import defaultdict
|
|
8
|
+
from pathlib import Path
|
|
9
|
+
from typing import Any
|
|
10
|
+
|
|
11
|
+
__all__ = ["hit_to_dict", "attach_content_by_sqlite"]
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
def _is_diary_kind(kind_value: Any) -> bool:
|
|
15
|
+
kind_str = str(kind_value)
|
|
16
|
+
return kind_str == "KGKind.DIARY" or kind_str.lower().endswith("diary")
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def hit_to_dict(hit: Any, include_diary_timestamp: bool = False) -> dict:
|
|
20
|
+
"""Serialize a KGRAG hit object into a plain dictionary.
|
|
21
|
+
|
|
22
|
+
:param hit: Hit-like object with standard retrieval attributes.
|
|
23
|
+
:param include_diary_timestamp: Include ``timestamp`` field for diary hits.
|
|
24
|
+
:returns: Serialized hit dictionary.
|
|
25
|
+
"""
|
|
26
|
+
out = {
|
|
27
|
+
"kg_name": hit.kg_name,
|
|
28
|
+
"kg_kind": str(hit.kg_kind),
|
|
29
|
+
"node_id": hit.node_id,
|
|
30
|
+
"name": hit.name,
|
|
31
|
+
"kind": hit.kind,
|
|
32
|
+
"score": round(float(hit.score), 4),
|
|
33
|
+
"summary": hit.summary,
|
|
34
|
+
"source_path": hit.source_path,
|
|
35
|
+
}
|
|
36
|
+
if include_diary_timestamp:
|
|
37
|
+
out["timestamp"] = hit.name if _is_diary_kind(hit.kg_kind) else None
|
|
38
|
+
return out
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
def attach_content_by_sqlite(hits: list[dict], kg_sqlite_map: dict[str, Path]) -> None:
|
|
42
|
+
"""Attach full node text under ``content`` via batched SQLite lookups.
|
|
43
|
+
|
|
44
|
+
Missing or unreadable databases are ignored to preserve permissive behavior.
|
|
45
|
+
|
|
46
|
+
:param hits: Mutable hit dictionaries. Each hit should include ``kg_name`` and ``node_id``.
|
|
47
|
+
:param kg_sqlite_map: Mapping of KG name to sqlite database path.
|
|
48
|
+
"""
|
|
49
|
+
by_kg: dict[str, list[dict]] = defaultdict(list)
|
|
50
|
+
for hit in hits:
|
|
51
|
+
by_kg[hit.get("kg_name", "")].append(hit)
|
|
52
|
+
|
|
53
|
+
for kg_name, kg_hits in by_kg.items():
|
|
54
|
+
db_path = kg_sqlite_map.get(kg_name)
|
|
55
|
+
if not db_path or not Path(db_path).exists():
|
|
56
|
+
continue
|
|
57
|
+
|
|
58
|
+
ids = [h.get("node_id") for h in kg_hits if h.get("node_id")]
|
|
59
|
+
if not ids:
|
|
60
|
+
continue
|
|
61
|
+
|
|
62
|
+
text_by_id: dict[str, str] = {}
|
|
63
|
+
try:
|
|
64
|
+
with sqlite3.connect(str(db_path)) as con:
|
|
65
|
+
placeholders = ",".join("?" * len(ids))
|
|
66
|
+
query = f"SELECT id, text FROM nodes WHERE id IN ({placeholders})"
|
|
67
|
+
for node_id, text in con.execute(query, ids):
|
|
68
|
+
text_by_id[node_id] = text or ""
|
|
69
|
+
except Exception: # noqa: BLE001 # pylint: disable=broad-exception-caught
|
|
70
|
+
continue
|
|
71
|
+
|
|
72
|
+
for hit in kg_hits:
|
|
73
|
+
node_id = hit.get("node_id")
|
|
74
|
+
if node_id:
|
|
75
|
+
hit["content"] = text_by_id.get(node_id, "")
|
|
@@ -53,6 +53,11 @@ from kg_utils.synthesis._config import (
|
|
|
53
53
|
)
|
|
54
54
|
from kg_utils.synthesis._image import ImageSynthesizer
|
|
55
55
|
from kg_utils.synthesis._text import TextSynthesizer
|
|
56
|
+
from kg_utils.synthesis.factory import (
|
|
57
|
+
image_synth_for_backend,
|
|
58
|
+
normalize_openai_base_url,
|
|
59
|
+
text_synth_for_backend,
|
|
60
|
+
)
|
|
56
61
|
|
|
57
62
|
|
|
58
63
|
def text_synthesizer_from_env() -> TextSynthesizer:
|
|
@@ -76,4 +81,7 @@ __all__ = [
|
|
|
76
81
|
"image_synthesizer_from_env",
|
|
77
82
|
"text_config_from_env",
|
|
78
83
|
"image_config_from_env",
|
|
84
|
+
"normalize_openai_base_url",
|
|
85
|
+
"text_synth_for_backend",
|
|
86
|
+
"image_synth_for_backend",
|
|
79
87
|
]
|
|
@@ -21,7 +21,7 @@ class ImageBackend(str, Enum):
|
|
|
21
21
|
|
|
22
22
|
MFLUX_LOCAL = "mflux-local" # local Flux2Klein via mflux (Apple Silicon only)
|
|
23
23
|
MFLUX_SERVE = "mflux-serve" # HTTP proxy to a running mflux-serve instance
|
|
24
|
-
OPENAI = "openai" # OpenAI
|
|
24
|
+
OPENAI = "openai" # OpenAI gpt-image-1 — requires OPENAI_API_KEY / IMAGE_API_KEY
|
|
25
25
|
|
|
26
26
|
|
|
27
27
|
# Per-backend defaults — filled in when the user has not provided an override.
|
|
@@ -49,7 +49,7 @@ _IMAGE_DEFAULTS: dict[ImageBackend, dict[str, str]] = {
|
|
|
49
49
|
"model": "flux2-klein-4b",
|
|
50
50
|
},
|
|
51
51
|
ImageBackend.OPENAI: {
|
|
52
|
-
"model": "
|
|
52
|
+
"model": "gpt-image-1",
|
|
53
53
|
},
|
|
54
54
|
}
|
|
55
55
|
|
|
@@ -39,6 +39,17 @@ _DALLE3_SIZES: dict[str, str] = {
|
|
|
39
39
|
"3:4": "1024x1792",
|
|
40
40
|
}
|
|
41
41
|
|
|
42
|
+
# gpt-image-1 supports portrait/landscape 1024×1536 instead of 1792-wide.
|
|
43
|
+
_GPT_IMAGE_SIZES: dict[str, str] = {
|
|
44
|
+
"1:1": "1024x1024",
|
|
45
|
+
"3:2": "1536x1024",
|
|
46
|
+
"2:3": "1024x1536",
|
|
47
|
+
"16:9": "1536x1024",
|
|
48
|
+
"9:16": "1024x1536",
|
|
49
|
+
"4:3": "1536x1024",
|
|
50
|
+
"3:4": "1024x1536",
|
|
51
|
+
}
|
|
52
|
+
|
|
42
53
|
|
|
43
54
|
# ---------------------------------------------------------------------------
|
|
44
55
|
# Synthesizer
|
|
@@ -221,7 +232,13 @@ class ImageSynthesizer:
|
|
|
221
232
|
|
|
222
233
|
cfg = self._cfg
|
|
223
234
|
api_key = cfg.api_key or os.environ.get("OPENAI_API_KEY", "")
|
|
224
|
-
|
|
235
|
+
|
|
236
|
+
# gpt-image-1 uses different valid sizes and does not accept response_format.
|
|
237
|
+
# dall-e-3 returns a URL by default; gpt-image-1 returns b64_json by default.
|
|
238
|
+
if model.startswith("gpt-image"):
|
|
239
|
+
size = _GPT_IMAGE_SIZES.get(aspect_ratio, _GPT_IMAGE_SIZES["3:2"])
|
|
240
|
+
else:
|
|
241
|
+
size = _DALLE3_SIZES.get(aspect_ratio, _DALLE3_SIZES["3:2"])
|
|
225
242
|
|
|
226
243
|
client = OpenAI(api_key=api_key)
|
|
227
244
|
resp = client.images.generate(
|
|
@@ -229,7 +246,13 @@ class ImageSynthesizer:
|
|
|
229
246
|
prompt=prompt,
|
|
230
247
|
n=1,
|
|
231
248
|
size=size, # type: ignore[arg-type]
|
|
232
|
-
response_format="b64_json",
|
|
233
249
|
)
|
|
234
|
-
|
|
235
|
-
|
|
250
|
+
item = resp.data[0] if resp.data else None
|
|
251
|
+
if item and item.b64_json:
|
|
252
|
+
return Image.open(BytesIO(base64.b64decode(item.b64_json)))
|
|
253
|
+
if item and item.url:
|
|
254
|
+
import httpx # type: ignore[import-unresolved]
|
|
255
|
+
|
|
256
|
+
raw = httpx.get(item.url, timeout=httpx.Timeout(30.0)).content
|
|
257
|
+
return Image.open(BytesIO(raw))
|
|
258
|
+
raise RuntimeError("OpenAI image response contained neither b64_json nor url")
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
# © 2026 Eric G. Suchanek, PhD — Flux-Frontiers · SPDX-License-Identifier: Elastic-2.0
|
|
2
|
+
"""Synthesis backend factory helpers for per-request backend overrides."""
|
|
3
|
+
|
|
4
|
+
from __future__ import annotations
|
|
5
|
+
|
|
6
|
+
import os
|
|
7
|
+
|
|
8
|
+
from kg_utils.synthesis._config import (
|
|
9
|
+
ImageBackend,
|
|
10
|
+
ImageConfig,
|
|
11
|
+
TextBackend,
|
|
12
|
+
TextConfig,
|
|
13
|
+
)
|
|
14
|
+
from kg_utils.synthesis._image import ImageSynthesizer
|
|
15
|
+
from kg_utils.synthesis._text import TextSynthesizer
|
|
16
|
+
|
|
17
|
+
__all__ = [
|
|
18
|
+
"normalize_openai_base_url",
|
|
19
|
+
"text_synth_for_backend",
|
|
20
|
+
"image_synth_for_backend",
|
|
21
|
+
]
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
def normalize_openai_base_url(endpoint: str) -> str:
|
|
25
|
+
"""Normalize an OpenAI-wire endpoint so it ends with /v1.
|
|
26
|
+
|
|
27
|
+
Returns an empty string when endpoint is empty.
|
|
28
|
+
"""
|
|
29
|
+
ep = (endpoint or "").strip().rstrip("/")
|
|
30
|
+
if not ep:
|
|
31
|
+
return ""
|
|
32
|
+
if ep.endswith("/v1"):
|
|
33
|
+
return ep
|
|
34
|
+
return f"{ep}/v1"
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def text_synth_for_backend(backend: str, fallback: TextSynthesizer) -> TextSynthesizer:
|
|
38
|
+
"""Return a TextSynthesizer configured for a specific backend override.
|
|
39
|
+
|
|
40
|
+
Unknown or empty backend strings return ``fallback``.
|
|
41
|
+
"""
|
|
42
|
+
backend_str = (backend or "").strip().lower()
|
|
43
|
+
if not backend_str:
|
|
44
|
+
return fallback
|
|
45
|
+
|
|
46
|
+
try:
|
|
47
|
+
selected = TextBackend(backend_str)
|
|
48
|
+
except ValueError:
|
|
49
|
+
return fallback
|
|
50
|
+
|
|
51
|
+
if selected == TextBackend.OMLX:
|
|
52
|
+
endpoint = os.environ.get("SYNTH_ENDPOINT") or os.environ.get("VLLM_ENDPOINT_URL") or ""
|
|
53
|
+
endpoint = normalize_openai_base_url(endpoint)
|
|
54
|
+
api_key = os.environ.get("SYNTH_API_KEY") or os.environ.get("VLLM_API_KEY") or ""
|
|
55
|
+
model = os.environ.get("SYNTH_MODEL") or os.environ.get("VLLM_MODEL") or ""
|
|
56
|
+
return TextSynthesizer(
|
|
57
|
+
TextConfig(backend=selected, endpoint=endpoint, api_key=api_key, model=model)
|
|
58
|
+
)
|
|
59
|
+
|
|
60
|
+
if selected == TextBackend.OLLAMA:
|
|
61
|
+
endpoint = os.environ.get("OLLAMA_ENDPOINT") or ""
|
|
62
|
+
return TextSynthesizer(TextConfig(backend=selected, endpoint=endpoint))
|
|
63
|
+
|
|
64
|
+
if selected == TextBackend.OPENAI:
|
|
65
|
+
api_key = os.environ.get("OPENAI_API_KEY") or os.environ.get("SYNTH_API_KEY") or ""
|
|
66
|
+
return TextSynthesizer(TextConfig(backend=selected, api_key=api_key))
|
|
67
|
+
|
|
68
|
+
return fallback
|
|
69
|
+
|
|
70
|
+
|
|
71
|
+
def image_synth_for_backend(backend: str, fallback: ImageSynthesizer) -> ImageSynthesizer:
|
|
72
|
+
"""Return an ImageSynthesizer configured for a specific backend override.
|
|
73
|
+
|
|
74
|
+
Unknown or empty backend strings return ``fallback``.
|
|
75
|
+
"""
|
|
76
|
+
backend_str = (backend or "").strip().lower()
|
|
77
|
+
if not backend_str:
|
|
78
|
+
return fallback
|
|
79
|
+
|
|
80
|
+
try:
|
|
81
|
+
selected = ImageBackend(backend_str)
|
|
82
|
+
except ValueError:
|
|
83
|
+
return fallback
|
|
84
|
+
|
|
85
|
+
if selected == ImageBackend.OPENAI:
|
|
86
|
+
api_key = os.environ.get("OPENAI_API_KEY") or os.environ.get("IMAGE_API_KEY") or ""
|
|
87
|
+
return ImageSynthesizer(ImageConfig(backend=selected, api_key=api_key))
|
|
88
|
+
|
|
89
|
+
if selected == ImageBackend.MFLUX_SERVE:
|
|
90
|
+
server_url = os.environ.get("IMAGE_ENDPOINT") or ""
|
|
91
|
+
return ImageSynthesizer(ImageConfig(backend=selected, server_url=server_url))
|
|
92
|
+
|
|
93
|
+
if selected == ImageBackend.MFLUX_LOCAL:
|
|
94
|
+
model = os.environ.get("IMAGE_MODEL") or os.environ.get("GUTENKG_IMAGE_MODEL") or ""
|
|
95
|
+
return ImageSynthesizer(ImageConfig(backend=selected, model=model))
|
|
96
|
+
|
|
97
|
+
return fallback
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
"""Worker protocol helpers and client for RunPod ``/runsync`` endpoints."""
|
|
2
|
+
|
|
3
|
+
from kg_utils.worker.client import (
|
|
4
|
+
WorkerClient,
|
|
5
|
+
WorkerError,
|
|
6
|
+
decode_worker_response,
|
|
7
|
+
extract_worker_error,
|
|
8
|
+
)
|
|
9
|
+
from kg_utils.worker.ops import handle_aux_ops
|
|
10
|
+
|
|
11
|
+
__all__ = [
|
|
12
|
+
"WorkerClient",
|
|
13
|
+
"WorkerError",
|
|
14
|
+
"decode_worker_response",
|
|
15
|
+
"extract_worker_error",
|
|
16
|
+
"handle_aux_ops",
|
|
17
|
+
]
|
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
# © 2026 Eric G. Suchanek, PhD — Flux-Frontiers · SPDX-License-Identifier: Elastic-2.0
|
|
2
|
+
"""RunPod worker client utilities for chat and handler front-ends.
|
|
3
|
+
|
|
4
|
+
This module centralizes payload construction and response/error decoding for
|
|
5
|
+
``/runsync`` worker calls used by Streamlit clients.
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
from __future__ import annotations
|
|
9
|
+
|
|
10
|
+
import json
|
|
11
|
+
|
|
12
|
+
import httpx
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
class WorkerError(Exception):
|
|
16
|
+
"""Raised when a worker response contains a structured application-level error."""
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def _format_error_data(error_data: object) -> str:
|
|
20
|
+
if isinstance(error_data, str):
|
|
21
|
+
try:
|
|
22
|
+
decoded = json.loads(error_data)
|
|
23
|
+
except (ValueError, TypeError):
|
|
24
|
+
return error_data
|
|
25
|
+
if isinstance(decoded, dict):
|
|
26
|
+
err_type = decoded.get("error_type", "Unknown")
|
|
27
|
+
err_msg = decoded.get("error_message", str(decoded))
|
|
28
|
+
return f"{err_type}: {err_msg}"
|
|
29
|
+
return str(decoded)
|
|
30
|
+
|
|
31
|
+
if isinstance(error_data, dict):
|
|
32
|
+
err_type = error_data.get("error_type", "Unknown")
|
|
33
|
+
err_msg = error_data.get("error_message", str(error_data))
|
|
34
|
+
return f"{err_type}: {err_msg}"
|
|
35
|
+
|
|
36
|
+
return str(error_data)
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
def extract_worker_error(data: object) -> str | None:
|
|
40
|
+
"""Extract a readable worker error from a raw RunPod response payload."""
|
|
41
|
+
if not isinstance(data, dict):
|
|
42
|
+
return str(data)
|
|
43
|
+
|
|
44
|
+
if data.get("status") == "FAILED" or "error_type" in data:
|
|
45
|
+
return _format_error_data(data.get("error", data))
|
|
46
|
+
|
|
47
|
+
out = data.get("output")
|
|
48
|
+
if isinstance(out, dict) and isinstance(out.get("error"), str):
|
|
49
|
+
return out["error"]
|
|
50
|
+
|
|
51
|
+
return None
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
def decode_worker_response(data: object) -> dict:
|
|
55
|
+
"""Decode a worker response payload and raise WorkerError on application errors."""
|
|
56
|
+
error = extract_worker_error(data)
|
|
57
|
+
if error:
|
|
58
|
+
raise WorkerError(error)
|
|
59
|
+
|
|
60
|
+
if not isinstance(data, dict):
|
|
61
|
+
raise WorkerError(f"unexpected worker response type: {type(data).__name__}")
|
|
62
|
+
|
|
63
|
+
out = data.get("output", data)
|
|
64
|
+
if not isinstance(out, dict):
|
|
65
|
+
raise WorkerError(f"unexpected worker output type: {type(out).__name__}")
|
|
66
|
+
return out
|
|
67
|
+
|
|
68
|
+
|
|
69
|
+
class WorkerClient:
|
|
70
|
+
"""Small client for RunPod ``/runsync`` worker endpoints."""
|
|
71
|
+
|
|
72
|
+
def __init__(self, base_url: str, secret: str = "") -> None:
|
|
73
|
+
self._base_url = base_url.rstrip("/")
|
|
74
|
+
self._secret = secret
|
|
75
|
+
|
|
76
|
+
def _post(self, payload: dict, timeout: httpx.Timeout) -> dict:
|
|
77
|
+
resp = httpx.post(f"{self._base_url}/runsync", json=payload, timeout=timeout)
|
|
78
|
+
resp.raise_for_status()
|
|
79
|
+
return resp.json()
|
|
80
|
+
|
|
81
|
+
def list_models(self, backend: str = "") -> tuple[list[str], str]:
|
|
82
|
+
payload: dict = {"input": {"op": "models"}}
|
|
83
|
+
if backend:
|
|
84
|
+
payload["input"]["backend"] = backend
|
|
85
|
+
if self._secret:
|
|
86
|
+
payload["input"]["secret"] = self._secret
|
|
87
|
+
|
|
88
|
+
try:
|
|
89
|
+
data = self._post(
|
|
90
|
+
payload,
|
|
91
|
+
timeout=httpx.Timeout(connect=5.0, read=20.0, write=5.0, pool=5.0),
|
|
92
|
+
)
|
|
93
|
+
out = data.get("output", {}) if isinstance(data, dict) else {}
|
|
94
|
+
if not isinstance(out, dict):
|
|
95
|
+
return [], ""
|
|
96
|
+
return out.get("models", []), out.get("default", "")
|
|
97
|
+
except Exception: # noqa: BLE001
|
|
98
|
+
return [], ""
|
|
99
|
+
|
|
100
|
+
def rewrite(
|
|
101
|
+
self,
|
|
102
|
+
text: str,
|
|
103
|
+
backend: str = "",
|
|
104
|
+
model: str = "",
|
|
105
|
+
) -> tuple[str, str | None]:
|
|
106
|
+
payload: dict = {"input": {"op": "rewrite", "text": text}}
|
|
107
|
+
if backend:
|
|
108
|
+
payload["input"]["backend"] = backend
|
|
109
|
+
if model:
|
|
110
|
+
payload["input"]["model"] = model
|
|
111
|
+
if self._secret:
|
|
112
|
+
payload["input"]["secret"] = self._secret
|
|
113
|
+
|
|
114
|
+
try:
|
|
115
|
+
data = self._post(
|
|
116
|
+
payload,
|
|
117
|
+
timeout=httpx.Timeout(connect=5.0, read=60.0, write=10.0, pool=5.0),
|
|
118
|
+
)
|
|
119
|
+
err = extract_worker_error(data)
|
|
120
|
+
if err:
|
|
121
|
+
return text, err
|
|
122
|
+
out = data.get("output", {}) if isinstance(data, dict) else {}
|
|
123
|
+
if not isinstance(out, dict):
|
|
124
|
+
return text, "unexpected worker output"
|
|
125
|
+
return out.get("prompt", text), out.get("error")
|
|
126
|
+
except Exception as exc: # noqa: BLE001
|
|
127
|
+
return text, str(exc)
|
|
128
|
+
|
|
129
|
+
def imagine(
|
|
130
|
+
self,
|
|
131
|
+
prompt: str,
|
|
132
|
+
*,
|
|
133
|
+
image_backend: str = "",
|
|
134
|
+
aspect_ratio: str = "3:2",
|
|
135
|
+
steps: int | None = None,
|
|
136
|
+
) -> tuple[str | None, str | None, str | None, str | None]:
|
|
137
|
+
payload: dict = {"input": {"op": "imagine", "prompt": prompt, "aspect_ratio": aspect_ratio}}
|
|
138
|
+
if image_backend:
|
|
139
|
+
payload["input"]["image_backend"] = image_backend
|
|
140
|
+
if steps is not None:
|
|
141
|
+
payload["input"]["steps"] = steps
|
|
142
|
+
if self._secret:
|
|
143
|
+
payload["input"]["secret"] = self._secret
|
|
144
|
+
|
|
145
|
+
try:
|
|
146
|
+
data = self._post(
|
|
147
|
+
payload,
|
|
148
|
+
timeout=httpx.Timeout(connect=5.0, read=300.0, write=10.0, pool=5.0),
|
|
149
|
+
)
|
|
150
|
+
err = extract_worker_error(data)
|
|
151
|
+
if err:
|
|
152
|
+
return None, None, None, err
|
|
153
|
+
|
|
154
|
+
out = data.get("output", {}) if isinstance(data, dict) else {}
|
|
155
|
+
if not isinstance(out, dict):
|
|
156
|
+
return None, None, None, "unexpected worker output"
|
|
157
|
+
if "error" in out:
|
|
158
|
+
return None, None, None, str(out["error"])
|
|
159
|
+
return out.get("image_b64"), out.get("image_model"), out.get("image_backend"), None
|
|
160
|
+
except Exception as exc: # noqa: BLE001
|
|
161
|
+
return None, None, None, str(exc)
|
|
162
|
+
|
|
163
|
+
def query(
|
|
164
|
+
self,
|
|
165
|
+
query: str,
|
|
166
|
+
*,
|
|
167
|
+
corpus: str = "all",
|
|
168
|
+
k: int = 8,
|
|
169
|
+
min_score: float = 0.0,
|
|
170
|
+
semantic_floor: float = 0.0,
|
|
171
|
+
synthesize: bool = False,
|
|
172
|
+
model: str = "",
|
|
173
|
+
backend: str = "",
|
|
174
|
+
) -> dict:
|
|
175
|
+
payload: dict = {
|
|
176
|
+
"input": {
|
|
177
|
+
"query": query,
|
|
178
|
+
"corpus": corpus,
|
|
179
|
+
"k": k,
|
|
180
|
+
"min_score": min_score,
|
|
181
|
+
"semantic_floor": semantic_floor,
|
|
182
|
+
"synthesize": synthesize,
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
if model:
|
|
186
|
+
payload["input"]["model"] = model
|
|
187
|
+
if backend:
|
|
188
|
+
payload["input"]["backend"] = backend
|
|
189
|
+
if self._secret:
|
|
190
|
+
payload["input"]["secret"] = self._secret
|
|
191
|
+
|
|
192
|
+
data = self._post(
|
|
193
|
+
payload,
|
|
194
|
+
timeout=httpx.Timeout(connect=5.0, read=600.0, write=30.0, pool=5.0),
|
|
195
|
+
)
|
|
196
|
+
return decode_worker_response(data)
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
# © 2026 Eric G. Suchanek, PhD — Flux-Frontiers · SPDX-License-Identifier: Elastic-2.0
|
|
2
|
+
"""Shared handler operation dispatch for models, rewrite, and imagine."""
|
|
3
|
+
|
|
4
|
+
from __future__ import annotations
|
|
5
|
+
|
|
6
|
+
from collections.abc import Callable
|
|
7
|
+
|
|
8
|
+
from kg_utils.synthesis._image import ImageSynthesizer
|
|
9
|
+
from kg_utils.synthesis._text import TextSynthesizer
|
|
10
|
+
|
|
11
|
+
__all__ = ["handle_aux_ops"]
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
def handle_aux_ops(
|
|
15
|
+
inp: dict,
|
|
16
|
+
text_synth_factory: Callable[[str], TextSynthesizer],
|
|
17
|
+
image_synth_factory: Callable[[str], ImageSynthesizer],
|
|
18
|
+
) -> dict | None:
|
|
19
|
+
"""Handle shared non-query worker operations.
|
|
20
|
+
|
|
21
|
+
Returns:
|
|
22
|
+
- operation payload dict when op is recognized
|
|
23
|
+
- ``None`` when input has no recognized operation
|
|
24
|
+
"""
|
|
25
|
+
op = inp.get("op")
|
|
26
|
+
|
|
27
|
+
if op == "models":
|
|
28
|
+
synth = text_synth_factory(inp.get("backend", ""))
|
|
29
|
+
# Existing handlers expose the active model via synthesizer config internals.
|
|
30
|
+
return {
|
|
31
|
+
"models": synth.list_models(),
|
|
32
|
+
"default": synth._cfg.resolved_model(), # pylint: disable=protected-access
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
if op == "rewrite":
|
|
36
|
+
text = (inp.get("text") or "").strip()
|
|
37
|
+
if not text:
|
|
38
|
+
return {"error": "rewrite requires a non-empty 'text'"}
|
|
39
|
+
|
|
40
|
+
synth = text_synth_factory(inp.get("backend", ""))
|
|
41
|
+
model_override = (inp.get("model") or "").strip() or None
|
|
42
|
+
prompt, error = synth.rewrite_for_image(text, model=model_override)
|
|
43
|
+
return {"prompt": prompt, "error": error}
|
|
44
|
+
|
|
45
|
+
if op == "imagine":
|
|
46
|
+
prompt = (inp.get("prompt") or "").strip()
|
|
47
|
+
if not prompt:
|
|
48
|
+
return {"error": "imagine requires a non-empty 'prompt'"}
|
|
49
|
+
|
|
50
|
+
aspect = inp.get("aspect_ratio", "3:2")
|
|
51
|
+
seed = inp.get("seed")
|
|
52
|
+
steps = inp.get("steps")
|
|
53
|
+
img_synth = image_synth_factory(inp.get("image_backend", ""))
|
|
54
|
+
|
|
55
|
+
try:
|
|
56
|
+
b64 = img_synth.generate_b64(
|
|
57
|
+
prompt,
|
|
58
|
+
aspect_ratio=aspect,
|
|
59
|
+
seed=int(seed) if seed is not None else None,
|
|
60
|
+
steps=int(steps) if steps is not None else None,
|
|
61
|
+
)
|
|
62
|
+
return {
|
|
63
|
+
"image_b64": b64,
|
|
64
|
+
"prompt": prompt,
|
|
65
|
+
"aspect_ratio": aspect,
|
|
66
|
+
"image_model": img_synth._cfg.resolved_model(), # pylint: disable=protected-access
|
|
67
|
+
"image_backend": img_synth._cfg.backend.value, # pylint: disable=protected-access
|
|
68
|
+
}
|
|
69
|
+
except Exception as exc: # noqa: BLE001 # pylint: disable=broad-exception-caught
|
|
70
|
+
return {"error": f"image generation failed: {exc}"}
|
|
71
|
+
|
|
72
|
+
return None
|
|
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
|