llms-py 2.0.34__py3-none-any.whl → 3.0.0__py3-none-any.whl

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.
Files changed (207) hide show
  1. llms/__init__.py +3 -1
  2. llms/__pycache__/__init__.cpython-312.pyc +0 -0
  3. llms/__pycache__/__init__.cpython-313.pyc +0 -0
  4. llms/__pycache__/__init__.cpython-314.pyc +0 -0
  5. llms/__pycache__/__main__.cpython-312.pyc +0 -0
  6. llms/__pycache__/__main__.cpython-314.pyc +0 -0
  7. llms/__pycache__/llms.cpython-312.pyc +0 -0
  8. llms/__pycache__/main.cpython-312.pyc +0 -0
  9. llms/__pycache__/main.cpython-313.pyc +0 -0
  10. llms/__pycache__/main.cpython-314.pyc +0 -0
  11. llms/__pycache__/plugins.cpython-314.pyc +0 -0
  12. llms/{ui/Analytics.mjs → extensions/analytics/ui/index.mjs} +154 -238
  13. llms/extensions/app/README.md +20 -0
  14. llms/extensions/app/__init__.py +530 -0
  15. llms/extensions/app/__pycache__/__init__.cpython-314.pyc +0 -0
  16. llms/extensions/app/__pycache__/db.cpython-314.pyc +0 -0
  17. llms/extensions/app/__pycache__/db_manager.cpython-314.pyc +0 -0
  18. llms/extensions/app/db.py +644 -0
  19. llms/extensions/app/db_manager.py +195 -0
  20. llms/extensions/app/requests.json +9073 -0
  21. llms/extensions/app/threads.json +15290 -0
  22. llms/{ui → extensions/app/ui}/Recents.mjs +91 -65
  23. llms/{ui/Sidebar.mjs → extensions/app/ui/index.mjs} +124 -58
  24. llms/extensions/app/ui/threadStore.mjs +411 -0
  25. llms/extensions/core_tools/CALCULATOR.md +32 -0
  26. llms/extensions/core_tools/__init__.py +598 -0
  27. llms/extensions/core_tools/__pycache__/__init__.cpython-314.pyc +0 -0
  28. llms/extensions/core_tools/ui/codemirror/addon/edit/closebrackets.js +201 -0
  29. llms/extensions/core_tools/ui/codemirror/addon/edit/closetag.js +185 -0
  30. llms/extensions/core_tools/ui/codemirror/addon/edit/continuelist.js +101 -0
  31. llms/extensions/core_tools/ui/codemirror/addon/edit/matchbrackets.js +160 -0
  32. llms/extensions/core_tools/ui/codemirror/addon/edit/matchtags.js +66 -0
  33. llms/extensions/core_tools/ui/codemirror/addon/edit/trailingspace.js +27 -0
  34. llms/extensions/core_tools/ui/codemirror/addon/selection/active-line.js +72 -0
  35. llms/extensions/core_tools/ui/codemirror/addon/selection/mark-selection.js +119 -0
  36. llms/extensions/core_tools/ui/codemirror/addon/selection/selection-pointer.js +98 -0
  37. llms/extensions/core_tools/ui/codemirror/doc/docs.css +225 -0
  38. llms/extensions/core_tools/ui/codemirror/doc/source_sans.woff +0 -0
  39. llms/extensions/core_tools/ui/codemirror/lib/codemirror.css +344 -0
  40. llms/extensions/core_tools/ui/codemirror/lib/codemirror.js +9884 -0
  41. llms/extensions/core_tools/ui/codemirror/mode/clike/clike.js +942 -0
  42. llms/extensions/core_tools/ui/codemirror/mode/javascript/index.html +118 -0
  43. llms/extensions/core_tools/ui/codemirror/mode/javascript/javascript.js +962 -0
  44. llms/extensions/core_tools/ui/codemirror/mode/javascript/typescript.html +62 -0
  45. llms/extensions/core_tools/ui/codemirror/mode/python/python.js +402 -0
  46. llms/extensions/core_tools/ui/codemirror/theme/dracula.css +40 -0
  47. llms/extensions/core_tools/ui/codemirror/theme/mocha.css +135 -0
  48. llms/extensions/core_tools/ui/index.mjs +650 -0
  49. llms/extensions/gallery/README.md +61 -0
  50. llms/extensions/gallery/__init__.py +61 -0
  51. llms/extensions/gallery/__pycache__/__init__.cpython-314.pyc +0 -0
  52. llms/extensions/gallery/__pycache__/db.cpython-314.pyc +0 -0
  53. llms/extensions/gallery/db.py +298 -0
  54. llms/extensions/gallery/ui/index.mjs +482 -0
  55. llms/extensions/katex/README.md +39 -0
  56. llms/extensions/katex/__init__.py +6 -0
  57. llms/extensions/katex/__pycache__/__init__.cpython-314.pyc +0 -0
  58. llms/extensions/katex/ui/README.md +125 -0
  59. llms/extensions/katex/ui/contrib/auto-render.js +338 -0
  60. llms/extensions/katex/ui/contrib/auto-render.min.js +1 -0
  61. llms/extensions/katex/ui/contrib/auto-render.mjs +244 -0
  62. llms/extensions/katex/ui/contrib/copy-tex.js +127 -0
  63. llms/extensions/katex/ui/contrib/copy-tex.min.js +1 -0
  64. llms/extensions/katex/ui/contrib/copy-tex.mjs +105 -0
  65. llms/extensions/katex/ui/contrib/mathtex-script-type.js +109 -0
  66. llms/extensions/katex/ui/contrib/mathtex-script-type.min.js +1 -0
  67. llms/extensions/katex/ui/contrib/mathtex-script-type.mjs +24 -0
  68. llms/extensions/katex/ui/contrib/mhchem.js +3213 -0
  69. llms/extensions/katex/ui/contrib/mhchem.min.js +1 -0
  70. llms/extensions/katex/ui/contrib/mhchem.mjs +3109 -0
  71. llms/extensions/katex/ui/contrib/render-a11y-string.js +887 -0
  72. llms/extensions/katex/ui/contrib/render-a11y-string.min.js +1 -0
  73. llms/extensions/katex/ui/contrib/render-a11y-string.mjs +800 -0
  74. llms/extensions/katex/ui/fonts/KaTeX_AMS-Regular.ttf +0 -0
  75. llms/extensions/katex/ui/fonts/KaTeX_AMS-Regular.woff +0 -0
  76. llms/extensions/katex/ui/fonts/KaTeX_AMS-Regular.woff2 +0 -0
  77. llms/extensions/katex/ui/fonts/KaTeX_Caligraphic-Bold.ttf +0 -0
  78. llms/extensions/katex/ui/fonts/KaTeX_Caligraphic-Bold.woff +0 -0
  79. llms/extensions/katex/ui/fonts/KaTeX_Caligraphic-Bold.woff2 +0 -0
  80. llms/extensions/katex/ui/fonts/KaTeX_Caligraphic-Regular.ttf +0 -0
  81. llms/extensions/katex/ui/fonts/KaTeX_Caligraphic-Regular.woff +0 -0
  82. llms/extensions/katex/ui/fonts/KaTeX_Caligraphic-Regular.woff2 +0 -0
  83. llms/extensions/katex/ui/fonts/KaTeX_Fraktur-Bold.ttf +0 -0
  84. llms/extensions/katex/ui/fonts/KaTeX_Fraktur-Bold.woff +0 -0
  85. llms/extensions/katex/ui/fonts/KaTeX_Fraktur-Bold.woff2 +0 -0
  86. llms/extensions/katex/ui/fonts/KaTeX_Fraktur-Regular.ttf +0 -0
  87. llms/extensions/katex/ui/fonts/KaTeX_Fraktur-Regular.woff +0 -0
  88. llms/extensions/katex/ui/fonts/KaTeX_Fraktur-Regular.woff2 +0 -0
  89. llms/extensions/katex/ui/fonts/KaTeX_Main-Bold.ttf +0 -0
  90. llms/extensions/katex/ui/fonts/KaTeX_Main-Bold.woff +0 -0
  91. llms/extensions/katex/ui/fonts/KaTeX_Main-Bold.woff2 +0 -0
  92. llms/extensions/katex/ui/fonts/KaTeX_Main-BoldItalic.ttf +0 -0
  93. llms/extensions/katex/ui/fonts/KaTeX_Main-BoldItalic.woff +0 -0
  94. llms/extensions/katex/ui/fonts/KaTeX_Main-BoldItalic.woff2 +0 -0
  95. llms/extensions/katex/ui/fonts/KaTeX_Main-Italic.ttf +0 -0
  96. llms/extensions/katex/ui/fonts/KaTeX_Main-Italic.woff +0 -0
  97. llms/extensions/katex/ui/fonts/KaTeX_Main-Italic.woff2 +0 -0
  98. llms/extensions/katex/ui/fonts/KaTeX_Main-Regular.ttf +0 -0
  99. llms/extensions/katex/ui/fonts/KaTeX_Main-Regular.woff +0 -0
  100. llms/extensions/katex/ui/fonts/KaTeX_Main-Regular.woff2 +0 -0
  101. llms/extensions/katex/ui/fonts/KaTeX_Math-BoldItalic.ttf +0 -0
  102. llms/extensions/katex/ui/fonts/KaTeX_Math-BoldItalic.woff +0 -0
  103. llms/extensions/katex/ui/fonts/KaTeX_Math-BoldItalic.woff2 +0 -0
  104. llms/extensions/katex/ui/fonts/KaTeX_Math-Italic.ttf +0 -0
  105. llms/extensions/katex/ui/fonts/KaTeX_Math-Italic.woff +0 -0
  106. llms/extensions/katex/ui/fonts/KaTeX_Math-Italic.woff2 +0 -0
  107. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Bold.ttf +0 -0
  108. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Bold.woff +0 -0
  109. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Bold.woff2 +0 -0
  110. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Italic.ttf +0 -0
  111. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Italic.woff +0 -0
  112. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Italic.woff2 +0 -0
  113. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Regular.ttf +0 -0
  114. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Regular.woff +0 -0
  115. llms/extensions/katex/ui/fonts/KaTeX_SansSerif-Regular.woff2 +0 -0
  116. llms/extensions/katex/ui/fonts/KaTeX_Script-Regular.ttf +0 -0
  117. llms/extensions/katex/ui/fonts/KaTeX_Script-Regular.woff +0 -0
  118. llms/extensions/katex/ui/fonts/KaTeX_Script-Regular.woff2 +0 -0
  119. llms/extensions/katex/ui/fonts/KaTeX_Size1-Regular.ttf +0 -0
  120. llms/extensions/katex/ui/fonts/KaTeX_Size1-Regular.woff +0 -0
  121. llms/extensions/katex/ui/fonts/KaTeX_Size1-Regular.woff2 +0 -0
  122. llms/extensions/katex/ui/fonts/KaTeX_Size2-Regular.ttf +0 -0
  123. llms/extensions/katex/ui/fonts/KaTeX_Size2-Regular.woff +0 -0
  124. llms/extensions/katex/ui/fonts/KaTeX_Size2-Regular.woff2 +0 -0
  125. llms/extensions/katex/ui/fonts/KaTeX_Size3-Regular.ttf +0 -0
  126. llms/extensions/katex/ui/fonts/KaTeX_Size3-Regular.woff +0 -0
  127. llms/extensions/katex/ui/fonts/KaTeX_Size3-Regular.woff2 +0 -0
  128. llms/extensions/katex/ui/fonts/KaTeX_Size4-Regular.ttf +0 -0
  129. llms/extensions/katex/ui/fonts/KaTeX_Size4-Regular.woff +0 -0
  130. llms/extensions/katex/ui/fonts/KaTeX_Size4-Regular.woff2 +0 -0
  131. llms/extensions/katex/ui/fonts/KaTeX_Typewriter-Regular.ttf +0 -0
  132. llms/extensions/katex/ui/fonts/KaTeX_Typewriter-Regular.woff +0 -0
  133. llms/extensions/katex/ui/fonts/KaTeX_Typewriter-Regular.woff2 +0 -0
  134. llms/extensions/katex/ui/index.mjs +92 -0
  135. llms/extensions/katex/ui/katex-swap.css +1230 -0
  136. llms/extensions/katex/ui/katex-swap.min.css +1 -0
  137. llms/extensions/katex/ui/katex.css +1230 -0
  138. llms/extensions/katex/ui/katex.js +19080 -0
  139. llms/extensions/katex/ui/katex.min.css +1 -0
  140. llms/extensions/katex/ui/katex.min.js +1 -0
  141. llms/extensions/katex/ui/katex.min.mjs +1 -0
  142. llms/extensions/katex/ui/katex.mjs +18547 -0
  143. llms/extensions/providers/__init__.py +18 -0
  144. llms/extensions/providers/__pycache__/__init__.cpython-314.pyc +0 -0
  145. llms/extensions/providers/__pycache__/anthropic.cpython-314.pyc +0 -0
  146. llms/extensions/providers/__pycache__/chutes.cpython-314.pyc +0 -0
  147. llms/extensions/providers/__pycache__/google.cpython-314.pyc +0 -0
  148. llms/extensions/providers/__pycache__/nvidia.cpython-314.pyc +0 -0
  149. llms/extensions/providers/__pycache__/openai.cpython-314.pyc +0 -0
  150. llms/extensions/providers/__pycache__/openrouter.cpython-314.pyc +0 -0
  151. llms/extensions/providers/anthropic.py +229 -0
  152. llms/extensions/providers/chutes.py +155 -0
  153. llms/extensions/providers/google.py +378 -0
  154. llms/extensions/providers/nvidia.py +105 -0
  155. llms/extensions/providers/openai.py +156 -0
  156. llms/extensions/providers/openrouter.py +72 -0
  157. llms/extensions/system_prompts/README.md +22 -0
  158. llms/extensions/system_prompts/__init__.py +45 -0
  159. llms/extensions/system_prompts/__pycache__/__init__.cpython-314.pyc +0 -0
  160. llms/extensions/system_prompts/ui/index.mjs +280 -0
  161. llms/extensions/system_prompts/ui/prompts.json +1067 -0
  162. llms/extensions/tools/__init__.py +5 -0
  163. llms/extensions/tools/__pycache__/__init__.cpython-314.pyc +0 -0
  164. llms/extensions/tools/ui/index.mjs +204 -0
  165. llms/index.html +35 -77
  166. llms/llms.json +357 -1186
  167. llms/main.py +2847 -999
  168. llms/providers-extra.json +356 -0
  169. llms/providers.json +1 -0
  170. llms/ui/App.mjs +151 -60
  171. llms/ui/ai.mjs +132 -60
  172. llms/ui/app.css +2173 -161
  173. llms/ui/ctx.mjs +365 -0
  174. llms/ui/index.mjs +129 -0
  175. llms/ui/lib/charts.mjs +9 -13
  176. llms/ui/lib/servicestack-vue.mjs +3 -3
  177. llms/ui/lib/vue.min.mjs +10 -9
  178. llms/ui/lib/vue.mjs +1796 -1635
  179. llms/ui/markdown.mjs +18 -7
  180. llms/ui/modules/chat/ChatBody.mjs +691 -0
  181. llms/ui/{SettingsDialog.mjs → modules/chat/SettingsDialog.mjs} +9 -9
  182. llms/ui/modules/chat/index.mjs +828 -0
  183. llms/ui/modules/layout.mjs +243 -0
  184. llms/ui/modules/model-selector.mjs +851 -0
  185. llms/ui/tailwind.input.css +496 -80
  186. llms/ui/utils.mjs +161 -93
  187. {llms_py-2.0.34.dist-info → llms_py-3.0.0.dist-info}/METADATA +1 -1
  188. llms_py-3.0.0.dist-info/RECORD +202 -0
  189. llms/ui/Avatar.mjs +0 -85
  190. llms/ui/Brand.mjs +0 -52
  191. llms/ui/ChatPrompt.mjs +0 -590
  192. llms/ui/Main.mjs +0 -823
  193. llms/ui/ModelSelector.mjs +0 -78
  194. llms/ui/OAuthSignIn.mjs +0 -92
  195. llms/ui/ProviderIcon.mjs +0 -30
  196. llms/ui/ProviderStatus.mjs +0 -105
  197. llms/ui/SignIn.mjs +0 -64
  198. llms/ui/SystemPromptEditor.mjs +0 -31
  199. llms/ui/SystemPromptSelector.mjs +0 -56
  200. llms/ui/Welcome.mjs +0 -8
  201. llms/ui/threadStore.mjs +0 -563
  202. llms/ui.json +0 -1069
  203. llms_py-2.0.34.dist-info/RECORD +0 -48
  204. {llms_py-2.0.34.dist-info → llms_py-3.0.0.dist-info}/WHEEL +0 -0
  205. {llms_py-2.0.34.dist-info → llms_py-3.0.0.dist-info}/entry_points.txt +0 -0
  206. {llms_py-2.0.34.dist-info → llms_py-3.0.0.dist-info}/licenses/LICENSE +0 -0
  207. {llms_py-2.0.34.dist-info → llms_py-3.0.0.dist-info}/top_level.txt +0 -0
llms/main.py CHANGED
@@ -6,104 +6,123 @@
6
6
  # A lightweight CLI tool and OpenAI-compatible server for querying multiple Large Language Model (LLM) providers.
7
7
  # Docs: https://github.com/ServiceStack/llms
8
8
 
9
- import os
10
- import time
11
- import json
12
9
  import argparse
13
10
  import asyncio
14
- import subprocess
15
11
  import base64
12
+ import contextlib
13
+ import hashlib
14
+ import importlib.util
15
+ import inspect
16
+ import json
16
17
  import mimetypes
17
- import traceback
18
- import sys
19
- import site
20
- import secrets
18
+ import os
21
19
  import re
20
+ import secrets
21
+ import shutil
22
+ import site
23
+ import subprocess
24
+ import sys
25
+ import time
26
+ import traceback
27
+ from datetime import datetime
28
+ from importlib import resources # Py≥3.9 (pip install importlib_resources for 3.7/3.8)
22
29
  from io import BytesIO
23
- from urllib.parse import parse_qs, urlencode
30
+ from pathlib import Path
31
+ from typing import get_type_hints
32
+ from urllib.parse import parse_qs, urlencode, urljoin
24
33
 
25
34
  import aiohttp
26
35
  from aiohttp import web
27
36
 
28
- from pathlib import Path
29
- from importlib import resources # Py≥3.9 (pip install importlib_resources for 3.7/3.8)
30
-
31
37
  try:
32
38
  from PIL import Image
39
+
33
40
  HAS_PIL = True
34
41
  except ImportError:
35
42
  HAS_PIL = False
36
43
 
37
- VERSION = "2.0.34"
44
+ VERSION = "3.0.0"
38
45
  _ROOT = None
46
+ DEBUG = os.getenv("DEBUG") == "1"
47
+ MOCK = os.getenv("MOCK") == "1"
48
+ MOCK_DIR = os.getenv("MOCK_DIR")
49
+ DISABLE_EXTENSIONS = (os.getenv("LLMS_DISABLE") or "").split(",")
39
50
  g_config_path = None
40
- g_ui_path = None
41
51
  g_config = None
52
+ g_providers = None
42
53
  g_handlers = {}
43
54
  g_verbose = False
44
- g_logprefix=""
45
- g_default_model=""
55
+ g_logprefix = ""
56
+ g_default_model = ""
46
57
  g_sessions = {} # OAuth session storage: {session_token: {userId, userName, displayName, profileUrl, email, created}}
47
58
  g_oauth_states = {} # CSRF protection: {state: {created, redirect_uri}}
59
+ g_app = None # ExtensionsContext Singleton
60
+
48
61
 
49
62
  def _log(message):
50
- """Helper method for logging from the global polling task."""
51
63
  if g_verbose:
52
64
  print(f"{g_logprefix}{message}", flush=True)
53
65
 
66
+
67
+ def _dbg(message):
68
+ if DEBUG:
69
+ print(f"DEBUG: {message}", flush=True)
70
+
71
+
72
+ def _err(message, e):
73
+ print(f"ERROR: {message}: {e}", flush=True)
74
+ if g_verbose:
75
+ print(traceback.format_exc(), flush=True)
76
+
77
+
54
78
  def printdump(obj):
55
- args = obj.__dict__ if hasattr(obj, '__dict__') else obj
79
+ args = obj.__dict__ if hasattr(obj, "__dict__") else obj
56
80
  print(json.dumps(args, indent=2))
57
81
 
82
+
58
83
  def print_chat(chat):
59
84
  _log(f"Chat: {chat_summary(chat)}")
60
85
 
86
+
61
87
  def chat_summary(chat):
62
88
  """Summarize chat completion request for logging."""
63
89
  # replace image_url.url with <image>
64
90
  clone = json.loads(json.dumps(chat))
65
- for message in clone['messages']:
66
- if 'content' in message:
67
- if isinstance(message['content'], list):
68
- for item in message['content']:
69
- if 'image_url' in item:
70
- if 'url' in item['image_url']:
71
- url = item['image_url']['url']
72
- prefix = url.split(',', 1)[0]
73
- item['image_url']['url'] = prefix + f",({len(url) - len(prefix)})"
74
- elif 'input_audio' in item:
75
- if 'data' in item['input_audio']:
76
- data = item['input_audio']['data']
77
- item['input_audio']['data'] = f"({len(data)})"
78
- elif 'file' in item:
79
- if 'file_data' in item['file']:
80
- data = item['file']['file_data']
81
- prefix = data.split(',', 1)[0]
82
- item['file']['file_data'] = prefix + f",({len(data) - len(prefix)})"
91
+ for message in clone["messages"]:
92
+ if "content" in message and isinstance(message["content"], list):
93
+ for item in message["content"]:
94
+ if "image_url" in item:
95
+ if "url" in item["image_url"]:
96
+ url = item["image_url"]["url"]
97
+ prefix = url.split(",", 1)[0]
98
+ item["image_url"]["url"] = prefix + f",({len(url) - len(prefix)})"
99
+ elif "input_audio" in item:
100
+ if "data" in item["input_audio"]:
101
+ data = item["input_audio"]["data"]
102
+ item["input_audio"]["data"] = f"({len(data)})"
103
+ elif "file" in item and "file_data" in item["file"]:
104
+ data = item["file"]["file_data"]
105
+ prefix = data.split(",", 1)[0]
106
+ item["file"]["file_data"] = prefix + f",({len(data) - len(prefix)})"
83
107
  return json.dumps(clone, indent=2)
84
108
 
85
- def gemini_chat_summary(gemini_chat):
86
- """Summarize Gemini chat completion request for logging. Replace inline_data with size of content only"""
87
- clone = json.loads(json.dumps(gemini_chat))
88
- for content in clone['contents']:
89
- for part in content['parts']:
90
- if 'inline_data' in part:
91
- data = part['inline_data']['data']
92
- part['inline_data']['data'] = f"({len(data)})"
93
- return json.dumps(clone, indent=2)
94
109
 
95
- image_exts = 'png,webp,jpg,jpeg,gif,bmp,svg,tiff,ico'.split(',')
96
- audio_exts = 'mp3,wav,ogg,flac,m4a,opus,webm'.split(',')
110
+ image_exts = ["png", "webp", "jpg", "jpeg", "gif", "bmp", "svg", "tiff", "ico"]
111
+ audio_exts = ["mp3", "wav", "ogg", "flac", "m4a", "opus", "webm"]
112
+
97
113
 
98
114
  def is_file_path(path):
99
115
  # macOs max path is 1023
100
116
  return path and len(path) < 1024 and os.path.exists(path)
101
117
 
118
+
102
119
  def is_url(url):
103
- return url and (url.startswith('http://') or url.startswith('https://'))
120
+ return url and (url.startswith("http://") or url.startswith("https://"))
121
+
104
122
 
105
123
  def get_filename(file):
106
- return file.rsplit('/',1)[1] if '/' in file else 'file'
124
+ return file.rsplit("/", 1)[1] if "/" in file else "file"
125
+
107
126
 
108
127
  def parse_args_params(args_str):
109
128
  """Parse URL-encoded parameters and return a dictionary."""
@@ -119,9 +138,9 @@ def parse_args_params(args_str):
119
138
  if len(values) == 1:
120
139
  value = values[0]
121
140
  # Try to convert to appropriate types
122
- if value.lower() == 'true':
141
+ if value.lower() == "true":
123
142
  result[key] = True
124
- elif value.lower() == 'false':
143
+ elif value.lower() == "false":
125
144
  result[key] = False
126
145
  elif value.isdigit():
127
146
  result[key] = int(value)
@@ -138,6 +157,7 @@ def parse_args_params(args_str):
138
157
 
139
158
  return result
140
159
 
160
+
141
161
  def apply_args_to_chat(chat, args_params):
142
162
  """Apply parsed arguments to the chat request."""
143
163
  if not args_params:
@@ -146,19 +166,32 @@ def apply_args_to_chat(chat, args_params):
146
166
  # Apply each parameter to the chat request
147
167
  for key, value in args_params.items():
148
168
  if isinstance(value, str):
149
- if key == 'stop':
150
- if ',' in value:
151
- value = value.split(',')
152
- elif key == 'max_completion_tokens' or key == 'max_tokens' or key == 'n' or key == 'seed' or key == 'top_logprobs':
169
+ if key == "stop":
170
+ if "," in value:
171
+ value = value.split(",")
172
+ elif (
173
+ key == "max_completion_tokens"
174
+ or key == "max_tokens"
175
+ or key == "n"
176
+ or key == "seed"
177
+ or key == "top_logprobs"
178
+ ):
153
179
  value = int(value)
154
- elif key == 'temperature' or key == 'top_p' or key == 'frequency_penalty' or key == 'presence_penalty':
180
+ elif key == "temperature" or key == "top_p" or key == "frequency_penalty" or key == "presence_penalty":
155
181
  value = float(value)
156
- elif key == 'store' or key == 'logprobs' or key == 'enable_thinking' or key == 'parallel_tool_calls' or key == 'stream':
182
+ elif (
183
+ key == "store"
184
+ or key == "logprobs"
185
+ or key == "enable_thinking"
186
+ or key == "parallel_tool_calls"
187
+ or key == "stream"
188
+ ):
157
189
  value = bool(value)
158
190
  chat[key] = value
159
191
 
160
192
  return chat
161
193
 
194
+
162
195
  def is_base_64(data):
163
196
  try:
164
197
  base64.b64decode(data)
@@ -166,6 +199,17 @@ def is_base_64(data):
166
199
  except Exception:
167
200
  return False
168
201
 
202
+
203
+ def id_to_name(id):
204
+ return id.replace("-", " ").title()
205
+
206
+
207
+ def pluralize(word, count):
208
+ if count == 1:
209
+ return word
210
+ return word + "s"
211
+
212
+
169
213
  def get_file_mime_type(filename):
170
214
  mime_type, _ = mimetypes.guess_type(filename)
171
215
  return mime_type or "application/octet-stream"
@@ -182,36 +226,38 @@ def price_to_string(price: float | int | str | None) -> str | None:
182
226
  try:
183
227
  price_float = float(price)
184
228
  # Format with enough decimal places to avoid scientific notation
185
- formatted = format(price_float, '.20f')
229
+ formatted = format(price_float, ".20f")
186
230
 
187
231
  # Detect recurring 9s pattern (e.g., "...9999999")
188
232
  # If we have 4 or more consecutive 9s, round up
189
- if '9999' in formatted:
233
+ if "9999" in formatted:
190
234
  # Round up by adding a small amount and reformatting
191
235
  # Find the position of the 9s to determine precision
192
236
  import decimal
237
+
193
238
  decimal.getcontext().prec = 28
194
239
  d = decimal.Decimal(str(price_float))
195
240
  # Round to one less decimal place than where the 9s start
196
- nines_pos = formatted.find('9999')
241
+ nines_pos = formatted.find("9999")
197
242
  if nines_pos > 0:
198
243
  # Round up at the position before the 9s
199
- decimal_places = nines_pos - formatted.find('.') - 1
244
+ decimal_places = nines_pos - formatted.find(".") - 1
200
245
  if decimal_places > 0:
201
- quantize_str = '0.' + '0' * (decimal_places - 1) + '1'
246
+ quantize_str = "0." + "0" * (decimal_places - 1) + "1"
202
247
  d = d.quantize(decimal.Decimal(quantize_str), rounding=decimal.ROUND_UP)
203
248
  result = str(d)
204
249
  # Remove trailing zeros
205
- if '.' in result:
206
- result = result.rstrip('0').rstrip('.')
250
+ if "." in result:
251
+ result = result.rstrip("0").rstrip(".")
207
252
  return result
208
253
 
209
254
  # Normal case: strip trailing zeros
210
- return formatted.rstrip('0').rstrip('.')
255
+ return formatted.rstrip("0").rstrip(".")
211
256
  except (ValueError, TypeError):
212
257
  return None
213
258
 
214
- def convert_image_if_needed(image_bytes, mimetype='image/png'):
259
+
260
+ def convert_image_if_needed(image_bytes, mimetype="image/png"):
215
261
  """
216
262
  Convert and resize image to WebP if it exceeds configured limits.
217
263
 
@@ -226,16 +272,16 @@ def convert_image_if_needed(image_bytes, mimetype='image/png'):
226
272
  return image_bytes, mimetype
227
273
 
228
274
  # Get conversion config
229
- convert_config = g_config.get('convert', {}).get('image', {}) if g_config else {}
275
+ convert_config = g_config.get("convert", {}).get("image", {}) if g_config else {}
230
276
  if not convert_config:
231
277
  return image_bytes, mimetype
232
278
 
233
- max_size_str = convert_config.get('max_size', '1536x1024')
234
- max_length = convert_config.get('max_length', 1.5*1024*1024) # 1.5MB
279
+ max_size_str = convert_config.get("max_size", "1536x1024")
280
+ max_length = convert_config.get("max_length", 1.5 * 1024 * 1024) # 1.5MB
235
281
 
236
282
  try:
237
283
  # Parse max_size (e.g., "1536x1024")
238
- max_width, max_height = map(int, max_size_str.split('x'))
284
+ max_width, max_height = map(int, max_size_str.split("x"))
239
285
 
240
286
  # Open image
241
287
  with Image.open(BytesIO(image_bytes)) as img:
@@ -253,15 +299,15 @@ def convert_image_if_needed(image_bytes, mimetype='image/png'):
253
299
  return image_bytes, mimetype
254
300
 
255
301
  # Convert RGBA to RGB if necessary (WebP doesn't support transparency in RGB mode)
256
- if img.mode in ('RGBA', 'LA', 'P'):
302
+ if img.mode in ("RGBA", "LA", "P"):
257
303
  # Create a white background
258
- background = Image.new('RGB', img.size, (255, 255, 255))
259
- if img.mode == 'P':
260
- img = img.convert('RGBA')
261
- background.paste(img, mask=img.split()[-1] if img.mode in ('RGBA', 'LA') else None)
304
+ background = Image.new("RGB", img.size, (255, 255, 255))
305
+ if img.mode == "P":
306
+ img = img.convert("RGBA")
307
+ background.paste(img, mask=img.split()[-1] if img.mode in ("RGBA", "LA") else None)
262
308
  img = background
263
- elif img.mode != 'RGB':
264
- img = img.convert('RGB')
309
+ elif img.mode != "RGB":
310
+ img = img.convert("RGB")
265
311
 
266
312
  # Resize if needed (preserve aspect ratio)
267
313
  if needs_resize:
@@ -270,39 +316,85 @@ def convert_image_if_needed(image_bytes, mimetype='image/png'):
270
316
 
271
317
  # Convert to WebP
272
318
  output = BytesIO()
273
- img.save(output, format='WEBP', quality=85, method=6)
319
+ img.save(output, format="WEBP", quality=85, method=6)
274
320
  converted_bytes = output.getvalue()
275
321
 
276
- _log(f"Converted image to WebP: {len(image_bytes)} bytes -> {len(converted_bytes)} bytes ({len(converted_bytes)*100//len(image_bytes)}%)")
322
+ _log(
323
+ f"Converted image to WebP: {len(image_bytes)} bytes -> {len(converted_bytes)} bytes ({len(converted_bytes) * 100 // len(image_bytes)}%)"
324
+ )
277
325
 
278
- return converted_bytes, 'image/webp'
326
+ return converted_bytes, "image/webp"
279
327
 
280
328
  except Exception as e:
281
329
  _log(f"Error converting image: {e}")
282
330
  # Return original if conversion fails
283
331
  return image_bytes, mimetype
284
332
 
285
- async def process_chat(chat):
333
+
334
+ def to_content(result):
335
+ if isinstance(result, (str, int, float, bool)):
336
+ return str(result)
337
+ elif isinstance(result, (list, set, tuple, dict)):
338
+ return json.dumps(result)
339
+ else:
340
+ return str(result)
341
+
342
+
343
+ def function_to_tool_definition(func):
344
+ type_hints = get_type_hints(func)
345
+ signature = inspect.signature(func)
346
+ parameters = {"type": "object", "properties": {}, "required": []}
347
+
348
+ for name, param in signature.parameters.items():
349
+ param_type = type_hints.get(name, str)
350
+ param_type_name = "string"
351
+ if param_type is int:
352
+ param_type_name = "integer"
353
+ elif param_type is float:
354
+ param_type_name = "number"
355
+ elif param_type is bool:
356
+ param_type_name = "boolean"
357
+
358
+ parameters["properties"][name] = {"type": param_type_name}
359
+ if param.default == inspect.Parameter.empty:
360
+ parameters["required"].append(name)
361
+
362
+ return {
363
+ "type": "function",
364
+ "function": {
365
+ "name": func.__name__,
366
+ "description": func.__doc__ or "",
367
+ "parameters": parameters,
368
+ },
369
+ }
370
+
371
+
372
+ async def process_chat(chat, provider_id=None):
286
373
  if not chat:
287
374
  raise Exception("No chat provided")
288
- if 'stream' not in chat:
289
- chat['stream'] = False
290
- if 'messages' not in chat:
375
+ if "stream" not in chat:
376
+ chat["stream"] = False
377
+ # Some providers don't support empty tools
378
+ if "tools" in chat and len(chat["tools"]) == 0:
379
+ del chat["tools"]
380
+ if "messages" not in chat:
291
381
  return chat
292
382
 
293
383
  async with aiohttp.ClientSession() as session:
294
- for message in chat['messages']:
295
- if 'content' not in message:
384
+ for message in chat["messages"]:
385
+ if "content" not in message:
296
386
  continue
297
387
 
298
- if isinstance(message['content'], list):
299
- for item in message['content']:
300
- if 'type' not in item:
388
+ if isinstance(message["content"], list):
389
+ for item in message["content"]:
390
+ if "type" not in item:
301
391
  continue
302
- if item['type'] == 'image_url' and 'image_url' in item:
303
- image_url = item['image_url']
304
- if 'url' in image_url:
305
- url = image_url['url']
392
+ if item["type"] == "image_url" and "image_url" in item:
393
+ image_url = item["image_url"]
394
+ if "url" in image_url:
395
+ url = image_url["url"]
396
+ if url.startswith("/~cache/"):
397
+ url = get_cache_path(url[8:])
306
398
  if is_url(url):
307
399
  _log(f"Downloading image: {url}")
308
400
  async with session.get(url, timeout=aiohttp.ClientTimeout(total=120)) as response:
@@ -310,12 +402,14 @@ async def process_chat(chat):
310
402
  content = await response.read()
311
403
  # get mimetype from response headers
312
404
  mimetype = get_file_mime_type(get_filename(url))
313
- if 'Content-Type' in response.headers:
314
- mimetype = response.headers['Content-Type']
405
+ if "Content-Type" in response.headers:
406
+ mimetype = response.headers["Content-Type"]
315
407
  # convert/resize image if needed
316
408
  content, mimetype = convert_image_if_needed(content, mimetype)
317
409
  # convert to data uri
318
- image_url['url'] = f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
410
+ image_url["url"] = (
411
+ f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
412
+ )
319
413
  elif is_file_path(url):
320
414
  _log(f"Reading image: {url}")
321
415
  with open(url, "rb") as f:
@@ -325,24 +419,30 @@ async def process_chat(chat):
325
419
  # convert/resize image if needed
326
420
  content, mimetype = convert_image_if_needed(content, mimetype)
327
421
  # convert to data uri
328
- image_url['url'] = f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
329
- elif url.startswith('data:'):
422
+ image_url["url"] = (
423
+ f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
424
+ )
425
+ elif url.startswith("data:"):
330
426
  # Extract existing data URI and process it
331
- if ';base64,' in url:
332
- prefix = url.split(';base64,')[0]
333
- mimetype = prefix.split(':')[1] if ':' in prefix else 'image/png'
334
- base64_data = url.split(';base64,')[1]
427
+ if ";base64," in url:
428
+ prefix = url.split(";base64,")[0]
429
+ mimetype = prefix.split(":")[1] if ":" in prefix else "image/png"
430
+ base64_data = url.split(";base64,")[1]
335
431
  content = base64.b64decode(base64_data)
336
432
  # convert/resize image if needed
337
433
  content, mimetype = convert_image_if_needed(content, mimetype)
338
434
  # update data uri with potentially converted image
339
- image_url['url'] = f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
435
+ image_url["url"] = (
436
+ f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
437
+ )
340
438
  else:
341
439
  raise Exception(f"Invalid image: {url}")
342
- elif item['type'] == 'input_audio' and 'input_audio' in item:
343
- input_audio = item['input_audio']
344
- if 'data' in input_audio:
345
- url = input_audio['data']
440
+ elif item["type"] == "input_audio" and "input_audio" in item:
441
+ input_audio = item["input_audio"]
442
+ if "data" in input_audio:
443
+ url = input_audio["data"]
444
+ if url.startswith("/~cache/"):
445
+ url = get_cache_path(url[8:])
346
446
  mimetype = get_file_mime_type(get_filename(url))
347
447
  if is_url(url):
348
448
  _log(f"Downloading audio: {url}")
@@ -350,48 +450,145 @@ async def process_chat(chat):
350
450
  response.raise_for_status()
351
451
  content = await response.read()
352
452
  # get mimetype from response headers
353
- if 'Content-Type' in response.headers:
354
- mimetype = response.headers['Content-Type']
453
+ if "Content-Type" in response.headers:
454
+ mimetype = response.headers["Content-Type"]
355
455
  # convert to base64
356
- input_audio['data'] = base64.b64encode(content).decode('utf-8')
357
- input_audio['format'] = mimetype.rsplit('/',1)[1]
456
+ input_audio["data"] = base64.b64encode(content).decode("utf-8")
457
+ if provider_id == "alibaba":
458
+ input_audio["data"] = f"data:{mimetype};base64,{input_audio['data']}"
459
+ input_audio["format"] = mimetype.rsplit("/", 1)[1]
358
460
  elif is_file_path(url):
359
461
  _log(f"Reading audio: {url}")
360
462
  with open(url, "rb") as f:
361
463
  content = f.read()
362
464
  # convert to base64
363
- input_audio['data'] = base64.b64encode(content).decode('utf-8')
364
- input_audio['format'] = mimetype.rsplit('/',1)[1]
465
+ input_audio["data"] = base64.b64encode(content).decode("utf-8")
466
+ if provider_id == "alibaba":
467
+ input_audio["data"] = f"data:{mimetype};base64,{input_audio['data']}"
468
+ input_audio["format"] = mimetype.rsplit("/", 1)[1]
365
469
  elif is_base_64(url):
366
- pass # use base64 data as-is
470
+ pass # use base64 data as-is
367
471
  else:
368
472
  raise Exception(f"Invalid audio: {url}")
369
- elif item['type'] == 'file' and 'file' in item:
370
- file = item['file']
371
- if 'file_data' in file:
372
- url = file['file_data']
473
+ elif item["type"] == "file" and "file" in item:
474
+ file = item["file"]
475
+ if "file_data" in file:
476
+ url = file["file_data"]
477
+ if url.startswith("/~cache/"):
478
+ url = get_cache_path(url[8:])
373
479
  mimetype = get_file_mime_type(get_filename(url))
374
480
  if is_url(url):
375
481
  _log(f"Downloading file: {url}")
376
482
  async with session.get(url, timeout=aiohttp.ClientTimeout(total=120)) as response:
377
483
  response.raise_for_status()
378
484
  content = await response.read()
379
- file['filename'] = get_filename(url)
380
- file['file_data'] = f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
485
+ file["filename"] = get_filename(url)
486
+ file["file_data"] = (
487
+ f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
488
+ )
381
489
  elif is_file_path(url):
382
490
  _log(f"Reading file: {url}")
383
491
  with open(url, "rb") as f:
384
492
  content = f.read()
385
- file['filename'] = get_filename(url)
386
- file['file_data'] = f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
387
- elif url.startswith('data:'):
388
- if 'filename' not in file:
389
- file['filename'] = 'file'
390
- pass # use base64 data as-is
493
+ file["filename"] = get_filename(url)
494
+ file["file_data"] = (
495
+ f"data:{mimetype};base64,{base64.b64encode(content).decode('utf-8')}"
496
+ )
497
+ elif url.startswith("data:"):
498
+ if "filename" not in file:
499
+ file["filename"] = "file"
500
+ pass # use base64 data as-is
391
501
  else:
392
502
  raise Exception(f"Invalid file: {url}")
393
503
  return chat
394
504
 
505
+
506
+ def image_ext_from_mimetype(mimetype, default="png"):
507
+ if "/" in mimetype:
508
+ _ext = mimetypes.guess_extension(mimetype)
509
+ if _ext:
510
+ return _ext.lstrip(".")
511
+ return default
512
+
513
+
514
+ def audio_ext_from_format(format, default="mp3"):
515
+ if format == "mpeg":
516
+ return "mp3"
517
+ return format or default
518
+
519
+
520
+ def file_ext_from_mimetype(mimetype, default="pdf"):
521
+ if "/" in mimetype:
522
+ _ext = mimetypes.guess_extension(mimetype)
523
+ if _ext:
524
+ return _ext.lstrip(".")
525
+ return default
526
+
527
+
528
+ def cache_message_inline_data(m):
529
+ """
530
+ Replaces and caches any inline data URIs in the message content.
531
+ """
532
+ if "content" not in m:
533
+ return
534
+
535
+ content = m["content"]
536
+ if isinstance(content, list):
537
+ for item in content:
538
+ if item.get("type") == "image_url":
539
+ image_url = item.get("image_url", {})
540
+ url = image_url.get("url")
541
+ if url and url.startswith("data:"):
542
+ # Extract base64 and mimetype
543
+ try:
544
+ header, base64_data = url.split(";base64,")
545
+ # header is like "data:image/png"
546
+ ext = image_ext_from_mimetype(header.split(":")[1])
547
+ filename = f"image.{ext}" # Hash will handle uniqueness
548
+
549
+ cache_url, _ = save_image_to_cache(base64_data, filename, {}, ignore_info=True)
550
+ image_url["url"] = cache_url
551
+ except Exception as e:
552
+ _log(f"Error caching inline image: {e}")
553
+
554
+ elif item.get("type") == "input_audio":
555
+ input_audio = item.get("input_audio", {})
556
+ data = input_audio.get("data")
557
+ if data:
558
+ # Handle data URI or raw base64
559
+ base64_data = data
560
+ if data.startswith("data:"):
561
+ with contextlib.suppress(ValueError):
562
+ header, base64_data = data.split(";base64,")
563
+
564
+ fmt = audio_ext_from_format(input_audio.get("format"))
565
+ filename = f"audio.{fmt}"
566
+
567
+ try:
568
+ cache_url, _ = save_bytes_to_cache(base64_data, filename, {}, ignore_info=True)
569
+ input_audio["data"] = cache_url
570
+ except Exception as e:
571
+ _log(f"Error caching inline audio: {e}")
572
+
573
+ elif item.get("type") == "file":
574
+ file_info = item.get("file", {})
575
+ file_data = file_info.get("file_data")
576
+ if file_data and file_data.startswith("data:"):
577
+ try:
578
+ header, base64_data = file_data.split(";base64,")
579
+ mimetype = header.split(":")[1]
580
+ # Try to get extension from filename if available, else mimetype
581
+ filename = file_info.get("filename", "file")
582
+ if "." not in filename:
583
+ ext = file_ext_from_mimetype(mimetype)
584
+ filename = f"{filename}.{ext}"
585
+
586
+ cache_url, _ = save_bytes_to_cache(base64_data, filename, {}, ignore_info=True)
587
+ file_info["file_data"] = cache_url
588
+ except Exception as e:
589
+ _log(f"Error caching inline file: {e}")
590
+
591
+
395
592
  class HTTPError(Exception):
396
593
  def __init__(self, status, reason, body, headers=None):
397
594
  self.status = status
@@ -400,448 +597,923 @@ class HTTPError(Exception):
400
597
  self.headers = headers
401
598
  super().__init__(f"HTTP {status} {reason}")
402
599
 
600
+
601
+ def save_bytes_to_cache(base64_data, filename, file_info, ignore_info=False):
602
+ ext = filename.split(".")[-1]
603
+ mimetype = get_file_mime_type(filename)
604
+ content = base64.b64decode(base64_data) if isinstance(base64_data, str) else base64_data
605
+ sha256_hash = hashlib.sha256(content).hexdigest()
606
+
607
+ save_filename = f"{sha256_hash}.{ext}" if ext else sha256_hash
608
+
609
+ # Use first 2 chars for subdir to avoid too many files in one dir
610
+ subdir = sha256_hash[:2]
611
+ relative_path = f"{subdir}/{save_filename}"
612
+ full_path = get_cache_path(relative_path)
613
+ url = f"/~cache/{relative_path}"
614
+
615
+ # if file and its .info.json already exists, return it
616
+ info_path = os.path.splitext(full_path)[0] + ".info.json"
617
+ if os.path.exists(full_path) and os.path.exists(info_path):
618
+ _dbg(f"Cached bytes exists: {relative_path}")
619
+ if ignore_info:
620
+ return url, None
621
+ return url, json.load(open(info_path))
622
+
623
+ os.makedirs(os.path.dirname(full_path), exist_ok=True)
624
+
625
+ with open(full_path, "wb") as f:
626
+ f.write(content)
627
+ info = {
628
+ "date": int(time.time()),
629
+ "url": url,
630
+ "size": len(content),
631
+ "type": mimetype,
632
+ "name": filename,
633
+ }
634
+ info.update(file_info)
635
+
636
+ # Save metadata
637
+ info_path = os.path.splitext(full_path)[0] + ".info.json"
638
+ with open(info_path, "w") as f:
639
+ json.dump(info, f)
640
+
641
+ _dbg(f"Saved cached bytes and info: {relative_path}")
642
+
643
+ g_app.on_cache_saved_filters({"url": url, "info": info})
644
+
645
+ return url, info
646
+
647
+
648
+ def save_image_to_cache(base64_data, filename, image_info, ignore_info=False):
649
+ ext = filename.split(".")[-1]
650
+ mimetype = get_file_mime_type(filename)
651
+ content = base64.b64decode(base64_data) if isinstance(base64_data, str) else base64_data
652
+ sha256_hash = hashlib.sha256(content).hexdigest()
653
+
654
+ save_filename = f"{sha256_hash}.{ext}" if ext else sha256_hash
655
+
656
+ # Use first 2 chars for subdir to avoid too many files in one dir
657
+ subdir = sha256_hash[:2]
658
+ relative_path = f"{subdir}/{save_filename}"
659
+ full_path = get_cache_path(relative_path)
660
+ url = f"/~cache/{relative_path}"
661
+
662
+ # if file and its .info.json already exists, return it
663
+ info_path = os.path.splitext(full_path)[0] + ".info.json"
664
+ if os.path.exists(full_path) and os.path.exists(info_path):
665
+ _dbg(f"Saved image exists: {relative_path}")
666
+ if ignore_info:
667
+ return url, None
668
+ return url, json.load(open(info_path))
669
+
670
+ os.makedirs(os.path.dirname(full_path), exist_ok=True)
671
+
672
+ with open(full_path, "wb") as f:
673
+ f.write(content)
674
+ info = {
675
+ "date": int(time.time()),
676
+ "url": url,
677
+ "size": len(content),
678
+ "type": mimetype,
679
+ "name": filename,
680
+ }
681
+ info.update(image_info)
682
+
683
+ # If image, get dimensions
684
+ if HAS_PIL and mimetype.startswith("image/"):
685
+ try:
686
+ with Image.open(BytesIO(content)) as img:
687
+ info["width"] = img.width
688
+ info["height"] = img.height
689
+ except Exception:
690
+ pass
691
+
692
+ if "width" in info and "height" in info:
693
+ _log(f"Saved image to cache: {full_path} ({len(content)} bytes) {info['width']}x{info['height']}")
694
+ else:
695
+ _log(f"Saved image to cache: {full_path} ({len(content)} bytes)")
696
+
697
+ # Save metadata
698
+ info_path = os.path.splitext(full_path)[0] + ".info.json"
699
+ with open(info_path, "w") as f:
700
+ json.dump(info, f)
701
+
702
+ _dbg(f"Saved image and info: {relative_path}")
703
+
704
+ g_app.on_cache_saved_filters({"url": url, "info": info})
705
+
706
+ return url, info
707
+
708
+
403
709
  async def response_json(response):
404
710
  text = await response.text()
405
711
  if response.status >= 400:
712
+ _dbg(f"HTTP {response.status} {response.reason}: {text}")
406
713
  raise HTTPError(response.status, reason=response.reason, body=text, headers=dict(response.headers))
407
714
  response.raise_for_status()
408
715
  body = json.loads(text)
409
716
  return body
410
717
 
411
- class OpenAiProvider:
412
- def __init__(self, base_url, api_key=None, models={}, **kwargs):
413
- self.base_url = base_url.strip("/")
414
- self.api_key = api_key
415
- self.models = models
416
718
 
417
- # check if base_url ends with /v{\d} to handle providers with different versions (e.g. z.ai uses /v4)
418
- last_segment = base_url.rsplit('/',1)[1]
419
- if last_segment.startswith('v') and last_segment[1:].isdigit():
420
- self.chat_url = f"{base_url}/chat/completions"
421
- else:
422
- self.chat_url = f"{base_url}/v1/chat/completions"
719
+ def chat_to_prompt(chat):
720
+ prompt = ""
721
+ if "messages" in chat:
722
+ for message in chat["messages"]:
723
+ if message["role"] == "user":
724
+ # if content is string
725
+ if isinstance(message["content"], str):
726
+ if prompt:
727
+ prompt += "\n"
728
+ prompt += message["content"]
729
+ elif isinstance(message["content"], list):
730
+ # if content is array of objects
731
+ for part in message["content"]:
732
+ if part["type"] == "text":
733
+ if prompt:
734
+ prompt += "\n"
735
+ prompt += part["text"]
736
+ return prompt
737
+
738
+
739
+ def chat_to_system_prompt(chat):
740
+ if "messages" in chat:
741
+ for message in chat["messages"]:
742
+ if message["role"] == "system":
743
+ # if content is string
744
+ if isinstance(message["content"], str):
745
+ return message["content"]
746
+ elif isinstance(message["content"], list):
747
+ # if content is array of objects
748
+ for part in message["content"]:
749
+ if part["type"] == "text":
750
+ return part["text"]
751
+ return None
752
+
753
+
754
+ def chat_to_username(chat):
755
+ if "metadata" in chat and "user" in chat["metadata"]:
756
+ return chat["metadata"]["user"]
757
+ return None
758
+
759
+
760
+ def last_user_prompt(chat):
761
+ prompt = ""
762
+ if "messages" in chat:
763
+ for message in chat["messages"]:
764
+ if message["role"] == "user":
765
+ # if content is string
766
+ if isinstance(message["content"], str):
767
+ prompt = message["content"]
768
+ elif isinstance(message["content"], list):
769
+ # if content is array of objects
770
+ for part in message["content"]:
771
+ if part["type"] == "text":
772
+ prompt = part["text"]
773
+ return prompt
774
+
775
+
776
+ def chat_response_to_message(openai_response):
777
+ """
778
+ Returns an assistant message from the OpenAI Response.
779
+ Handles normalizing text, image, and audio responses into the message content.
780
+ """
781
+ timestamp = int(time.time() * 1000) # openai_response.get("created")
782
+ choices = openai_response
783
+ if isinstance(openai_response, dict) and "choices" in openai_response:
784
+ choices = openai_response["choices"]
785
+
786
+ choice = choices[0] if isinstance(choices, list) and choices else choices
787
+
788
+ if isinstance(choice, str):
789
+ return {"role": "assistant", "content": choice, "timestamp": timestamp}
790
+
791
+ if isinstance(choice, dict):
792
+ message = choice.get("message", choice)
793
+ else:
794
+ return {"role": "assistant", "content": str(choice), "timestamp": timestamp}
795
+
796
+ # Ensure message is a dict
797
+ if not isinstance(message, dict):
798
+ return {"role": "assistant", "content": message, "timestamp": timestamp}
799
+
800
+ message.update({"timestamp": timestamp})
801
+ return message
802
+
803
+
804
+ def to_file_info(chat, info=None, response=None):
805
+ prompt = last_user_prompt(chat)
806
+ ret = info or {}
807
+ if chat["model"] and "model" not in ret:
808
+ ret["model"] = chat["model"]
809
+ if prompt and "prompt" not in ret:
810
+ ret["prompt"] = prompt
811
+ if "image_config" in chat:
812
+ ret.update(chat["image_config"])
813
+ user = chat_to_username(chat)
814
+ if user:
815
+ ret["user"] = user
816
+ return ret
817
+
423
818
 
424
- self.headers = kwargs['headers'] if 'headers' in kwargs else {
819
+ # Image Generator Providers
820
+ class GeneratorBase:
821
+ def __init__(self, **kwargs):
822
+ self.id = kwargs.get("id")
823
+ self.api = kwargs.get("api")
824
+ self.api_key = kwargs.get("api_key")
825
+ self.headers = {
826
+ "Accept": "application/json",
425
827
  "Content-Type": "application/json",
426
828
  }
427
- if api_key is not None:
428
- self.headers["Authorization"] = f"Bearer {api_key}"
429
-
430
- self.frequency_penalty = float(kwargs['frequency_penalty']) if 'frequency_penalty' in kwargs else None
431
- self.max_completion_tokens = int(kwargs['max_completion_tokens']) if 'max_completion_tokens' in kwargs else None
432
- self.n = int(kwargs['n']) if 'n' in kwargs else None
433
- self.parallel_tool_calls = bool(kwargs['parallel_tool_calls']) if 'parallel_tool_calls' in kwargs else None
434
- self.presence_penalty = float(kwargs['presence_penalty']) if 'presence_penalty' in kwargs else None
435
- self.prompt_cache_key = kwargs['prompt_cache_key'] if 'prompt_cache_key' in kwargs else None
436
- self.reasoning_effort = kwargs['reasoning_effort'] if 'reasoning_effort' in kwargs else None
437
- self.safety_identifier = kwargs['safety_identifier'] if 'safety_identifier' in kwargs else None
438
- self.seed = int(kwargs['seed']) if 'seed' in kwargs else None
439
- self.service_tier = kwargs['service_tier'] if 'service_tier' in kwargs else None
440
- self.stop = kwargs['stop'] if 'stop' in kwargs else None
441
- self.store = bool(kwargs['store']) if 'store' in kwargs else None
442
- self.temperature = float(kwargs['temperature']) if 'temperature' in kwargs else None
443
- self.top_logprobs = int(kwargs['top_logprobs']) if 'top_logprobs' in kwargs else None
444
- self.top_p = float(kwargs['top_p']) if 'top_p' in kwargs else None
445
- self.verbosity = kwargs['verbosity'] if 'verbosity' in kwargs else None
446
- self.stream = bool(kwargs['stream']) if 'stream' in kwargs else None
447
- self.enable_thinking = bool(kwargs['enable_thinking']) if 'enable_thinking' in kwargs else None
448
- self.pricing = kwargs['pricing'] if 'pricing' in kwargs else None
449
- self.default_pricing = kwargs['default_pricing'] if 'default_pricing' in kwargs else None
450
- self.check = kwargs['check'] if 'check' in kwargs else None
451
-
452
- @classmethod
453
- def test(cls, base_url=None, api_key=None, models={}, **kwargs):
454
- return base_url and api_key and len(models) > 0
829
+ self.chat_url = f"{self.api}/chat/completions"
830
+ self.default_content = "I've generated the image for you."
831
+
832
+ def validate(self, **kwargs):
833
+ if not self.api_key:
834
+ api_keys = ", ".join(self.env)
835
+ return f"Provider '{self.name}' requires API Key {api_keys}"
836
+ return None
837
+
838
+ def test(self, **kwargs):
839
+ error_msg = self.validate(**kwargs)
840
+ if error_msg:
841
+ _log(error_msg)
842
+ return False
843
+ return True
455
844
 
456
845
  async def load(self):
457
846
  pass
458
847
 
459
- def model_pricing(self, model):
848
+ def gen_summary(self, gen):
849
+ """Summarize gen response for logging."""
850
+ clone = json.loads(json.dumps(gen))
851
+ return json.dumps(clone, indent=2)
852
+
853
+ def chat_summary(self, chat):
854
+ return chat_summary(chat)
855
+
856
+ def process_chat(self, chat, provider_id=None):
857
+ return process_chat(chat, provider_id)
858
+
859
+ async def response_json(self, response):
860
+ return await response_json(response)
861
+
862
+ def get_headers(self, provider, chat):
863
+ headers = self.headers.copy()
864
+ if provider is not None:
865
+ headers["Authorization"] = f"Bearer {provider.api_key}"
866
+ elif self.api_key:
867
+ headers["Authorization"] = f"Bearer {self.api_key}"
868
+ return headers
869
+
870
+ def to_response(self, response, chat, started_at):
871
+ raise NotImplementedError
872
+
873
+ async def chat(self, chat, provider=None):
874
+ return {
875
+ "choices": [
876
+ {
877
+ "message": {
878
+ "role": "assistant",
879
+ "content": "Not Implemented",
880
+ "images": [
881
+ {
882
+ "type": "image_url",
883
+ "image_url": {
884
+ "url": "",
885
+ },
886
+ }
887
+ ],
888
+ }
889
+ }
890
+ ]
891
+ }
892
+
893
+
894
+ # OpenAI Providers
895
+
896
+
897
+ class OpenAiCompatible:
898
+ sdk = "@ai-sdk/openai-compatible"
899
+
900
+ def __init__(self, **kwargs):
901
+ required_args = ["id", "api"]
902
+ for arg in required_args:
903
+ if arg not in kwargs:
904
+ raise ValueError(f"Missing required argument: {arg}")
905
+
906
+ self.id = kwargs.get("id")
907
+ self.api = kwargs.get("api").strip("/")
908
+ self.env = kwargs.get("env", [])
909
+ self.api_key = kwargs.get("api_key")
910
+ self.name = kwargs.get("name", id_to_name(self.id))
911
+ self.set_models(**kwargs)
912
+
913
+ self.chat_url = f"{self.api}/chat/completions"
914
+
915
+ self.headers = kwargs.get("headers", {"Content-Type": "application/json"})
916
+ if self.api_key is not None:
917
+ self.headers["Authorization"] = f"Bearer {self.api_key}"
918
+
919
+ self.frequency_penalty = float(kwargs["frequency_penalty"]) if "frequency_penalty" in kwargs else None
920
+ self.max_completion_tokens = int(kwargs["max_completion_tokens"]) if "max_completion_tokens" in kwargs else None
921
+ self.n = int(kwargs["n"]) if "n" in kwargs else None
922
+ self.parallel_tool_calls = bool(kwargs["parallel_tool_calls"]) if "parallel_tool_calls" in kwargs else None
923
+ self.presence_penalty = float(kwargs["presence_penalty"]) if "presence_penalty" in kwargs else None
924
+ self.prompt_cache_key = kwargs.get("prompt_cache_key")
925
+ self.reasoning_effort = kwargs.get("reasoning_effort")
926
+ self.safety_identifier = kwargs.get("safety_identifier")
927
+ self.seed = int(kwargs["seed"]) if "seed" in kwargs else None
928
+ self.service_tier = kwargs.get("service_tier")
929
+ self.stop = kwargs.get("stop")
930
+ self.store = bool(kwargs["store"]) if "store" in kwargs else None
931
+ self.temperature = float(kwargs["temperature"]) if "temperature" in kwargs else None
932
+ self.top_logprobs = int(kwargs["top_logprobs"]) if "top_logprobs" in kwargs else None
933
+ self.top_p = float(kwargs["top_p"]) if "top_p" in kwargs else None
934
+ self.verbosity = kwargs.get("verbosity")
935
+ self.stream = bool(kwargs["stream"]) if "stream" in kwargs else None
936
+ self.enable_thinking = bool(kwargs["enable_thinking"]) if "enable_thinking" in kwargs else None
937
+ self.check = kwargs.get("check")
938
+ self.modalities = kwargs.get("modalities", {})
939
+
940
+ def set_models(self, **kwargs):
941
+ models = kwargs.get("models", {})
942
+ self.map_models = kwargs.get("map_models", {})
943
+ # if 'map_models' is provided, only include models in `map_models[model_id] = provider_model_id`
944
+ if self.map_models:
945
+ self.models = {}
946
+ for provider_model_id in self.map_models.values():
947
+ if provider_model_id in models:
948
+ self.models[provider_model_id] = models[provider_model_id]
949
+ else:
950
+ self.models = models
951
+
952
+ include_models = kwargs.get("include_models") # string regex pattern
953
+ # only include models that match the regex pattern
954
+ if include_models:
955
+ _log(f"Filtering {len(self.models)} models, only including models that match regex: {include_models}")
956
+ self.models = {k: v for k, v in self.models.items() if re.search(include_models, k)}
957
+
958
+ exclude_models = kwargs.get("exclude_models") # string regex pattern
959
+ # exclude models that match the regex pattern
960
+ if exclude_models:
961
+ _log(f"Filtering {len(self.models)} models, excluding models that match regex: {exclude_models}")
962
+ self.models = {k: v for k, v in self.models.items() if not re.search(exclude_models, k)}
963
+
964
+ def validate(self, **kwargs):
965
+ if not self.api_key:
966
+ api_keys = ", ".join(self.env)
967
+ return f"Provider '{self.name}' requires API Key {api_keys}"
968
+ return None
969
+
970
+ def test(self, **kwargs):
971
+ error_msg = self.validate(**kwargs)
972
+ if error_msg:
973
+ _log(error_msg)
974
+ return False
975
+ return True
976
+
977
+ async def load(self):
978
+ if not self.models:
979
+ await self.load_models()
980
+
981
+ def model_info(self, model):
460
982
  provider_model = self.provider_model(model) or model
461
- if self.pricing and provider_model in self.pricing:
462
- return self.pricing[provider_model]
463
- return self.default_pricing or None
983
+ for model_id, model_info in self.models.items():
984
+ if model_id.lower() == provider_model.lower():
985
+ return model_info
986
+ return None
987
+
988
+ def model_cost(self, model):
989
+ model_info = self.model_info(model)
990
+ return model_info.get("cost") if model_info else None
464
991
 
465
992
  def provider_model(self, model):
466
- if model in self.models:
467
- return self.models[model]
993
+ # convert model to lowercase for case-insensitive comparison
994
+ model_lower = model.lower()
995
+
996
+ # if model is a map model id, return the provider model id
997
+ for model_id, provider_model in self.map_models.items():
998
+ if model_id.lower() == model_lower:
999
+ return provider_model
1000
+
1001
+ # if model is a provider model id, try again with just the model name
1002
+ for provider_model in self.map_models.values():
1003
+ if provider_model.lower() == model_lower:
1004
+ return provider_model
1005
+
1006
+ # if model is a model id, try again with just the model id or name
1007
+ for model_id, provider_model_info in self.models.items():
1008
+ id = provider_model_info.get("id") or model_id
1009
+ if model_id.lower() == model_lower or id.lower() == model_lower:
1010
+ return id
1011
+ name = provider_model_info.get("name")
1012
+ if name and name.lower() == model_lower:
1013
+ return id
1014
+
1015
+ # fallback to trying again with just the model short name
1016
+ for model_id, provider_model_info in self.models.items():
1017
+ id = provider_model_info.get("id") or model_id
1018
+ if "/" in id:
1019
+ model_name = id.split("/")[-1]
1020
+ if model_name.lower() == model_lower:
1021
+ return id
1022
+
1023
+ # if model is a full provider model id, try again with just the model name
1024
+ if "/" in model:
1025
+ last_part = model.split("/")[-1]
1026
+ return self.provider_model(last_part)
1027
+
468
1028
  return None
469
1029
 
1030
+ def response_json(self, response):
1031
+ return response_json(response)
1032
+
470
1033
  def to_response(self, response, chat, started_at):
471
- if 'metadata' not in response:
472
- response['metadata'] = {}
473
- response['metadata']['duration'] = int((time.time() - started_at) * 1000)
474
- if chat is not None and 'model' in chat:
475
- pricing = self.model_pricing(chat['model'])
476
- if pricing and 'input' in pricing and 'output' in pricing:
477
- response['metadata']['pricing'] = f"{pricing['input']}/{pricing['output']}"
478
- _log(json.dumps(response, indent=2))
1034
+ if "metadata" not in response:
1035
+ response["metadata"] = {}
1036
+ response["metadata"]["duration"] = int((time.time() - started_at) * 1000)
1037
+ if chat is not None and "model" in chat:
1038
+ pricing = self.model_cost(chat["model"])
1039
+ if pricing and "input" in pricing and "output" in pricing:
1040
+ response["metadata"]["pricing"] = f"{pricing['input']}/{pricing['output']}"
479
1041
  return response
480
1042
 
1043
+ def chat_summary(self, chat):
1044
+ return chat_summary(chat)
1045
+
1046
+ def process_chat(self, chat, provider_id=None):
1047
+ return process_chat(chat, provider_id)
1048
+
481
1049
  async def chat(self, chat):
482
- chat['model'] = self.provider_model(chat['model']) or chat['model']
1050
+ chat["model"] = self.provider_model(chat["model"]) or chat["model"]
1051
+
1052
+ if "modalities" in chat:
1053
+ for modality in chat.get("modalities", []):
1054
+ # use default implementation for text modalities
1055
+ if modality == "text":
1056
+ continue
1057
+ modality_provider = self.modalities.get(modality)
1058
+ if modality_provider:
1059
+ return await modality_provider.chat(chat, self)
1060
+ else:
1061
+ raise Exception(f"Provider {self.name} does not support '{modality}' modality")
483
1062
 
484
1063
  # with open(os.path.join(os.path.dirname(__file__), 'chat.wip.json'), "w") as f:
485
1064
  # f.write(json.dumps(chat, indent=2))
486
1065
 
487
1066
  if self.frequency_penalty is not None:
488
- chat['frequency_penalty'] = self.frequency_penalty
1067
+ chat["frequency_penalty"] = self.frequency_penalty
489
1068
  if self.max_completion_tokens is not None:
490
- chat['max_completion_tokens'] = self.max_completion_tokens
1069
+ chat["max_completion_tokens"] = self.max_completion_tokens
491
1070
  if self.n is not None:
492
- chat['n'] = self.n
1071
+ chat["n"] = self.n
493
1072
  if self.parallel_tool_calls is not None:
494
- chat['parallel_tool_calls'] = self.parallel_tool_calls
1073
+ chat["parallel_tool_calls"] = self.parallel_tool_calls
495
1074
  if self.presence_penalty is not None:
496
- chat['presence_penalty'] = self.presence_penalty
1075
+ chat["presence_penalty"] = self.presence_penalty
497
1076
  if self.prompt_cache_key is not None:
498
- chat['prompt_cache_key'] = self.prompt_cache_key
1077
+ chat["prompt_cache_key"] = self.prompt_cache_key
499
1078
  if self.reasoning_effort is not None:
500
- chat['reasoning_effort'] = self.reasoning_effort
1079
+ chat["reasoning_effort"] = self.reasoning_effort
501
1080
  if self.safety_identifier is not None:
502
- chat['safety_identifier'] = self.safety_identifier
1081
+ chat["safety_identifier"] = self.safety_identifier
503
1082
  if self.seed is not None:
504
- chat['seed'] = self.seed
1083
+ chat["seed"] = self.seed
505
1084
  if self.service_tier is not None:
506
- chat['service_tier'] = self.service_tier
1085
+ chat["service_tier"] = self.service_tier
507
1086
  if self.stop is not None:
508
- chat['stop'] = self.stop
1087
+ chat["stop"] = self.stop
509
1088
  if self.store is not None:
510
- chat['store'] = self.store
1089
+ chat["store"] = self.store
511
1090
  if self.temperature is not None:
512
- chat['temperature'] = self.temperature
1091
+ chat["temperature"] = self.temperature
513
1092
  if self.top_logprobs is not None:
514
- chat['top_logprobs'] = self.top_logprobs
1093
+ chat["top_logprobs"] = self.top_logprobs
515
1094
  if self.top_p is not None:
516
- chat['top_p'] = self.top_p
1095
+ chat["top_p"] = self.top_p
517
1096
  if self.verbosity is not None:
518
- chat['verbosity'] = self.verbosity
1097
+ chat["verbosity"] = self.verbosity
519
1098
  if self.enable_thinking is not None:
520
- chat['enable_thinking'] = self.enable_thinking
1099
+ chat["enable_thinking"] = self.enable_thinking
521
1100
 
522
- chat = await process_chat(chat)
1101
+ chat = await process_chat(chat, provider_id=self.id)
523
1102
  _log(f"POST {self.chat_url}")
524
1103
  _log(chat_summary(chat))
525
1104
  # remove metadata if any (conflicts with some providers, e.g. Z.ai)
526
- chat.pop('metadata', None)
1105
+ metadata = chat.pop("metadata", None)
527
1106
 
528
1107
  async with aiohttp.ClientSession() as session:
529
1108
  started_at = time.time()
530
- async with session.post(self.chat_url, headers=self.headers, data=json.dumps(chat), timeout=aiohttp.ClientTimeout(total=120)) as response:
1109
+ async with session.post(
1110
+ self.chat_url, headers=self.headers, data=json.dumps(chat), timeout=aiohttp.ClientTimeout(total=120)
1111
+ ) as response:
1112
+ chat["metadata"] = metadata
531
1113
  return self.to_response(await response_json(response), chat, started_at)
532
1114
 
533
- class OllamaProvider(OpenAiProvider):
534
- def __init__(self, base_url, models, all_models=False, **kwargs):
535
- super().__init__(base_url=base_url, models=models, **kwargs)
536
- self.all_models = all_models
1115
+
1116
+ class MistralProvider(OpenAiCompatible):
1117
+ sdk = "@ai-sdk/mistral"
1118
+
1119
+ def __init__(self, **kwargs):
1120
+ if "api" not in kwargs:
1121
+ kwargs["api"] = "https://api.mistral.ai/v1"
1122
+ super().__init__(**kwargs)
1123
+
1124
+
1125
+ class GroqProvider(OpenAiCompatible):
1126
+ sdk = "@ai-sdk/groq"
1127
+
1128
+ def __init__(self, **kwargs):
1129
+ if "api" not in kwargs:
1130
+ kwargs["api"] = "https://api.groq.com/openai/v1"
1131
+ super().__init__(**kwargs)
1132
+
1133
+
1134
+ class XaiProvider(OpenAiCompatible):
1135
+ sdk = "@ai-sdk/xai"
1136
+
1137
+ def __init__(self, **kwargs):
1138
+ if "api" not in kwargs:
1139
+ kwargs["api"] = "https://api.x.ai/v1"
1140
+ super().__init__(**kwargs)
1141
+
1142
+
1143
+ class CodestralProvider(OpenAiCompatible):
1144
+ sdk = "codestral"
1145
+
1146
+ def __init__(self, **kwargs):
1147
+ super().__init__(**kwargs)
1148
+
1149
+
1150
+ class OllamaProvider(OpenAiCompatible):
1151
+ sdk = "ollama"
1152
+
1153
+ def __init__(self, **kwargs):
1154
+ super().__init__(**kwargs)
1155
+ # Ollama's OpenAI-compatible endpoint is at /v1/chat/completions
1156
+ self.chat_url = f"{self.api}/v1/chat/completions"
537
1157
 
538
1158
  async def load(self):
539
- if self.all_models:
540
- await self.load_models(default_models=self.models)
1159
+ if not self.models:
1160
+ await self.load_models()
541
1161
 
542
1162
  async def get_models(self):
543
1163
  ret = {}
544
1164
  try:
545
1165
  async with aiohttp.ClientSession() as session:
546
- _log(f"GET {self.base_url}/api/tags")
547
- async with session.get(f"{self.base_url}/api/tags", headers=self.headers, timeout=aiohttp.ClientTimeout(total=120)) as response:
1166
+ _log(f"GET {self.api}/api/tags")
1167
+ async with session.get(
1168
+ f"{self.api}/api/tags", headers=self.headers, timeout=aiohttp.ClientTimeout(total=120)
1169
+ ) as response:
548
1170
  data = await response_json(response)
549
- for model in data.get('models', []):
550
- name = model['model']
551
- if name.endswith(":latest"):
552
- name = name[:-7]
553
- ret[name] = name
1171
+ for model in data.get("models", []):
1172
+ model_id = model["model"]
1173
+ if model_id.endswith(":latest"):
1174
+ model_id = model_id[:-7]
1175
+ ret[model_id] = model_id
554
1176
  _log(f"Loaded Ollama models: {ret}")
555
1177
  except Exception as e:
556
1178
  _log(f"Error getting Ollama models: {e}")
557
1179
  # return empty dict if ollama is not available
558
1180
  return ret
559
1181
 
560
- async def load_models(self, default_models):
1182
+ async def load_models(self):
561
1183
  """Load models if all_models was requested"""
562
- if self.all_models:
563
- self.models = await self.get_models()
564
- if default_models:
565
- self.models = {**default_models, **self.models}
566
-
567
- @classmethod
568
- def test(cls, base_url=None, models={}, all_models=False, **kwargs):
569
- return base_url and (len(models) > 0 or all_models)
570
-
571
- class GoogleOpenAiProvider(OpenAiProvider):
572
- def __init__(self, api_key, models, **kwargs):
573
- super().__init__(base_url="https://generativelanguage.googleapis.com", api_key=api_key, models=models, **kwargs)
574
- self.chat_url = "https://generativelanguage.googleapis.com/v1beta/chat/completions"
575
-
576
- @classmethod
577
- def test(cls, api_key=None, models={}, **kwargs):
578
- return api_key and len(models) > 0
579
-
580
- class GoogleProvider(OpenAiProvider):
581
- def __init__(self, models, api_key, safety_settings=None, thinking_config=None, curl=False, **kwargs):
582
- super().__init__(base_url="https://generativelanguage.googleapis.com", api_key=api_key, models=models, **kwargs)
583
- self.safety_settings = safety_settings
584
- self.thinking_config = thinking_config
585
- self.curl = curl
586
- self.headers = kwargs['headers'] if 'headers' in kwargs else {
587
- "Content-Type": "application/json",
588
- }
589
- # Google fails when using Authorization header, use query string param instead
590
- if 'Authorization' in self.headers:
591
- del self.headers['Authorization']
592
-
593
- @classmethod
594
- def test(cls, api_key=None, models={}, **kwargs):
595
- return api_key is not None and len(models) > 0
596
1184
 
597
- async def chat(self, chat):
598
- chat['model'] = self.provider_model(chat['model']) or chat['model']
1185
+ # Map models to provider models {model_id:model_id}
1186
+ model_map = await self.get_models()
1187
+ if self.map_models:
1188
+ map_model_values = set(self.map_models.values())
1189
+ to = {}
1190
+ for k, v in model_map.items():
1191
+ if k in self.map_models:
1192
+ to[k] = v
1193
+ if v in map_model_values:
1194
+ to[k] = v
1195
+ model_map = to
1196
+ else:
1197
+ self.map_models = model_map
1198
+ models = {}
1199
+ for k, v in model_map.items():
1200
+ models[k] = {
1201
+ "id": k,
1202
+ "name": v.replace(":", " "),
1203
+ "modalities": {"input": ["text"], "output": ["text"]},
1204
+ "cost": {
1205
+ "input": 0,
1206
+ "output": 0,
1207
+ },
1208
+ }
1209
+ self.models = models
599
1210
 
600
- chat = await process_chat(chat)
601
- generationConfig = {}
1211
+ def validate(self, **kwargs):
1212
+ return None
602
1213
 
603
- # Filter out system messages and convert to proper Gemini format
604
- contents = []
605
- system_prompt = None
606
1214
 
607
- async with aiohttp.ClientSession() as session:
608
- for message in chat['messages']:
609
- if message['role'] == 'system':
610
- content = message['content']
611
- if isinstance(content, list):
612
- for item in content:
613
- if 'text' in item:
614
- system_prompt = item['text']
615
- break
616
- elif isinstance(content, str):
617
- system_prompt = content
618
- elif 'content' in message:
619
- if isinstance(message['content'], list):
620
- parts = []
621
- for item in message['content']:
622
- if 'type' in item:
623
- if item['type'] == 'image_url' and 'image_url' in item:
624
- image_url = item['image_url']
625
- if 'url' not in image_url:
626
- continue
627
- url = image_url['url']
628
- if not url.startswith('data:'):
629
- raise(Exception("Image was not downloaded: " + url))
630
- # Extract mime type from data uri
631
- mimetype = url.split(';',1)[0].split(':',1)[1] if ';' in url else "image/png"
632
- base64Data = url.split(',',1)[1]
633
- parts.append({
634
- "inline_data": {
635
- "mime_type": mimetype,
636
- "data": base64Data
637
- }
638
- })
639
- elif item['type'] == 'input_audio' and 'input_audio' in item:
640
- input_audio = item['input_audio']
641
- if 'data' not in input_audio:
642
- continue
643
- data = input_audio['data']
644
- format = input_audio['format']
645
- mimetype = f"audio/{format}"
646
- parts.append({
647
- "inline_data": {
648
- "mime_type": mimetype,
649
- "data": data
650
- }
651
- })
652
- elif item['type'] == 'file' and 'file' in item:
653
- file = item['file']
654
- if 'file_data' not in file:
655
- continue
656
- data = file['file_data']
657
- if not data.startswith('data:'):
658
- raise(Exception("File was not downloaded: " + data))
659
- # Extract mime type from data uri
660
- mimetype = data.split(';',1)[0].split(':',1)[1] if ';' in data else "application/octet-stream"
661
- base64Data = data.split(',',1)[1]
662
- parts.append({
663
- "inline_data": {
664
- "mime_type": mimetype,
665
- "data": base64Data
666
- }
667
- })
668
- if 'text' in item:
669
- text = item['text']
670
- parts.append({"text": text})
671
- if len(parts) > 0:
672
- contents.append({
673
- "role": message['role'] if 'role' in message and message['role'] == 'user' else 'model',
674
- "parts": parts
675
- })
676
- else:
677
- content = message['content']
678
- contents.append({
679
- "role": message['role'] if 'role' in message and message['role'] == 'user' else 'model',
680
- "parts": [{"text": content}]
681
- })
682
-
683
- gemini_chat = {
684
- "contents": contents,
685
- }
1215
+ class LMStudioProvider(OllamaProvider):
1216
+ sdk = "lmstudio"
686
1217
 
687
- if self.safety_settings:
688
- gemini_chat['safetySettings'] = self.safety_settings
1218
+ def __init__(self, **kwargs):
1219
+ super().__init__(**kwargs)
1220
+ self.chat_url = f"{self.api}/chat/completions"
689
1221
 
690
- # Add system instruction if present
691
- if system_prompt is not None:
692
- gemini_chat['systemInstruction'] = {
693
- "parts": [{"text": system_prompt}]
694
- }
1222
+ async def get_models(self):
1223
+ ret = {}
1224
+ try:
1225
+ async with aiohttp.ClientSession() as session:
1226
+ _log(f"GET {self.api}/models")
1227
+ async with session.get(
1228
+ f"{self.api}/models", headers=self.headers, timeout=aiohttp.ClientTimeout(total=120)
1229
+ ) as response:
1230
+ data = await response_json(response)
1231
+ for model in data.get("data", []):
1232
+ id = model["id"]
1233
+ ret[id] = id
1234
+ _log(f"Loaded LMStudio models: {ret}")
1235
+ except Exception as e:
1236
+ _log(f"Error getting LMStudio models: {e}")
1237
+ # return empty dict if ollama is not available
1238
+ return ret
695
1239
 
696
- if 'max_completion_tokens' in chat:
697
- generationConfig['maxOutputTokens'] = chat['max_completion_tokens']
698
- if 'stop' in chat:
699
- generationConfig['stopSequences'] = [chat['stop']]
700
- if 'temperature' in chat:
701
- generationConfig['temperature'] = chat['temperature']
702
- if 'top_p' in chat:
703
- generationConfig['topP'] = chat['top_p']
704
- if 'top_logprobs' in chat:
705
- generationConfig['topK'] = chat['top_logprobs']
706
-
707
- if 'thinkingConfig' in chat:
708
- generationConfig['thinkingConfig'] = chat['thinkingConfig']
709
- elif self.thinking_config:
710
- generationConfig['thinkingConfig'] = self.thinking_config
711
-
712
- if len(generationConfig) > 0:
713
- gemini_chat['generationConfig'] = generationConfig
714
-
715
- started_at = int(time.time() * 1000)
716
- gemini_chat_url = f"https://generativelanguage.googleapis.com/v1beta/models/{chat['model']}:generateContent?key={self.api_key}"
717
-
718
- _log(f"POST {gemini_chat_url}")
719
- _log(gemini_chat_summary(gemini_chat))
720
- started_at = time.time()
721
1240
 
722
- if self.curl:
723
- curl_args = [
724
- 'curl',
725
- '-X', 'POST',
726
- '-H', 'Content-Type: application/json',
727
- '-d', json.dumps(gemini_chat),
728
- gemini_chat_url
729
- ]
730
- try:
731
- o = subprocess.run(curl_args, check=True, capture_output=True, text=True, timeout=120)
732
- obj = json.loads(o.stdout)
733
- except Exception as e:
734
- raise Exception(f"Error executing curl: {e}")
735
- else:
736
- async with session.post(gemini_chat_url, headers=self.headers, data=json.dumps(gemini_chat), timeout=aiohttp.ClientTimeout(total=120)) as res:
737
- obj = await response_json(res)
738
- _log(f"google response:\n{json.dumps(obj, indent=2)}")
739
-
740
- response = {
741
- "id": f"chatcmpl-{started_at}",
742
- "created": started_at,
743
- "model": obj.get('modelVersion', chat['model']),
744
- }
745
- choices = []
746
- i = 0
747
- if 'error' in obj:
748
- _log(f"Error: {obj['error']}")
749
- raise Exception(obj['error']['message'])
750
- for candidate in obj['candidates']:
751
- role = "assistant"
752
- if 'content' in candidate and 'role' in candidate['content']:
753
- role = "assistant" if candidate['content']['role'] == 'model' else candidate['content']['role']
754
-
755
- # Safely extract content from all text parts
756
- content = ""
757
- reasoning = ""
758
- if 'content' in candidate and 'parts' in candidate['content']:
759
- text_parts = []
760
- reasoning_parts = []
761
- for part in candidate['content']['parts']:
762
- if 'text' in part:
763
- if 'thought' in part and part['thought']:
764
- reasoning_parts.append(part['text'])
765
- else:
766
- text_parts.append(part['text'])
767
- content = ' '.join(text_parts)
768
- reasoning = ' '.join(reasoning_parts)
1241
+ def get_provider_model(model_name):
1242
+ for provider in g_handlers.values():
1243
+ provider_model = provider.provider_model(model_name)
1244
+ if provider_model:
1245
+ return provider_model
1246
+ return None
769
1247
 
770
- choice = {
771
- "index": i,
772
- "finish_reason": candidate.get('finishReason', 'stop'),
773
- "message": {
774
- "role": role,
775
- "content": content,
776
- },
777
- }
778
- if reasoning:
779
- choice['message']['reasoning'] = reasoning
780
- choices.append(choice)
781
- i += 1
782
- response['choices'] = choices
783
- if 'usageMetadata' in obj:
784
- usage = obj['usageMetadata']
785
- response['usage'] = {
786
- "completion_tokens": usage['candidatesTokenCount'],
787
- "total_tokens": usage['totalTokenCount'],
788
- "prompt_tokens": usage['promptTokenCount'],
789
- }
790
- return self.to_response(response, chat, started_at)
791
1248
 
792
1249
  def get_models():
793
1250
  ret = []
794
1251
  for provider in g_handlers.values():
795
- for model in provider.models.keys():
1252
+ for model in provider.models:
796
1253
  if model not in ret:
797
1254
  ret.append(model)
798
1255
  ret.sort()
799
1256
  return ret
800
1257
 
1258
+
801
1259
  def get_active_models():
802
1260
  ret = []
803
1261
  existing_models = set()
804
- for id, provider in g_handlers.items():
805
- for model in provider.models.keys():
806
- if model not in existing_models:
807
- existing_models.add(model)
808
- provider_model = provider.models[model]
809
- pricing = provider.model_pricing(model)
810
- ret.append({
811
- "id": model,
812
- "provider": id,
813
- "provider_model": provider_model,
814
- "pricing": pricing
815
- })
1262
+ for provider_id, provider in g_handlers.items():
1263
+ for model in provider.models.values():
1264
+ name = model.get("name")
1265
+ if not name:
1266
+ _log(f"Provider {provider_id} model {model} has no name")
1267
+ continue
1268
+ if name not in existing_models:
1269
+ existing_models.add(name)
1270
+ item = model.copy()
1271
+ item.update({"provider": provider_id})
1272
+ ret.append(item)
816
1273
  ret.sort(key=lambda x: x["id"])
817
1274
  return ret
818
1275
 
819
- async def chat_completion(chat):
820
- model = chat['model']
821
- # get first provider that has the model
822
- candidate_providers = [name for name, provider in g_handlers.items() if model in provider.models]
823
- if len(candidate_providers) == 0:
824
- raise(Exception(f"Model {model} not found"))
825
1276
 
1277
+ def api_providers():
1278
+ ret = []
1279
+ for id, provider in g_handlers.items():
1280
+ ret.append({"id": id, "name": provider.name, "models": provider.models})
1281
+ return ret
1282
+
1283
+
1284
+ def to_error_message(e):
1285
+ return str(e)
1286
+
1287
+
1288
+ def to_error_response(e, stacktrace=False):
1289
+ status = {"errorCode": "Error", "message": to_error_message(e)}
1290
+ if stacktrace:
1291
+ status["stackTrace"] = traceback.format_exc()
1292
+ return {"responseStatus": status}
1293
+
1294
+
1295
+ def create_error_response(message, error_code="Error", stack_trace=None):
1296
+ ret = {"responseStatus": {"errorCode": error_code, "message": message}}
1297
+ if stack_trace:
1298
+ ret["responseStatus"]["stackTrace"] = stack_trace
1299
+ return ret
1300
+
1301
+
1302
+ def should_cancel_thread(context):
1303
+ ret = context.get("cancelled", False)
1304
+ if ret:
1305
+ thread_id = context.get("threadId")
1306
+ _dbg(f"Thread cancelled {thread_id}")
1307
+ return ret
1308
+
1309
+
1310
+ def g_chat_request(template=None, text=None, model=None, system_prompt=None):
1311
+ chat_template = g_config["defaults"].get(template or "text")
1312
+ if not chat_template:
1313
+ raise Exception(f"Chat template '{template}' not found")
1314
+
1315
+ chat = chat_template.copy()
1316
+ if model:
1317
+ chat["model"] = model
1318
+ if system_prompt is not None:
1319
+ chat["messages"].insert(0, {"role": "system", "content": system_prompt})
1320
+ if text is not None:
1321
+ if not chat["messages"] or len(chat["messages"]) == 0:
1322
+ chat["messages"] = [{"role": "user", "content": [{"type": "text", "text": ""}]}]
1323
+
1324
+ # replace content of last message if exists, else add
1325
+ last_msg = chat["messages"][-1] if "messages" in chat else None
1326
+ if last_msg and last_msg["role"] == "user":
1327
+ if isinstance(last_msg["content"], list):
1328
+ last_msg["content"][-1]["text"] = text
1329
+ else:
1330
+ last_msg["content"] = text
1331
+ else:
1332
+ chat["messages"].append({"role": "user", "content": text})
1333
+
1334
+ return chat
1335
+
1336
+
1337
+ async def g_chat_completion(chat, context=None):
1338
+ try:
1339
+ model = chat.get("model")
1340
+ if not model:
1341
+ raise Exception("Model not specified")
1342
+
1343
+ if context is None:
1344
+ context = {"chat": chat, "tools": "all"}
1345
+
1346
+ # get first provider that has the model
1347
+ candidate_providers = [name for name, provider in g_handlers.items() if provider.provider_model(model)]
1348
+ if len(candidate_providers) == 0:
1349
+ raise (Exception(f"Model {model} not found"))
1350
+ except Exception as e:
1351
+ await g_app.on_chat_error(e, context or {"chat": chat})
1352
+ raise e
1353
+
1354
+ started_at = time.time()
826
1355
  first_exception = None
1356
+ provider_name = "Unknown"
827
1357
  for name in candidate_providers:
828
- provider = g_handlers[name]
829
- _log(f"provider: {name} {type(provider).__name__}")
830
1358
  try:
831
- response = await provider.chat(chat.copy())
832
- return response
1359
+ provider_name = name
1360
+ provider = g_handlers[name]
1361
+ _log(f"provider: {name} {type(provider).__name__}")
1362
+ started_at = time.time()
1363
+ context["startedAt"] = datetime.now()
1364
+ context["provider"] = name
1365
+ model_info = provider.model_info(model)
1366
+ context["modelCost"] = model_info.get("cost", provider.model_cost(model)) or {"input": 0, "output": 0}
1367
+ context["modelInfo"] = model_info
1368
+
1369
+ # Accumulate usage across tool calls
1370
+ total_usage = {
1371
+ "prompt_tokens": 0,
1372
+ "completion_tokens": 0,
1373
+ "total_tokens": 0,
1374
+ }
1375
+ accumulated_cost = 0.0
1376
+
1377
+ # Inject global tools if present
1378
+ current_chat = chat.copy()
1379
+ if g_app.tool_definitions:
1380
+ only_tools_str = context.get("tools", "all")
1381
+ include_all_tools = only_tools_str == "all"
1382
+ only_tools = only_tools_str.split(",")
1383
+
1384
+ if include_all_tools or len(only_tools) > 0:
1385
+ if "tools" not in current_chat:
1386
+ current_chat["tools"] = []
1387
+
1388
+ existing_tools = {t["function"]["name"] for t in current_chat["tools"]}
1389
+ for tool_def in g_app.tool_definitions:
1390
+ name = tool_def["function"]["name"]
1391
+ if name not in existing_tools and (include_all_tools or name in only_tools):
1392
+ current_chat["tools"].append(tool_def)
1393
+
1394
+ # Apply pre-chat filters ONCE
1395
+ context["chat"] = current_chat
1396
+ for filter_func in g_app.chat_request_filters:
1397
+ await filter_func(current_chat, context)
1398
+
1399
+ # Tool execution loop
1400
+ max_iterations = 10
1401
+ tool_history = []
1402
+ final_response = None
1403
+
1404
+ for _ in range(max_iterations):
1405
+ if should_cancel_thread(context):
1406
+ return
1407
+
1408
+ response = await provider.chat(current_chat)
1409
+
1410
+ if should_cancel_thread(context):
1411
+ return
1412
+
1413
+ # Aggregate usage
1414
+ if "usage" in response:
1415
+ usage = response["usage"]
1416
+ total_usage["prompt_tokens"] += usage.get("prompt_tokens", 0)
1417
+ total_usage["completion_tokens"] += usage.get("completion_tokens", 0)
1418
+ total_usage["total_tokens"] += usage.get("total_tokens", 0)
1419
+
1420
+ # Calculate cost for this step if available
1421
+ if "cost" in response and isinstance(response["cost"], (int, float)):
1422
+ accumulated_cost += response["cost"]
1423
+ elif "cost" in usage and isinstance(usage["cost"], (int, float)):
1424
+ accumulated_cost += usage["cost"]
1425
+
1426
+ # Check for tool_calls in the response
1427
+ choice = response.get("choices", [])[0] if response.get("choices") else {}
1428
+ message = choice.get("message", {})
1429
+ tool_calls = message.get("tool_calls")
1430
+
1431
+ if tool_calls:
1432
+ # Append the assistant's message with tool calls to history
1433
+ if "messages" not in current_chat:
1434
+ current_chat["messages"] = []
1435
+ if "timestamp" not in message:
1436
+ message["timestamp"] = int(time.time() * 1000)
1437
+ current_chat["messages"].append(message)
1438
+ tool_history.append(message)
1439
+
1440
+ await g_app.on_chat_tool(current_chat, context)
1441
+
1442
+ for tool_call in tool_calls:
1443
+ function_name = tool_call["function"]["name"]
1444
+ try:
1445
+ function_args = json.loads(tool_call["function"]["arguments"])
1446
+ except Exception as e:
1447
+ tool_result = f"Error parsing JSON arguments for tool {function_name}: {e}"
1448
+ else:
1449
+ tool_result = f"Error: Tool {function_name} not found"
1450
+ if function_name in g_app.tools:
1451
+ try:
1452
+ func = g_app.tools[function_name]
1453
+ if inspect.iscoroutinefunction(func):
1454
+ tool_result = await func(**function_args)
1455
+ else:
1456
+ tool_result = func(**function_args)
1457
+ except Exception as e:
1458
+ tool_result = f"Error executing tool {function_name}: {e}"
1459
+
1460
+ # Append tool result to history
1461
+ tool_msg = {"role": "tool", "tool_call_id": tool_call["id"], "content": to_content(tool_result)}
1462
+ current_chat["messages"].append(tool_msg)
1463
+ tool_history.append(tool_msg)
1464
+
1465
+ await g_app.on_chat_tool(current_chat, context)
1466
+
1467
+ if should_cancel_thread(context):
1468
+ return
1469
+
1470
+ # Continue loop to send tool results back to LLM
1471
+ continue
1472
+
1473
+ # If no tool calls, this is the final response
1474
+ if tool_history:
1475
+ response["tool_history"] = tool_history
1476
+
1477
+ # Update final response with aggregated usage
1478
+ if "usage" not in response:
1479
+ response["usage"] = {}
1480
+ # convert to int seconds
1481
+ context["duration"] = duration = int(time.time() - started_at)
1482
+ total_usage.update({"duration": duration})
1483
+ response["usage"].update(total_usage)
1484
+ # If we accumulated cost, set it on the response
1485
+ if accumulated_cost > 0:
1486
+ response["cost"] = accumulated_cost
1487
+
1488
+ final_response = response
1489
+ break # Exit tool loop
1490
+
1491
+ if final_response:
1492
+ # Apply post-chat filters ONCE on final response
1493
+ for filter_func in g_app.chat_response_filters:
1494
+ await filter_func(final_response, context)
1495
+
1496
+ if DEBUG:
1497
+ _dbg(json.dumps(final_response, indent=2))
1498
+
1499
+ return final_response
1500
+
833
1501
  except Exception as e:
834
1502
  if first_exception is None:
835
1503
  first_exception = e
836
- _log(f"Provider {name} failed: {e}")
1504
+ context["stackTrace"] = traceback.format_exc()
1505
+ _err(f"Provider {provider_name} failed", first_exception)
1506
+ await g_app.on_chat_error(e, context)
1507
+
837
1508
  continue
838
1509
 
839
1510
  # If we get here, all providers failed
840
1511
  raise first_exception
841
1512
 
842
- async def cli_chat(chat, image=None, audio=None, file=None, args=None, raw=False):
1513
+
1514
+ async def cli_chat(chat, tools=None, image=None, audio=None, file=None, args=None, raw=False):
843
1515
  if g_default_model:
844
- chat['model'] = g_default_model
1516
+ chat["model"] = g_default_model
845
1517
 
846
1518
  # Apply args parameters to chat request
847
1519
  if args:
@@ -850,176 +1522,244 @@ async def cli_chat(chat, image=None, audio=None, file=None, args=None, raw=False
850
1522
  # process_chat downloads the image, just adding the reference here
851
1523
  if image is not None:
852
1524
  first_message = None
853
- for message in chat['messages']:
854
- if message['role'] == 'user':
1525
+ for message in chat["messages"]:
1526
+ if message["role"] == "user":
855
1527
  first_message = message
856
1528
  break
857
- image_content = {
858
- "type": "image_url",
859
- "image_url": {
860
- "url": image
861
- }
862
- }
863
- if 'content' in first_message:
864
- if isinstance(first_message['content'], list):
1529
+ image_content = {"type": "image_url", "image_url": {"url": image}}
1530
+ if "content" in first_message:
1531
+ if isinstance(first_message["content"], list):
865
1532
  image_url = None
866
- for item in first_message['content']:
867
- if 'image_url' in item:
868
- image_url = item['image_url']
1533
+ for item in first_message["content"]:
1534
+ if "image_url" in item:
1535
+ image_url = item["image_url"]
869
1536
  # If no image_url, add one
870
1537
  if image_url is None:
871
- first_message['content'].insert(0,image_content)
1538
+ first_message["content"].insert(0, image_content)
872
1539
  else:
873
- image_url['url'] = image
1540
+ image_url["url"] = image
874
1541
  else:
875
- first_message['content'] = [
876
- image_content,
877
- { "type": "text", "text": first_message['content'] }
878
- ]
1542
+ first_message["content"] = [image_content, {"type": "text", "text": first_message["content"]}]
879
1543
  if audio is not None:
880
1544
  first_message = None
881
- for message in chat['messages']:
882
- if message['role'] == 'user':
1545
+ for message in chat["messages"]:
1546
+ if message["role"] == "user":
883
1547
  first_message = message
884
1548
  break
885
- audio_content = {
886
- "type": "input_audio",
887
- "input_audio": {
888
- "data": audio,
889
- "format": "mp3"
890
- }
891
- }
892
- if 'content' in first_message:
893
- if isinstance(first_message['content'], list):
1549
+ audio_content = {"type": "input_audio", "input_audio": {"data": audio, "format": "mp3"}}
1550
+ if "content" in first_message:
1551
+ if isinstance(first_message["content"], list):
894
1552
  input_audio = None
895
- for item in first_message['content']:
896
- if 'input_audio' in item:
897
- input_audio = item['input_audio']
1553
+ for item in first_message["content"]:
1554
+ if "input_audio" in item:
1555
+ input_audio = item["input_audio"]
898
1556
  # If no input_audio, add one
899
1557
  if input_audio is None:
900
- first_message['content'].insert(0,audio_content)
1558
+ first_message["content"].insert(0, audio_content)
901
1559
  else:
902
- input_audio['data'] = audio
1560
+ input_audio["data"] = audio
903
1561
  else:
904
- first_message['content'] = [
905
- audio_content,
906
- { "type": "text", "text": first_message['content'] }
907
- ]
1562
+ first_message["content"] = [audio_content, {"type": "text", "text": first_message["content"]}]
908
1563
  if file is not None:
909
1564
  first_message = None
910
- for message in chat['messages']:
911
- if message['role'] == 'user':
1565
+ for message in chat["messages"]:
1566
+ if message["role"] == "user":
912
1567
  first_message = message
913
1568
  break
914
- file_content = {
915
- "type": "file",
916
- "file": {
917
- "filename": get_filename(file),
918
- "file_data": file
919
- }
920
- }
921
- if 'content' in first_message:
922
- if isinstance(first_message['content'], list):
1569
+ file_content = {"type": "file", "file": {"filename": get_filename(file), "file_data": file}}
1570
+ if "content" in first_message:
1571
+ if isinstance(first_message["content"], list):
923
1572
  file_data = None
924
- for item in first_message['content']:
925
- if 'file' in item:
926
- file_data = item['file']
1573
+ for item in first_message["content"]:
1574
+ if "file" in item:
1575
+ file_data = item["file"]
927
1576
  # If no file_data, add one
928
1577
  if file_data is None:
929
- first_message['content'].insert(0,file_content)
1578
+ first_message["content"].insert(0, file_content)
930
1579
  else:
931
- file_data['filename'] = get_filename(file)
932
- file_data['file_data'] = file
1580
+ file_data["filename"] = get_filename(file)
1581
+ file_data["file_data"] = file
933
1582
  else:
934
- first_message['content'] = [
935
- file_content,
936
- { "type": "text", "text": first_message['content'] }
937
- ]
1583
+ first_message["content"] = [file_content, {"type": "text", "text": first_message["content"]}]
938
1584
 
939
1585
  if g_verbose:
940
1586
  printdump(chat)
941
1587
 
942
1588
  try:
943
- response = await chat_completion(chat)
1589
+ context = {
1590
+ "tools": tools or "all",
1591
+ }
1592
+ response = await g_app.chat_completion(chat, context=context)
1593
+
944
1594
  if raw:
945
1595
  print(json.dumps(response, indent=2))
946
1596
  exit(0)
947
1597
  else:
948
- answer = response['choices'][0]['message']['content']
949
- print(answer)
1598
+ msg = response["choices"][0]["message"]
1599
+ if "content" in msg or "answer" in msg:
1600
+ print(msg["content"])
1601
+
1602
+ generated_files = []
1603
+ for choice in response["choices"]:
1604
+ if "message" in choice:
1605
+ msg = choice["message"]
1606
+ if "images" in msg:
1607
+ for image in msg["images"]:
1608
+ image_url = image["image_url"]["url"]
1609
+ generated_files.append(image_url)
1610
+ if "audios" in msg:
1611
+ for audio in msg["audios"]:
1612
+ audio_url = audio["audio_url"]["url"]
1613
+ generated_files.append(audio_url)
1614
+
1615
+ if len(generated_files) > 0:
1616
+ print("\nSaved files:")
1617
+ for file in generated_files:
1618
+ if file.startswith("/~cache"):
1619
+ print(get_cache_path(file[8:]))
1620
+ print(urljoin("http://localhost:8000", file))
1621
+ else:
1622
+ print(file)
1623
+
950
1624
  except HTTPError as e:
951
1625
  # HTTP error (4xx, 5xx)
952
1626
  print(f"{e}:\n{e.body}")
953
- exit(1)
1627
+ g_app.exit(1)
954
1628
  except aiohttp.ClientConnectionError as e:
955
1629
  # Connection issues
956
1630
  print(f"Connection error: {e}")
957
- exit(1)
1631
+ g_app.exit(1)
958
1632
  except asyncio.TimeoutError as e:
959
1633
  # Timeout
960
1634
  print(f"Timeout error: {e}")
961
- exit(1)
1635
+ g_app.exit(1)
1636
+
962
1637
 
963
1638
  def config_str(key):
964
1639
  return key in g_config and g_config[key] or None
965
1640
 
966
- def init_llms(config):
967
- global g_config, g_handlers
968
1641
 
1642
+ def load_config(config, providers, verbose=None):
1643
+ global g_config, g_providers, g_verbose
969
1644
  g_config = config
1645
+ g_providers = providers
1646
+ if verbose:
1647
+ g_verbose = verbose
1648
+
1649
+
1650
+ def init_llms(config, providers):
1651
+ global g_config, g_handlers
1652
+
1653
+ load_config(config, providers)
970
1654
  g_handlers = {}
971
1655
  # iterate over config and replace $ENV with env value
972
1656
  for key, value in g_config.items():
973
1657
  if isinstance(value, str) and value.startswith("$"):
974
- g_config[key] = os.environ.get(value[1:], "")
1658
+ g_config[key] = os.getenv(value[1:], "")
975
1659
 
976
1660
  # if g_verbose:
977
1661
  # printdump(g_config)
978
- providers = g_config['providers']
1662
+ providers = g_config["providers"]
979
1663
 
980
- for name, orig in providers.items():
981
- definition = orig.copy()
982
- provider_type = definition['type']
983
- if 'enabled' in definition and not definition['enabled']:
1664
+ for id, orig in providers.items():
1665
+ if "enabled" in orig and not orig["enabled"]:
984
1666
  continue
985
1667
 
986
- # Replace API keys with environment variables if they start with $
987
- if 'api_key' in definition:
988
- value = definition['api_key']
989
- if isinstance(value, str) and value.startswith("$"):
990
- definition['api_key'] = os.environ.get(value[1:], "")
991
-
992
- # Create a copy of definition without the 'type' key for constructor kwargs
993
- constructor_kwargs = {k: v for k, v in definition.items() if k != 'type' and k != 'enabled'}
994
- constructor_kwargs['headers'] = g_config['defaults']['headers'].copy()
995
-
996
- if provider_type == 'OpenAiProvider' and OpenAiProvider.test(**constructor_kwargs):
997
- g_handlers[name] = OpenAiProvider(**constructor_kwargs)
998
- elif provider_type == 'OllamaProvider' and OllamaProvider.test(**constructor_kwargs):
999
- g_handlers[name] = OllamaProvider(**constructor_kwargs)
1000
- elif provider_type == 'GoogleProvider' and GoogleProvider.test(**constructor_kwargs):
1001
- g_handlers[name] = GoogleProvider(**constructor_kwargs)
1002
- elif provider_type == 'GoogleOpenAiProvider' and GoogleOpenAiProvider.test(**constructor_kwargs):
1003
- g_handlers[name] = GoogleOpenAiProvider(**constructor_kwargs)
1004
-
1668
+ provider, constructor_kwargs = create_provider_from_definition(id, orig)
1669
+ if provider and provider.test(**constructor_kwargs):
1670
+ g_handlers[id] = provider
1005
1671
  return g_handlers
1006
1672
 
1673
+
1674
+ def create_provider_from_definition(id, orig):
1675
+ definition = orig.copy()
1676
+ provider_id = definition.get("id", id)
1677
+ if "id" not in definition:
1678
+ definition["id"] = provider_id
1679
+ provider = g_providers.get(provider_id)
1680
+ constructor_kwargs = create_provider_kwargs(definition, provider)
1681
+ provider = create_provider(constructor_kwargs)
1682
+ return provider, constructor_kwargs
1683
+
1684
+
1685
+ def create_provider_kwargs(definition, provider=None):
1686
+ if provider:
1687
+ provider = provider.copy()
1688
+ provider.update(definition)
1689
+ else:
1690
+ provider = definition.copy()
1691
+
1692
+ # Replace API keys with environment variables if they start with $
1693
+ if "api_key" in provider:
1694
+ value = provider["api_key"]
1695
+ if isinstance(value, str) and value.startswith("$"):
1696
+ provider["api_key"] = os.getenv(value[1:], "")
1697
+
1698
+ if "api_key" not in provider and "env" in provider:
1699
+ for env_var in provider["env"]:
1700
+ val = os.getenv(env_var)
1701
+ if val:
1702
+ provider["api_key"] = val
1703
+ break
1704
+
1705
+ # Create a copy of provider
1706
+ constructor_kwargs = dict(provider.items())
1707
+ # Create a copy of all list and dict values
1708
+ for key, value in constructor_kwargs.items():
1709
+ if isinstance(value, (list, dict)):
1710
+ constructor_kwargs[key] = value.copy()
1711
+ constructor_kwargs["headers"] = g_config["defaults"]["headers"].copy()
1712
+
1713
+ if "modalities" in definition:
1714
+ constructor_kwargs["modalities"] = {}
1715
+ for modality, modality_definition in definition["modalities"].items():
1716
+ modality_provider = create_provider(modality_definition)
1717
+ if not modality_provider:
1718
+ return None
1719
+ constructor_kwargs["modalities"][modality] = modality_provider
1720
+
1721
+ return constructor_kwargs
1722
+
1723
+
1724
+ def create_provider(provider):
1725
+ if not isinstance(provider, dict):
1726
+ return None
1727
+ provider_label = provider.get("id", provider.get("name", "unknown"))
1728
+ npm_sdk = provider.get("npm")
1729
+ if not npm_sdk:
1730
+ _log(f"Provider {provider_label} is missing 'npm' sdk")
1731
+ return None
1732
+
1733
+ for provider_type in g_app.all_providers:
1734
+ if provider_type.sdk == npm_sdk:
1735
+ kwargs = create_provider_kwargs(provider)
1736
+ if kwargs is None:
1737
+ kwargs = provider
1738
+ return provider_type(**kwargs)
1739
+
1740
+ _log(f"Could not find provider {provider_label} with npm sdk {npm_sdk}")
1741
+ return None
1742
+
1743
+
1007
1744
  async def load_llms():
1008
1745
  global g_handlers
1009
1746
  _log("Loading providers...")
1010
- for name, provider in g_handlers.items():
1747
+ for _name, provider in g_handlers.items():
1011
1748
  await provider.load()
1012
1749
 
1750
+
1013
1751
  def save_config(config):
1014
1752
  global g_config, g_config_path
1015
1753
  g_config = config
1016
- with open(g_config_path, "w") as f:
1754
+ with open(g_config_path, "w", encoding="utf-8") as f:
1017
1755
  json.dump(g_config, f, indent=4)
1018
1756
  _log(f"Saved config to {g_config_path}")
1019
1757
 
1758
+
1020
1759
  def github_url(filename):
1021
1760
  return f"https://raw.githubusercontent.com/ServiceStack/llms/refs/heads/main/llms/{filename}"
1022
1761
 
1762
+
1023
1763
  async def get_text(url):
1024
1764
  async with aiohttp.ClientSession() as session:
1025
1765
  _log(f"GET {url}")
@@ -1029,25 +1769,58 @@ async def get_text(url):
1029
1769
  raise HTTPError(resp.status, reason=resp.reason, body=text, headers=dict(resp.headers))
1030
1770
  return text
1031
1771
 
1772
+
1032
1773
  async def save_text_url(url, save_path):
1033
1774
  text = await get_text(url)
1034
1775
  os.makedirs(os.path.dirname(save_path), exist_ok=True)
1035
- with open(save_path, "w") as f:
1776
+ with open(save_path, "w", encoding="utf-8") as f:
1036
1777
  f.write(text)
1037
1778
  return text
1038
1779
 
1780
+
1039
1781
  async def save_default_config(config_path):
1040
1782
  global g_config
1041
1783
  config_json = await save_text_url(github_url("llms.json"), config_path)
1042
1784
  g_config = json.loads(config_json)
1043
1785
 
1786
+
1787
+ async def update_providers(home_providers_path):
1788
+ global g_providers
1789
+ text = await get_text("https://models.dev/api.json")
1790
+ all_providers = json.loads(text)
1791
+ extra_providers = {}
1792
+ extra_providers_path = home_providers_path.replace("providers.json", "providers-extra.json")
1793
+ if os.path.exists(extra_providers_path):
1794
+ with open(extra_providers_path) as f:
1795
+ extra_providers = json.load(f)
1796
+
1797
+ filtered_providers = {}
1798
+ for id, provider in all_providers.items():
1799
+ if id in g_config["providers"]:
1800
+ filtered_providers[id] = provider
1801
+ if id in extra_providers and "models" in extra_providers[id]:
1802
+ for model_id, model in extra_providers[id]["models"].items():
1803
+ if "id" not in model:
1804
+ model["id"] = model_id
1805
+ if "name" not in model:
1806
+ model["name"] = id_to_name(model["id"])
1807
+ filtered_providers[id]["models"][model_id] = model
1808
+
1809
+ os.makedirs(os.path.dirname(home_providers_path), exist_ok=True)
1810
+ with open(home_providers_path, "w", encoding="utf-8") as f:
1811
+ json.dump(filtered_providers, f)
1812
+
1813
+ g_providers = filtered_providers
1814
+
1815
+
1044
1816
  def provider_status():
1045
1817
  enabled = list(g_handlers.keys())
1046
- disabled = [provider for provider in g_config['providers'].keys() if provider not in enabled]
1818
+ disabled = [provider for provider in g_config["providers"] if provider not in enabled]
1047
1819
  enabled.sort()
1048
1820
  disabled.sort()
1049
1821
  return enabled, disabled
1050
1822
 
1823
+
1051
1824
  def print_status():
1052
1825
  enabled, disabled = provider_status()
1053
1826
  if len(enabled) > 0:
@@ -1059,8 +1832,14 @@ def print_status():
1059
1832
  else:
1060
1833
  print("Disabled: None")
1061
1834
 
1835
+
1062
1836
  def home_llms_path(filename):
1063
- return f"{os.environ.get('HOME')}/.llms/{filename}"
1837
+ return f"{os.getenv('HOME')}/.llms/{filename}"
1838
+
1839
+
1840
+ def get_cache_path(path=""):
1841
+ return home_llms_path(f"cache/{path}") if path else home_llms_path("cache")
1842
+
1064
1843
 
1065
1844
  def get_config_path():
1066
1845
  home_config_path = home_llms_path("llms.json")
@@ -1068,8 +1847,8 @@ def get_config_path():
1068
1847
  "./llms.json",
1069
1848
  home_config_path,
1070
1849
  ]
1071
- if os.environ.get("LLMS_CONFIG_PATH"):
1072
- check_paths.insert(0, os.environ.get("LLMS_CONFIG_PATH"))
1850
+ if os.getenv("LLMS_CONFIG_PATH"):
1851
+ check_paths.insert(0, os.getenv("LLMS_CONFIG_PATH"))
1073
1852
 
1074
1853
  for check_path in check_paths:
1075
1854
  g_config_path = os.path.normpath(os.path.join(os.path.dirname(__file__), check_path))
@@ -1077,37 +1856,30 @@ def get_config_path():
1077
1856
  return g_config_path
1078
1857
  return None
1079
1858
 
1080
- def get_ui_path():
1081
- ui_paths = [
1082
- home_llms_path("ui.json"),
1083
- "ui.json"
1084
- ]
1085
- for ui_path in ui_paths:
1086
- if os.path.exists(ui_path):
1087
- return ui_path
1088
- return None
1089
1859
 
1090
1860
  def enable_provider(provider):
1091
1861
  msg = None
1092
- provider_config = g_config['providers'][provider]
1093
- provider_config['enabled'] = True
1094
- if 'api_key' in provider_config:
1095
- api_key = provider_config['api_key']
1096
- if isinstance(api_key, str):
1097
- if api_key.startswith("$"):
1098
- if not os.environ.get(api_key[1:], ""):
1099
- msg = f"WARNING: {provider} requires missing API Key in Environment Variable {api_key}"
1100
- else:
1101
- msg = f"WARNING: {provider} is not configured with an API Key"
1862
+ provider_config = g_config["providers"][provider]
1863
+ if not provider_config:
1864
+ return None, f"Provider {provider} not found"
1865
+
1866
+ provider, constructor_kwargs = create_provider_from_definition(provider, provider_config)
1867
+ msg = provider.validate(**constructor_kwargs)
1868
+ if msg:
1869
+ return None, msg
1870
+
1871
+ provider_config["enabled"] = True
1102
1872
  save_config(g_config)
1103
- init_llms(g_config)
1873
+ init_llms(g_config, g_providers)
1104
1874
  return provider_config, msg
1105
1875
 
1876
+
1106
1877
  def disable_provider(provider):
1107
- provider_config = g_config['providers'][provider]
1108
- provider_config['enabled'] = False
1878
+ provider_config = g_config["providers"][provider]
1879
+ provider_config["enabled"] = False
1109
1880
  save_config(g_config)
1110
- init_llms(g_config)
1881
+ init_llms(g_config, g_providers)
1882
+
1111
1883
 
1112
1884
  def resolve_root():
1113
1885
  # Try to find the resource root directory
@@ -1119,7 +1891,7 @@ def resolve_root():
1119
1891
  # Try to access the package resources
1120
1892
  pkg_files = resources.files("llms")
1121
1893
  # Check if ui directory exists in package resources
1122
- if hasattr(pkg_files, 'is_dir') and (pkg_files / "ui").is_dir():
1894
+ if hasattr(pkg_files, "is_dir") and (pkg_files / "ui").is_dir():
1123
1895
  _log(f"RESOURCE ROOT (package): {pkg_files}")
1124
1896
  return pkg_files
1125
1897
  except (FileNotFoundError, AttributeError, TypeError):
@@ -1132,8 +1904,9 @@ def resolve_root():
1132
1904
  # Method 1b: Look for the installed package and check for UI files
1133
1905
  try:
1134
1906
  import llms
1907
+
1135
1908
  # If llms is a package, check its directory
1136
- if hasattr(llms, '__path__'):
1909
+ if hasattr(llms, "__path__"):
1137
1910
  # It's a package
1138
1911
  package_path = Path(llms.__path__[0])
1139
1912
 
@@ -1170,21 +1943,25 @@ def resolve_root():
1170
1943
 
1171
1944
  # Add site-packages directories
1172
1945
  for site_dir in site.getsitepackages():
1173
- possible_roots.extend([
1174
- Path(site_dir),
1175
- Path(site_dir).parent,
1176
- Path(site_dir).parent / "share",
1177
- ])
1946
+ possible_roots.extend(
1947
+ [
1948
+ Path(site_dir),
1949
+ Path(site_dir).parent,
1950
+ Path(site_dir).parent / "share",
1951
+ ]
1952
+ )
1178
1953
 
1179
1954
  # Add user site directory
1180
1955
  try:
1181
1956
  user_site = site.getusersitepackages()
1182
1957
  if user_site:
1183
- possible_roots.extend([
1184
- Path(user_site),
1185
- Path(user_site).parent,
1186
- Path(user_site).parent / "share",
1187
- ])
1958
+ possible_roots.extend(
1959
+ [
1960
+ Path(user_site),
1961
+ Path(user_site).parent,
1962
+ Path(user_site).parent / "share",
1963
+ ]
1964
+ )
1188
1965
  except AttributeError:
1189
1966
  pass
1190
1967
 
@@ -1195,12 +1972,17 @@ def resolve_root():
1195
1972
  homebrew_prefixes = ["/opt/homebrew", "/usr/local"] # Apple Silicon and Intel
1196
1973
  for prefix in homebrew_prefixes:
1197
1974
  if Path(prefix).exists():
1198
- homebrew_roots.extend([
1199
- Path(prefix),
1200
- Path(prefix) / "share",
1201
- Path(prefix) / "lib" / "python3.11" / "site-packages",
1202
- Path(prefix) / "lib" / f"python{sys.version_info.major}.{sys.version_info.minor}" / "site-packages",
1203
- ])
1975
+ homebrew_roots.extend(
1976
+ [
1977
+ Path(prefix),
1978
+ Path(prefix) / "share",
1979
+ Path(prefix) / "lib" / "python3.11" / "site-packages",
1980
+ Path(prefix)
1981
+ / "lib"
1982
+ / f"python{sys.version_info.major}.{sys.version_info.minor}"
1983
+ / "site-packages",
1984
+ ]
1985
+ )
1204
1986
 
1205
1987
  possible_roots.extend(homebrew_roots)
1206
1988
 
@@ -1232,26 +2014,29 @@ def resolve_root():
1232
2014
  _log(f"RESOURCE ROOT (fallback): {from_file}")
1233
2015
  return from_file
1234
2016
 
2017
+
1235
2018
  def resource_exists(resource_path):
1236
2019
  # Check if resource files exist (handle both Path and Traversable objects)
1237
2020
  try:
1238
- if hasattr(resource_path, 'is_file'):
2021
+ if hasattr(resource_path, "is_file"):
1239
2022
  return resource_path.is_file()
1240
2023
  else:
1241
2024
  return os.path.exists(resource_path)
1242
2025
  except (OSError, AttributeError):
1243
2026
  pass
1244
2027
 
2028
+
1245
2029
  def read_resource_text(resource_path):
1246
- if hasattr(resource_path, 'read_text'):
2030
+ if hasattr(resource_path, "read_text"):
1247
2031
  return resource_path.read_text()
1248
2032
  else:
1249
- with open(resource_path, "r") as f:
2033
+ with open(resource_path, encoding="utf-8") as f:
1250
2034
  return f.read()
1251
2035
 
2036
+
1252
2037
  def read_resource_file_bytes(resource_file):
1253
2038
  try:
1254
- if hasattr(_ROOT, 'joinpath'):
2039
+ if hasattr(_ROOT, "joinpath"):
1255
2040
  # importlib.resources Traversable
1256
2041
  index_resource = _ROOT.joinpath(resource_file)
1257
2042
  if index_resource.is_file():
@@ -1264,6 +2049,7 @@ def read_resource_file_bytes(resource_file):
1264
2049
  except (OSError, PermissionError, AttributeError) as e:
1265
2050
  _log(f"Error reading resource bytes: {e}")
1266
2051
 
2052
+
1267
2053
  async def check_models(provider_name, model_names=None):
1268
2054
  """
1269
2055
  Check validity of models for a specific provider by sending a ping message.
@@ -1281,13 +2067,14 @@ async def check_models(provider_name, model_names=None):
1281
2067
  models_to_check = []
1282
2068
 
1283
2069
  # Determine which models to check
1284
- if model_names is None or (len(model_names) == 1 and model_names[0] == 'all'):
2070
+ if model_names is None or (len(model_names) == 1 and model_names[0] == "all"):
1285
2071
  # Check all models for this provider
1286
2072
  models_to_check = list(provider.models.keys())
1287
2073
  else:
1288
2074
  # Check only specified models
1289
2075
  for model_name in model_names:
1290
- if model_name in provider.models:
2076
+ provider_model = provider.provider_model(model_name)
2077
+ if provider_model:
1291
2078
  models_to_check.append(model_name)
1292
2079
  else:
1293
2080
  print(f"Model '{model_name}' not found in provider '{provider_name}'")
@@ -1296,68 +2083,83 @@ async def check_models(provider_name, model_names=None):
1296
2083
  print(f"No models to check for provider '{provider_name}'")
1297
2084
  return
1298
2085
 
1299
- print(f"\nChecking {len(models_to_check)} model{'' if len(models_to_check) == 1 else 's'} for provider '{provider_name}':\n")
2086
+ print(
2087
+ f"\nChecking {len(models_to_check)} model{'' if len(models_to_check) == 1 else 's'} for provider '{provider_name}':\n"
2088
+ )
1300
2089
 
1301
2090
  # Test each model
1302
2091
  for model in models_to_check:
1303
- # Create a simple ping chat request
1304
- chat = (provider.check or g_config['defaults']['check']).copy()
1305
- chat["model"] = model
2092
+ await check_provider_model(provider, model)
1306
2093
 
1307
- started_at = time.time()
1308
- try:
1309
- # Try to get a response from the model
1310
- response = await provider.chat(chat)
1311
- duration_ms = int((time.time() - started_at) * 1000)
2094
+ print()
1312
2095
 
1313
- # Check if we got a valid response
1314
- if response and 'choices' in response and len(response['choices']) > 0:
1315
- print(f" ✓ {model:<40} ({duration_ms}ms)")
1316
- else:
1317
- print(f" ✗ {model:<40} Invalid response format")
1318
- except HTTPError as e:
1319
- duration_ms = int((time.time() - started_at) * 1000)
1320
- error_msg = f"HTTP {e.status}"
1321
- try:
1322
- # Try to parse error body for more details
1323
- error_body = json.loads(e.body) if e.body else {}
1324
- if 'error' in error_body:
1325
- error = error_body['error']
1326
- if isinstance(error, dict):
1327
- if 'message' in error:
1328
- # OpenRouter
1329
- if isinstance(error['message'], str):
1330
- error_msg = error['message']
1331
- if 'code' in error:
1332
- error_msg = f"{error['code']} {error_msg}"
1333
- if 'metadata' in error and 'raw' in error['metadata']:
1334
- error_msg += f" - {error['metadata']['raw']}"
1335
- if 'provider' in error:
1336
- error_msg += f" ({error['provider']})"
1337
- elif isinstance(error, str):
1338
- error_msg = error
1339
- elif 'message' in error_body:
1340
- if isinstance(error_body['message'], str):
1341
- error_msg = error_body['message']
1342
- elif isinstance(error_body['message'], dict):
1343
- # codestral error format
1344
- if 'detail' in error_body['message'] and isinstance(error_body['message']['detail'], list):
1345
- error_msg = error_body['message']['detail'][0]['msg']
1346
- if 'loc' in error_body['message']['detail'][0] and len(error_body['message']['detail'][0]['loc']) > 0:
1347
- error_msg += f" (in {' '.join(error_body['message']['detail'][0]['loc'])})"
1348
- except Exception as parse_error:
1349
- _log(f"Error parsing error body: {parse_error}")
1350
- error_msg = e.body[:100] if e.body else f"HTTP {e.status}"
1351
- print(f" ✗ {model:<40} {error_msg}")
1352
- except asyncio.TimeoutError:
1353
- duration_ms = int((time.time() - started_at) * 1000)
1354
- print(f" ✗ {model:<40} Timeout after {duration_ms}ms")
1355
- except Exception as e:
1356
- duration_ms = int((time.time() - started_at) * 1000)
1357
- error_msg = str(e)[:100]
1358
- print(f" ✗ {model:<40} {error_msg}")
1359
2096
 
1360
- print()
2097
+ async def check_provider_model(provider, model):
2098
+ # Create a simple ping chat request
2099
+ chat = (provider.check or g_config["defaults"]["check"]).copy()
2100
+ chat["model"] = model
2101
+
2102
+ success = False
2103
+ started_at = time.time()
2104
+ try:
2105
+ # Try to get a response from the model
2106
+ response = await provider.chat(chat)
2107
+ duration_ms = int((time.time() - started_at) * 1000)
2108
+
2109
+ # Check if we got a valid response
2110
+ if response and "choices" in response and len(response["choices"]) > 0:
2111
+ success = True
2112
+ print(f" ✓ {model:<40} ({duration_ms}ms)")
2113
+ else:
2114
+ print(f" ✗ {model:<40} Invalid response format")
2115
+ except HTTPError as e:
2116
+ duration_ms = int((time.time() - started_at) * 1000)
2117
+ error_msg = f"HTTP {e.status}"
2118
+ try:
2119
+ # Try to parse error body for more details
2120
+ error_body = json.loads(e.body) if e.body else {}
2121
+ if "error" in error_body:
2122
+ error = error_body["error"]
2123
+ if isinstance(error, dict):
2124
+ if "message" in error and isinstance(error["message"], str):
2125
+ # OpenRouter
2126
+ error_msg = error["message"]
2127
+ if "code" in error:
2128
+ error_msg = f"{error['code']} {error_msg}"
2129
+ if "metadata" in error and "raw" in error["metadata"]:
2130
+ error_msg += f" - {error['metadata']['raw']}"
2131
+ if "provider" in error:
2132
+ error_msg += f" ({error['provider']})"
2133
+ elif isinstance(error, str):
2134
+ error_msg = error
2135
+ elif "message" in error_body:
2136
+ if isinstance(error_body["message"], str):
2137
+ error_msg = error_body["message"]
2138
+ elif (
2139
+ isinstance(error_body["message"], dict)
2140
+ and "detail" in error_body["message"]
2141
+ and isinstance(error_body["message"]["detail"], list)
2142
+ ):
2143
+ # codestral error format
2144
+ error_msg = error_body["message"]["detail"][0]["msg"]
2145
+ if (
2146
+ "loc" in error_body["message"]["detail"][0]
2147
+ and len(error_body["message"]["detail"][0]["loc"]) > 0
2148
+ ):
2149
+ error_msg += f" (in {' '.join(error_body['message']['detail'][0]['loc'])})"
2150
+ except Exception as parse_error:
2151
+ _log(f"Error parsing error body: {parse_error}")
2152
+ error_msg = e.body[:100] if e.body else f"HTTP {e.status}"
2153
+ print(f" ✗ {model:<40} {error_msg}")
2154
+ except asyncio.TimeoutError:
2155
+ duration_ms = int((time.time() - started_at) * 1000)
2156
+ print(f" ✗ {model:<40} Timeout after {duration_ms}ms")
2157
+ except Exception as e:
2158
+ duration_ms = int((time.time() - started_at) * 1000)
2159
+ error_msg = str(e)[:100]
2160
+ print(f" ✗ {model:<40} {error_msg}")
2161
+ return success
2162
+
1361
2163
 
1362
2164
  def text_from_resource(filename):
1363
2165
  global _ROOT
@@ -1369,12 +2171,14 @@ def text_from_resource(filename):
1369
2171
  _log(f"Error reading resource config {filename}: {e}")
1370
2172
  return None
1371
2173
 
2174
+
1372
2175
  def text_from_file(filename):
1373
2176
  if os.path.exists(filename):
1374
- with open(filename, "r") as f:
2177
+ with open(filename, encoding="utf-8") as f:
1375
2178
  return f.read()
1376
2179
  return None
1377
2180
 
2181
+
1378
2182
  async def text_from_resource_or_url(filename):
1379
2183
  text = text_from_resource(filename)
1380
2184
  if not text:
@@ -1386,10 +2190,17 @@ async def text_from_resource_or_url(filename):
1386
2190
  raise e
1387
2191
  return text
1388
2192
 
2193
+
1389
2194
  async def save_home_configs():
1390
2195
  home_config_path = home_llms_path("llms.json")
1391
- home_ui_path = home_llms_path("ui.json")
1392
- if os.path.exists(home_config_path) and os.path.exists(home_ui_path):
2196
+ home_providers_path = home_llms_path("providers.json")
2197
+ home_providers_extra_path = home_llms_path("providers-extra.json")
2198
+
2199
+ if (
2200
+ os.path.exists(home_config_path)
2201
+ and os.path.exists(home_providers_path)
2202
+ and os.path.exists(home_providers_extra_path)
2203
+ ):
1393
2204
  return
1394
2205
 
1395
2206
  llms_home = os.path.dirname(home_config_path)
@@ -1397,114 +2208,717 @@ async def save_home_configs():
1397
2208
  try:
1398
2209
  if not os.path.exists(home_config_path):
1399
2210
  config_json = await text_from_resource_or_url("llms.json")
1400
- with open(home_config_path, "w") as f:
2211
+ with open(home_config_path, "w", encoding="utf-8") as f:
1401
2212
  f.write(config_json)
1402
2213
  _log(f"Created default config at {home_config_path}")
1403
2214
 
1404
- if not os.path.exists(home_ui_path):
1405
- ui_json = await text_from_resource_or_url("ui.json")
1406
- with open(home_ui_path, "w") as f:
1407
- f.write(ui_json)
1408
- _log(f"Created default ui config at {home_ui_path}")
1409
- except Exception as e:
2215
+ if not os.path.exists(home_providers_path):
2216
+ providers_json = await text_from_resource_or_url("providers.json")
2217
+ with open(home_providers_path, "w", encoding="utf-8") as f:
2218
+ f.write(providers_json)
2219
+ _log(f"Created default providers config at {home_providers_path}")
2220
+
2221
+ if not os.path.exists(home_providers_extra_path):
2222
+ extra_json = await text_from_resource_or_url("providers-extra.json")
2223
+ with open(home_providers_extra_path, "w", encoding="utf-8") as f:
2224
+ f.write(extra_json)
2225
+ _log(f"Created default extra providers config at {home_providers_extra_path}")
2226
+ except Exception:
1410
2227
  print("Could not create llms.json. Create one with --init or use --config <path>")
1411
2228
  exit(1)
1412
2229
 
2230
+
2231
+ def load_config_json(config_json):
2232
+ if config_json is None:
2233
+ return None
2234
+ config = json.loads(config_json)
2235
+ if not config or "version" not in config or config["version"] < 3:
2236
+ preserve_keys = ["auth", "defaults", "limits", "convert"]
2237
+ new_config = json.loads(text_from_resource("llms.json"))
2238
+ if config:
2239
+ for key in preserve_keys:
2240
+ if key in config:
2241
+ new_config[key] = config[key]
2242
+ config = new_config
2243
+ # move old config to YYYY-MM-DD.bak
2244
+ new_path = f"{g_config_path}.{datetime.now().strftime('%Y-%m-%d')}.bak"
2245
+ if os.path.exists(new_path):
2246
+ os.remove(new_path)
2247
+ os.rename(g_config_path, new_path)
2248
+ print(f"llms.json migrated. old config moved to {new_path}")
2249
+ # save new config
2250
+ save_config(g_config)
2251
+ return config
2252
+
2253
+
1413
2254
  async def reload_providers():
1414
2255
  global g_config, g_handlers
1415
- g_handlers = init_llms(g_config)
2256
+ g_handlers = init_llms(g_config, g_providers)
1416
2257
  await load_llms()
1417
2258
  _log(f"{len(g_handlers)} providers loaded")
1418
2259
  return g_handlers
1419
2260
 
1420
- async def watch_config_files(config_path, ui_path, interval=1):
2261
+
2262
+ async def watch_config_files(config_path, providers_path, interval=1):
1421
2263
  """Watch config files and reload providers when they change"""
1422
2264
  global g_config
1423
2265
 
1424
2266
  config_path = Path(config_path)
1425
- ui_path = Path(ui_path) if ui_path else None
2267
+ providers_path = Path(providers_path)
1426
2268
 
1427
- file_mtimes = {}
2269
+ _log(f"Watching config file: {config_path}")
2270
+ _log(f"Watching providers file: {providers_path}")
1428
2271
 
1429
- _log(f"Watching config files: {config_path}" + (f", {ui_path}" if ui_path else ""))
2272
+ def get_latest_mtime():
2273
+ ret = 0
2274
+ name = "llms.json"
2275
+ if config_path.is_file():
2276
+ ret = config_path.stat().st_mtime
2277
+ name = config_path.name
2278
+ if providers_path.is_file() and providers_path.stat().st_mtime > ret:
2279
+ ret = providers_path.stat().st_mtime
2280
+ name = providers_path.name
2281
+ return ret, name
2282
+
2283
+ latest_mtime, name = get_latest_mtime()
1430
2284
 
1431
2285
  while True:
1432
2286
  await asyncio.sleep(interval)
1433
2287
 
1434
2288
  # Check llms.json
1435
2289
  try:
1436
- if config_path.is_file():
1437
- mtime = config_path.stat().st_mtime
1438
-
1439
- if str(config_path) not in file_mtimes:
1440
- file_mtimes[str(config_path)] = mtime
1441
- elif file_mtimes[str(config_path)] != mtime:
1442
- _log(f"Config file changed: {config_path.name}")
1443
- file_mtimes[str(config_path)] = mtime
2290
+ new_mtime, name = get_latest_mtime()
2291
+ if new_mtime > latest_mtime:
2292
+ _log(f"Config file changed: {name}")
2293
+ latest_mtime = new_mtime
1444
2294
 
1445
- try:
1446
- # Reload llms.json
1447
- with open(config_path, "r") as f:
1448
- g_config = json.load(f)
2295
+ try:
2296
+ # Reload llms.json
2297
+ with open(config_path) as f:
2298
+ g_config = json.load(f)
1449
2299
 
1450
- # Reload providers
1451
- await reload_providers()
1452
- _log("Providers reloaded successfully")
1453
- except Exception as e:
1454
- _log(f"Error reloading config: {e}")
2300
+ # Reload providers
2301
+ await reload_providers()
2302
+ _log("Providers reloaded successfully")
2303
+ except Exception as e:
2304
+ _log(f"Error reloading config: {e}")
1455
2305
  except FileNotFoundError:
1456
2306
  pass
1457
2307
 
1458
- # Check ui.json
1459
- if ui_path:
2308
+
2309
+ def get_session_token(request):
2310
+ return request.query.get("session") or request.headers.get("X-Session-Token") or request.cookies.get("llms-token")
2311
+
2312
+
2313
+ class AppExtensions:
2314
+ """
2315
+ APIs extensions can use to extend the app
2316
+ """
2317
+
2318
+ def __init__(self, cli_args, extra_args):
2319
+ self.cli_args = cli_args
2320
+ self.extra_args = extra_args
2321
+ self.config = None
2322
+ self.error_auth_required = create_error_response("Authentication required", "Unauthorized")
2323
+ self.ui_extensions = []
2324
+ self.chat_request_filters = []
2325
+ self.chat_tool_filters = []
2326
+ self.chat_response_filters = []
2327
+ self.chat_error_filters = []
2328
+ self.server_add_get = []
2329
+ self.server_add_post = []
2330
+ self.server_add_put = []
2331
+ self.server_add_delete = []
2332
+ self.server_add_patch = []
2333
+ self.cache_saved_filters = []
2334
+ self.shutdown_handlers = []
2335
+ self.tools = {}
2336
+ self.tool_definitions = []
2337
+ self.index_headers = []
2338
+ self.index_footers = []
2339
+ self.request_args = {
2340
+ "image_config": dict, # e.g. { "aspect_ratio": "1:1" }
2341
+ "temperature": float, # e.g: 0.7
2342
+ "max_completion_tokens": int, # e.g: 2048
2343
+ "seed": int, # e.g: 42
2344
+ "top_p": float, # e.g: 0.9
2345
+ "frequency_penalty": float, # e.g: 0.5
2346
+ "presence_penalty": float, # e.g: 0.5
2347
+ "stop": list, # e.g: ["Stop"]
2348
+ "reasoning_effort": str, # e.g: minimal, low, medium, high
2349
+ "verbosity": str, # e.g: low, medium, high
2350
+ "service_tier": str, # e.g: auto, default
2351
+ "top_logprobs": int,
2352
+ "safety_identifier": str,
2353
+ "store": bool,
2354
+ "enable_thinking": bool,
2355
+ }
2356
+ self.all_providers = [
2357
+ OpenAiCompatible,
2358
+ MistralProvider,
2359
+ GroqProvider,
2360
+ XaiProvider,
2361
+ CodestralProvider,
2362
+ OllamaProvider,
2363
+ LMStudioProvider,
2364
+ ]
2365
+ self.aspect_ratios = {
2366
+ "1:1": "1024×1024",
2367
+ "2:3": "832×1248",
2368
+ "3:2": "1248×832",
2369
+ "3:4": "864×1184",
2370
+ "4:3": "1184×864",
2371
+ "4:5": "896×1152",
2372
+ "5:4": "1152×896",
2373
+ "9:16": "768×1344",
2374
+ "16:9": "1344×768",
2375
+ "21:9": "1536×672",
2376
+ }
2377
+ self.import_maps = {
2378
+ "vue-prod": "/ui/lib/vue.min.mjs",
2379
+ "vue": "/ui/lib/vue.mjs",
2380
+ "vue-router": "/ui/lib/vue-router.min.mjs",
2381
+ "@servicestack/client": "/ui/lib/servicestack-client.mjs",
2382
+ "@servicestack/vue": "/ui/lib/servicestack-vue.mjs",
2383
+ "idb": "/ui/lib/idb.min.mjs",
2384
+ "marked": "/ui/lib/marked.min.mjs",
2385
+ "highlight.js": "/ui/lib/highlight.min.mjs",
2386
+ "chart.js": "/ui/lib/chart.js",
2387
+ "color.js": "/ui/lib/color.js",
2388
+ "ctx.mjs": "/ui/ctx.mjs",
2389
+ }
2390
+
2391
+ def set_config(self, config):
2392
+ self.config = config
2393
+ self.auth_enabled = self.config.get("auth", {}).get("enabled", False)
2394
+
2395
+ # Authentication middleware helper
2396
+ def check_auth(self, request):
2397
+ """Check if request is authenticated. Returns (is_authenticated, user_data)"""
2398
+ if not self.auth_enabled:
2399
+ return True, None
2400
+
2401
+ # Check for OAuth session token
2402
+ session_token = get_session_token(request)
2403
+ if session_token and session_token in g_sessions:
2404
+ return True, g_sessions[session_token]
2405
+
2406
+ # Check for API key
2407
+ auth_header = request.headers.get("Authorization", "")
2408
+ if auth_header.startswith("Bearer "):
2409
+ api_key = auth_header[7:]
2410
+ if api_key:
2411
+ return True, {"authProvider": "apikey"}
2412
+
2413
+ return False, None
2414
+
2415
+ def get_session(self, request):
2416
+ session_token = get_session_token(request)
2417
+
2418
+ if not session_token or session_token not in g_sessions:
2419
+ return None
2420
+
2421
+ session_data = g_sessions[session_token]
2422
+ return session_data
2423
+
2424
+ def get_username(self, request):
2425
+ session = self.get_session(request)
2426
+ if session:
2427
+ return session.get("userName")
2428
+ return None
2429
+
2430
+ def get_user_path(self, username=None):
2431
+ if username:
2432
+ return home_llms_path(os.path.join("user", username))
2433
+ return home_llms_path(os.path.join("user", "default"))
2434
+
2435
+ def chat_request(self, template=None, text=None, model=None, system_prompt=None):
2436
+ return g_chat_request(template=template, text=text, model=model, system_prompt=system_prompt)
2437
+
2438
+ async def chat_completion(self, chat, context=None):
2439
+ response = await g_chat_completion(chat, context)
2440
+ return response
2441
+
2442
+ def on_cache_saved_filters(self, context):
2443
+ # _log(f"on_cache_saved_filters {len(self.cache_saved_filters)}: {context['url']}")
2444
+ for filter_func in self.cache_saved_filters:
2445
+ filter_func(context)
2446
+
2447
+ async def on_chat_error(self, e, context):
2448
+ # Apply chat error filters
2449
+ if "stackTrace" not in context:
2450
+ context["stackTrace"] = traceback.format_exc()
2451
+ for filter_func in self.chat_error_filters:
1460
2452
  try:
1461
- if ui_path.is_file():
1462
- mtime = ui_path.stat().st_mtime
1463
-
1464
- if str(ui_path) not in file_mtimes:
1465
- file_mtimes[str(ui_path)] = mtime
1466
- elif file_mtimes[str(ui_path)] != mtime:
1467
- _log(f"Config file changed: {ui_path.name}")
1468
- file_mtimes[str(ui_path)] = mtime
1469
- _log("ui.json reloaded - reload page to update")
1470
- except FileNotFoundError:
1471
- pass
2453
+ await filter_func(e, context)
2454
+ except Exception as e:
2455
+ _err("chat error filter failed", e)
2456
+
2457
+ async def on_chat_tool(self, chat, context):
2458
+ m_len = len(chat.get("messages", []))
2459
+ t_len = len(self.chat_tool_filters)
2460
+ _dbg(
2461
+ f"on_tool_call for thread {context.get('threadId', None)} with {m_len} {pluralize('message', m_len)}, invoking {t_len} {pluralize('filter', t_len)}:"
2462
+ )
2463
+ for filter_func in self.chat_tool_filters:
2464
+ await filter_func(chat, context)
2465
+
2466
+ def exit(self, exit_code=0):
2467
+ if len(self.shutdown_handlers) > 0:
2468
+ _dbg(f"running {len(self.shutdown_handlers)} shutdown handlers...")
2469
+ for handler in self.shutdown_handlers:
2470
+ handler()
2471
+
2472
+ _dbg(f"exit({exit_code})")
2473
+ sys.exit(exit_code)
2474
+
2475
+
2476
+ def handler_name(handler):
2477
+ if hasattr(handler, "__name__"):
2478
+ return handler.__name__
2479
+ return "unknown"
2480
+
2481
+
2482
+ class ExtensionContext:
2483
+ def __init__(self, app, path):
2484
+ self.app = app
2485
+ self.cli_args = app.cli_args
2486
+ self.extra_args = app.extra_args
2487
+ self.error_auth_required = app.error_auth_required
2488
+ self.path = path
2489
+ self.name = os.path.basename(path)
2490
+ if self.name.endswith(".py"):
2491
+ self.name = self.name[:-3]
2492
+ self.ext_prefix = f"/ext/{self.name}"
2493
+ self.MOCK = MOCK
2494
+ self.MOCK_DIR = MOCK_DIR
2495
+ self.debug = DEBUG
2496
+ self.verbose = g_verbose
2497
+ self.aspect_ratios = app.aspect_ratios
2498
+ self.request_args = app.request_args
2499
+
2500
+ def chat_to_prompt(self, chat):
2501
+ return chat_to_prompt(chat)
2502
+
2503
+ def chat_to_system_prompt(self, chat):
2504
+ return chat_to_system_prompt(chat)
2505
+
2506
+ def chat_response_to_message(self, response):
2507
+ return chat_response_to_message(response)
2508
+
2509
+ def last_user_prompt(self, chat):
2510
+ return last_user_prompt(chat)
2511
+
2512
+ def to_file_info(self, chat, info=None, response=None):
2513
+ return to_file_info(chat, info=info, response=response)
2514
+
2515
+ def save_image_to_cache(self, base64_data, filename, image_info):
2516
+ return save_image_to_cache(base64_data, filename, image_info)
2517
+
2518
+ def save_bytes_to_cache(self, bytes_data, filename, file_info):
2519
+ return save_bytes_to_cache(bytes_data, filename, file_info)
2520
+
2521
+ def text_from_file(self, path):
2522
+ return text_from_file(path)
2523
+
2524
+ def log(self, message):
2525
+ if self.verbose:
2526
+ print(f"[{self.name}] {message}", flush=True)
2527
+ return message
1472
2528
 
1473
- def main():
1474
- global _ROOT, g_verbose, g_default_model, g_logprefix, g_config, g_config_path, g_ui_path
2529
+ def log_json(self, obj):
2530
+ if self.verbose:
2531
+ print(f"[{self.name}] {json.dumps(obj, indent=2)}", flush=True)
2532
+ return obj
1475
2533
 
1476
- parser = argparse.ArgumentParser(description=f"llms v{VERSION}")
1477
- parser.add_argument('--config', default=None, help='Path to config file', metavar='FILE')
1478
- parser.add_argument('-m', '--model', default=None, help='Model to use')
2534
+ def dbg(self, message):
2535
+ if self.debug:
2536
+ print(f"DEBUG [{self.name}]: {message}", flush=True)
1479
2537
 
1480
- parser.add_argument('--chat', default=None, help='OpenAI Chat Completion Request to send', metavar='REQUEST')
1481
- parser.add_argument('-s', '--system', default=None, help='System prompt to use for chat completion', metavar='PROMPT')
1482
- parser.add_argument('--image', default=None, help='Image input to use in chat completion')
1483
- parser.add_argument('--audio', default=None, help='Audio input to use in chat completion')
1484
- parser.add_argument('--file', default=None, help='File input to use in chat completion')
1485
- parser.add_argument('--args', default=None, help='URL-encoded parameters to add to chat request (e.g. "temperature=0.7&seed=111")', metavar='PARAMS')
1486
- parser.add_argument('--raw', action='store_true', help='Return raw AI JSON response')
2538
+ def err(self, message, e):
2539
+ print(f"ERROR [{self.name}]: {message}", e)
2540
+ if self.verbose:
2541
+ print(traceback.format_exc(), flush=True)
1487
2542
 
1488
- parser.add_argument('--list', action='store_true', help='Show list of enabled providers and their models (alias ls provider?)')
1489
- parser.add_argument('--check', default=None, help='Check validity of models for a provider', metavar='PROVIDER')
2543
+ def error_message(self, e):
2544
+ return to_error_message(e)
1490
2545
 
1491
- parser.add_argument('--serve', default=None, help='Port to start an OpenAI Chat compatible server on', metavar='PORT')
2546
+ def error_response(self, e, stacktrace=False):
2547
+ return to_error_response(e, stacktrace=stacktrace)
1492
2548
 
1493
- parser.add_argument('--enable', default=None, help='Enable a provider', metavar='PROVIDER')
1494
- parser.add_argument('--disable', default=None, help='Disable a provider', metavar='PROVIDER')
1495
- parser.add_argument('--default', default=None, help='Configure the default model to use', metavar='MODEL')
2549
+ def add_provider(self, provider):
2550
+ self.log(f"Registered provider: {provider.__name__}")
2551
+ self.app.all_providers.append(provider)
1496
2552
 
1497
- parser.add_argument('--init', action='store_true', help='Create a default llms.json')
2553
+ def register_ui_extension(self, index):
2554
+ path = os.path.join(self.ext_prefix, index)
2555
+ self.log(f"Registered UI extension: {path}")
2556
+ self.app.ui_extensions.append({"id": self.name, "path": path})
1498
2557
 
1499
- parser.add_argument('--root', default=None, help='Change root directory for UI files', metavar='PATH')
1500
- parser.add_argument('--logprefix', default="", help='Prefix used in log messages', metavar='PREFIX')
1501
- parser.add_argument('--verbose', action='store_true', help='Verbose output')
2558
+ def register_chat_request_filter(self, handler):
2559
+ self.log(f"Registered chat request filter: {handler_name(handler)}")
2560
+ self.app.chat_request_filters.append(handler)
2561
+
2562
+ def register_chat_tool_filter(self, handler):
2563
+ self.log(f"Registered chat tool filter: {handler_name(handler)}")
2564
+ self.app.chat_tool_filters.append(handler)
2565
+
2566
+ def register_chat_response_filter(self, handler):
2567
+ self.log(f"Registered chat response filter: {handler_name(handler)}")
2568
+ self.app.chat_response_filters.append(handler)
2569
+
2570
+ def register_chat_error_filter(self, handler):
2571
+ self.log(f"Registered chat error filter: {handler_name(handler)}")
2572
+ self.app.chat_error_filters.append(handler)
2573
+
2574
+ def register_cache_saved_filter(self, handler):
2575
+ self.log(f"Registered cache saved filter: {handler_name(handler)}")
2576
+ self.app.cache_saved_filters.append(handler)
2577
+
2578
+ def register_shutdown_handler(self, handler):
2579
+ self.log(f"Registered shutdown handler: {handler_name(handler)}")
2580
+ self.app.shutdown_handlers.append(handler)
2581
+
2582
+ def add_static_files(self, ext_dir):
2583
+ self.log(f"Registered static files: {ext_dir}")
2584
+
2585
+ async def serve_static(request):
2586
+ path = request.match_info["path"]
2587
+ file_path = os.path.join(ext_dir, path)
2588
+ if os.path.exists(file_path):
2589
+ return web.FileResponse(file_path)
2590
+ return web.Response(status=404)
2591
+
2592
+ self.app.server_add_get.append((os.path.join(self.ext_prefix, "{path:.*}"), serve_static, {}))
2593
+
2594
+ def add_get(self, path, handler, **kwargs):
2595
+ self.dbg(f"Registered GET: {os.path.join(self.ext_prefix, path)}")
2596
+ self.app.server_add_get.append((os.path.join(self.ext_prefix, path), handler, kwargs))
2597
+
2598
+ def add_post(self, path, handler, **kwargs):
2599
+ self.dbg(f"Registered POST: {os.path.join(self.ext_prefix, path)}")
2600
+ self.app.server_add_post.append((os.path.join(self.ext_prefix, path), handler, kwargs))
2601
+
2602
+ def add_put(self, path, handler, **kwargs):
2603
+ self.dbg(f"Registered PUT: {os.path.join(self.ext_prefix, path)}")
2604
+ self.app.server_add_put.append((os.path.join(self.ext_prefix, path), handler, kwargs))
2605
+
2606
+ def add_delete(self, path, handler, **kwargs):
2607
+ self.dbg(f"Registered DELETE: {os.path.join(self.ext_prefix, path)}")
2608
+ self.app.server_add_delete.append((os.path.join(self.ext_prefix, path), handler, kwargs))
2609
+
2610
+ def add_patch(self, path, handler, **kwargs):
2611
+ self.dbg(f"Registered PATCH: {os.path.join(self.ext_prefix, path)}")
2612
+ self.app.server_add_patch.append((os.path.join(self.ext_prefix, path), handler, kwargs))
2613
+
2614
+ def add_importmaps(self, dict):
2615
+ self.app.import_maps.update(dict)
2616
+
2617
+ def add_index_header(self, html):
2618
+ self.app.index_headers.append(html)
2619
+
2620
+ def add_index_footer(self, html):
2621
+ self.app.index_footers.append(html)
2622
+
2623
+ def get_config(self):
2624
+ return g_config
2625
+
2626
+ def get_cache_path(self, path=""):
2627
+ return get_cache_path(path)
2628
+
2629
+ def chat_request(self, template=None, text=None, model=None, system_prompt=None):
2630
+ return self.app.chat_request(template=template, text=text, model=model, system_prompt=system_prompt)
2631
+
2632
+ def chat_completion(self, chat, context=None):
2633
+ return self.app.chat_completion(chat, context=context)
2634
+
2635
+ def get_providers(self):
2636
+ return g_handlers
2637
+
2638
+ def get_provider(self, name):
2639
+ return g_handlers.get(name)
2640
+
2641
+ def register_tool(self, func, tool_def=None):
2642
+ if tool_def is None:
2643
+ tool_def = function_to_tool_definition(func)
2644
+
2645
+ name = tool_def["function"]["name"]
2646
+ self.log(f"Registered tool: {name}")
2647
+ self.app.tools[name] = func
2648
+ self.app.tool_definitions.append(tool_def)
2649
+
2650
+ def check_auth(self, request):
2651
+ return self.app.check_auth(request)
2652
+
2653
+ def get_session(self, request):
2654
+ return self.app.get_session(request)
2655
+
2656
+ def get_username(self, request):
2657
+ return self.app.get_username(request)
2658
+
2659
+ def get_user_path(self, username=None):
2660
+ return self.app.get_user_path(username)
2661
+
2662
+ def should_cancel_thread(self, context):
2663
+ return should_cancel_thread(context)
2664
+
2665
+ def cache_message_inline_data(self, message):
2666
+ return cache_message_inline_data(message)
2667
+
2668
+ def to_content(self, result):
2669
+ return to_content(result)
2670
+
2671
+
2672
+ def get_extensions_path():
2673
+ return os.getenv("LLMS_EXTENSIONS_DIR", os.path.join(Path.home(), ".llms", "extensions"))
2674
+
2675
+
2676
+ def get_disabled_extensions():
2677
+ ret = DISABLE_EXTENSIONS.copy()
2678
+ if g_config:
2679
+ for ext in g_config.get("disable_extensions", []):
2680
+ if ext not in ret:
2681
+ ret.append(ext)
2682
+ return ret
2683
+
2684
+
2685
+ def get_extensions_dirs():
2686
+ """
2687
+ Returns a list of extension directories.
2688
+ """
2689
+ extensions_path = get_extensions_path()
2690
+ os.makedirs(extensions_path, exist_ok=True)
2691
+
2692
+ # allow overriding builtin extensions
2693
+ override_extensions = []
2694
+ if os.path.exists(extensions_path):
2695
+ override_extensions = os.listdir(extensions_path)
2696
+
2697
+ ret = []
2698
+ disabled_extensions = get_disabled_extensions()
2699
+
2700
+ builtin_extensions_dir = _ROOT / "extensions"
2701
+ if os.path.exists(builtin_extensions_dir):
2702
+ for item in os.listdir(builtin_extensions_dir):
2703
+ if os.path.isdir(os.path.join(builtin_extensions_dir, item)):
2704
+ if item in override_extensions:
2705
+ continue
2706
+ if item in disabled_extensions:
2707
+ continue
2708
+ ret.append(os.path.join(builtin_extensions_dir, item))
2709
+
2710
+ if os.path.exists(extensions_path):
2711
+ for item in os.listdir(extensions_path):
2712
+ if os.path.isdir(os.path.join(extensions_path, item)):
2713
+ if item in disabled_extensions:
2714
+ continue
2715
+ ret.append(os.path.join(extensions_path, item))
2716
+
2717
+ return ret
2718
+
2719
+
2720
+ def init_extensions(parser):
2721
+ """
2722
+ Initializes extensions by loading their __init__.py files and calling the __parser__ function if it exists.
2723
+ """
2724
+ for item_path in get_extensions_dirs():
2725
+ item = os.path.basename(item_path)
2726
+
2727
+ if os.path.isdir(item_path):
2728
+ try:
2729
+ # check for __parser__ function if exists in __init.__.py and call it with parser
2730
+ init_file = os.path.join(item_path, "__init__.py")
2731
+ if os.path.exists(init_file):
2732
+ spec = importlib.util.spec_from_file_location(item, init_file)
2733
+ if spec and spec.loader:
2734
+ module = importlib.util.module_from_spec(spec)
2735
+ sys.modules[item] = module
2736
+ spec.loader.exec_module(module)
2737
+
2738
+ parser_func = getattr(module, "__parser__", None)
2739
+ if callable(parser_func):
2740
+ parser_func(parser)
2741
+ _log(f"Extension {item} parser loaded")
2742
+ except Exception as e:
2743
+ _err(f"Failed to load extension {item} parser", e)
2744
+
2745
+
2746
+ def install_extensions():
2747
+ """
2748
+ Scans ensure ~/.llms/extensions/ for directories with __init__.py and loads them as extensions.
2749
+ Calls the `__install__(ctx)` function in the extension module.
2750
+ """
2751
+
2752
+ extension_dirs = get_extensions_dirs()
2753
+ ext_count = len(list(extension_dirs))
2754
+ if ext_count == 0:
2755
+ _log("No extensions found")
2756
+ return
2757
+
2758
+ disabled_extensions = get_disabled_extensions()
2759
+ if len(disabled_extensions) > 0:
2760
+ _log(f"Disabled extensions: {', '.join(disabled_extensions)}")
2761
+
2762
+ _log(f"Installing {ext_count} extension{'' if ext_count == 1 else 's'}...")
2763
+
2764
+ for item_path in extension_dirs:
2765
+ item = os.path.basename(item_path)
2766
+
2767
+ if os.path.isdir(item_path):
2768
+ sys.path.append(item_path)
2769
+ try:
2770
+ ctx = ExtensionContext(g_app, item_path)
2771
+ init_file = os.path.join(item_path, "__init__.py")
2772
+ if os.path.exists(init_file):
2773
+ spec = importlib.util.spec_from_file_location(item, init_file)
2774
+ if spec and spec.loader:
2775
+ module = importlib.util.module_from_spec(spec)
2776
+ sys.modules[item] = module
2777
+ spec.loader.exec_module(module)
2778
+
2779
+ install_func = getattr(module, "__install__", None)
2780
+ if callable(install_func):
2781
+ install_func(ctx)
2782
+ _log(f"Extension {item} installed")
2783
+ else:
2784
+ _dbg(f"Extension {item} has no __install__ function")
2785
+ else:
2786
+ _dbg(f"Extension {item} has no __init__.py")
2787
+ else:
2788
+ _dbg(f"Extension {init_file} not found")
2789
+
2790
+ # if ui folder exists, serve as static files at /ext/{item}/
2791
+ ui_path = os.path.join(item_path, "ui")
2792
+ if os.path.exists(ui_path):
2793
+ ctx.add_static_files(ui_path)
2794
+
2795
+ # Register UI extension if index.mjs exists (/ext/{item}/index.mjs)
2796
+ if os.path.exists(os.path.join(ui_path, "index.mjs")):
2797
+ ctx.register_ui_extension("index.mjs")
2798
+
2799
+ except Exception as e:
2800
+ _err(f"Failed to install extension {item}", e)
2801
+ else:
2802
+ _dbg(f"Extension {item} not found: {item_path} is not a directory {os.path.exists(item_path)}")
2803
+
2804
+
2805
+ def run_extension_cli():
2806
+ """
2807
+ Run the CLI for an extension.
2808
+ """
2809
+ for item_path in get_extensions_dirs():
2810
+ item = os.path.basename(item_path)
2811
+
2812
+ if os.path.isdir(item_path):
2813
+ init_file = os.path.join(item_path, "__init__.py")
2814
+ if os.path.exists(init_file):
2815
+ ctx = ExtensionContext(g_app, item_path)
2816
+ try:
2817
+ spec = importlib.util.spec_from_file_location(item, init_file)
2818
+ if spec and spec.loader:
2819
+ module = importlib.util.module_from_spec(spec)
2820
+ sys.modules[item] = module
2821
+ spec.loader.exec_module(module)
2822
+
2823
+ # Check for __run__ function if exists in __init__.py and call it with ctx
2824
+ run_func = getattr(module, "__run__", None)
2825
+ if callable(run_func):
2826
+ _log(f"Running extension {item}...")
2827
+ handled = run_func(ctx)
2828
+ return handled
2829
+
2830
+ except Exception as e:
2831
+ _err(f"Failed to run extension {item}", e)
2832
+ return False
2833
+
2834
+
2835
+ def main():
2836
+ global _ROOT, g_verbose, g_default_model, g_logprefix, g_providers, g_config, g_config_path, g_app
2837
+
2838
+ _ROOT = os.getenv("LLMS_ROOT", resolve_root())
2839
+ if not _ROOT:
2840
+ print("Resource root not found")
2841
+ exit(1)
2842
+
2843
+ parser = argparse.ArgumentParser(description=f"llms v{VERSION}")
2844
+ parser.add_argument("--config", default=None, help="Path to config file", metavar="FILE")
2845
+ parser.add_argument("--providers", default=None, help="Path to models.dev providers file", metavar="FILE")
2846
+ parser.add_argument("-m", "--model", default=None, help="Model to use")
2847
+
2848
+ parser.add_argument("--chat", default=None, help="OpenAI Chat Completion Request to send", metavar="REQUEST")
2849
+ parser.add_argument(
2850
+ "-s", "--system", default=None, help="System prompt to use for chat completion", metavar="PROMPT"
2851
+ )
2852
+ parser.add_argument(
2853
+ "--tools", default=None, help="Tools to use for chat completion (all|none|<tool>,<tool>...)", metavar="TOOLS"
2854
+ )
2855
+ parser.add_argument("--image", default=None, help="Image input to use in chat completion")
2856
+ parser.add_argument("--audio", default=None, help="Audio input to use in chat completion")
2857
+ parser.add_argument("--file", default=None, help="File input to use in chat completion")
2858
+ parser.add_argument("--out", default=None, help="Image or Video Generation Request", metavar="MODALITY")
2859
+ parser.add_argument(
2860
+ "--args",
2861
+ default=None,
2862
+ help='URL-encoded parameters to add to chat request (e.g. "temperature=0.7&seed=111")',
2863
+ metavar="PARAMS",
2864
+ )
2865
+ parser.add_argument("--raw", action="store_true", help="Return raw AI JSON response")
2866
+
2867
+ parser.add_argument(
2868
+ "--list", action="store_true", help="Show list of enabled providers and their models (alias ls provider?)"
2869
+ )
2870
+ parser.add_argument("--check", default=None, help="Check validity of models for a provider", metavar="PROVIDER")
2871
+
2872
+ parser.add_argument(
2873
+ "--serve", default=None, help="Port to start an OpenAI Chat compatible server on", metavar="PORT"
2874
+ )
2875
+
2876
+ parser.add_argument("--enable", default=None, help="Enable a provider", metavar="PROVIDER")
2877
+ parser.add_argument("--disable", default=None, help="Disable a provider", metavar="PROVIDER")
2878
+ parser.add_argument("--default", default=None, help="Configure the default model to use", metavar="MODEL")
2879
+
2880
+ parser.add_argument("--init", action="store_true", help="Create a default llms.json")
2881
+ parser.add_argument("--update-providers", action="store_true", help="Update local models.dev providers.json")
2882
+
2883
+ parser.add_argument("--logprefix", default="", help="Prefix used in log messages", metavar="PREFIX")
2884
+ parser.add_argument("--verbose", action="store_true", help="Verbose output")
2885
+
2886
+ parser.add_argument(
2887
+ "--add",
2888
+ nargs="?",
2889
+ const="ls",
2890
+ default=None,
2891
+ help="Install an extension (lists available extensions if no name provided)",
2892
+ metavar="EXTENSION",
2893
+ )
2894
+ parser.add_argument(
2895
+ "--remove",
2896
+ nargs="?",
2897
+ const="ls",
2898
+ default=None,
2899
+ help="Remove an extension (lists installed extensions if no name provided)",
2900
+ metavar="EXTENSION",
2901
+ )
2902
+
2903
+ parser.add_argument(
2904
+ "--update",
2905
+ nargs="?",
2906
+ const="ls",
2907
+ default=None,
2908
+ help="Update an extension (use 'all' to update all extensions)",
2909
+ metavar="EXTENSION",
2910
+ )
2911
+
2912
+ # Load parser extensions, go through all extensions and load their parser arguments
2913
+ init_extensions(parser)
1502
2914
 
1503
2915
  cli_args, extra_args = parser.parse_known_args()
1504
2916
 
2917
+ g_app = AppExtensions(cli_args, extra_args)
2918
+
1505
2919
  # Check for verbose mode from CLI argument or environment variables
1506
- verbose_env = os.environ.get('VERBOSE', '').lower()
1507
- if cli_args.verbose or verbose_env in ('1', 'true'):
2920
+ verbose_env = os.getenv("VERBOSE", "").lower()
2921
+ if cli_args.verbose or verbose_env in ("1", "true"):
1508
2922
  g_verbose = True
1509
2923
  # printdump(cli_args)
1510
2924
  if cli_args.model:
@@ -1512,13 +2926,9 @@ def main():
1512
2926
  if cli_args.logprefix:
1513
2927
  g_logprefix = cli_args.logprefix
1514
2928
 
1515
- _ROOT = Path(cli_args.root) if cli_args.root else resolve_root()
1516
- if not _ROOT:
1517
- print("Resource root not found")
1518
- exit(1)
1519
-
1520
2929
  home_config_path = home_llms_path("llms.json")
1521
- home_ui_path = home_llms_path("ui.json")
2930
+ home_providers_path = home_llms_path("providers.json")
2931
+ home_providers_extra_path = home_llms_path("providers-extra.json")
1522
2932
 
1523
2933
  if cli_args.init:
1524
2934
  if os.path.exists(home_config_path):
@@ -1527,38 +2937,215 @@ def main():
1527
2937
  asyncio.run(save_default_config(home_config_path))
1528
2938
  print(f"Created default config at {home_config_path}")
1529
2939
 
1530
- if os.path.exists(home_ui_path):
1531
- print(f"ui.json already exists at {home_ui_path}")
2940
+ if os.path.exists(home_providers_path):
2941
+ print(f"providers.json already exists at {home_providers_path}")
2942
+ else:
2943
+ asyncio.run(save_text_url(github_url("providers.json"), home_providers_path))
2944
+ print(f"Created default providers config at {home_providers_path}")
2945
+
2946
+ if os.path.exists(home_providers_extra_path):
2947
+ print(f"providers-extra.json already exists at {home_providers_extra_path}")
1532
2948
  else:
1533
- asyncio.run(save_text_url(github_url("ui.json"), home_ui_path))
1534
- print(f"Created default ui config at {home_ui_path}")
2949
+ asyncio.run(save_text_url(github_url("providers-extra.json"), home_providers_extra_path))
2950
+ print(f"Created default extra providers config at {home_providers_extra_path}")
1535
2951
  exit(0)
1536
2952
 
2953
+ if cli_args.providers:
2954
+ if not os.path.exists(cli_args.providers):
2955
+ print(f"providers.json not found at {cli_args.providers}")
2956
+ exit(1)
2957
+ g_providers = json.loads(text_from_file(cli_args.providers))
2958
+
1537
2959
  if cli_args.config:
1538
2960
  # read contents
1539
2961
  g_config_path = cli_args.config
1540
- with open(g_config_path, "r") as f:
2962
+ with open(g_config_path, encoding="utf-8") as f:
1541
2963
  config_json = f.read()
1542
- g_config = json.loads(config_json)
2964
+ g_config = load_config_json(config_json)
1543
2965
 
1544
2966
  config_dir = os.path.dirname(g_config_path)
1545
- # look for ui.json in same directory as config
1546
- ui_path = os.path.join(config_dir, "ui.json")
1547
- if os.path.exists(ui_path):
1548
- g_ui_path = ui_path
1549
- else:
1550
- if not os.path.exists(home_ui_path):
1551
- ui_json = text_from_resource("ui.json")
1552
- with open(home_ui_path, "w") as f:
1553
- f.write(ui_json)
1554
- _log(f"Created default ui config at {home_ui_path}")
1555
- g_ui_path = home_ui_path
2967
+
2968
+ if not g_providers and os.path.exists(os.path.join(config_dir, "providers.json")):
2969
+ g_providers = json.loads(text_from_file(os.path.join(config_dir, "providers.json")))
2970
+
1556
2971
  else:
1557
- # ensure llms.json and ui.json exist in home directory
2972
+ # ensure llms.json and providers.json exist in home directory
1558
2973
  asyncio.run(save_home_configs())
1559
2974
  g_config_path = home_config_path
1560
- g_ui_path = home_ui_path
1561
- g_config = json.loads(text_from_file(g_config_path))
2975
+ g_config = load_config_json(text_from_file(g_config_path))
2976
+
2977
+ g_app.set_config(g_config)
2978
+
2979
+ if not g_providers:
2980
+ g_providers = json.loads(text_from_file(home_providers_path))
2981
+
2982
+ if cli_args.update_providers:
2983
+ asyncio.run(update_providers(home_providers_path))
2984
+ print(f"Updated {home_providers_path}")
2985
+ exit(0)
2986
+
2987
+ # if home_providers_path is older than 1 day, update providers list
2988
+ if (
2989
+ os.path.exists(home_providers_path)
2990
+ and (time.time() - os.path.getmtime(home_providers_path)) > 86400
2991
+ and os.getenv("LLMS_DISABLE_UPDATE", "") != "1"
2992
+ ):
2993
+ try:
2994
+ asyncio.run(update_providers(home_providers_path))
2995
+ _log(f"Updated {home_providers_path}")
2996
+ except Exception as e:
2997
+ _err("Failed to update providers", e)
2998
+
2999
+ if cli_args.add is not None:
3000
+ if cli_args.add == "ls":
3001
+
3002
+ async def list_extensions():
3003
+ print("\nAvailable extensions:")
3004
+ text = await get_text("https://api.github.com/orgs/llmspy/repos?per_page=100&sort=updated")
3005
+ repos = json.loads(text)
3006
+ max_name_length = 0
3007
+ for repo in repos:
3008
+ max_name_length = max(max_name_length, len(repo["name"]))
3009
+
3010
+ for repo in repos:
3011
+ print(f" {repo['name']:<{max_name_length + 2}} {repo['description']}")
3012
+
3013
+ print("\nUsage:")
3014
+ print(" llms --add <extension>")
3015
+ print(" llms --add <github-user>/<repo>")
3016
+
3017
+ asyncio.run(list_extensions())
3018
+ exit(0)
3019
+
3020
+ async def install_extension(name):
3021
+ # Determine git URL and target directory name
3022
+ if "/" in name:
3023
+ git_url = f"https://github.com/{name}"
3024
+ target_name = name.split("/")[-1]
3025
+ else:
3026
+ git_url = f"https://github.com/llmspy/{name}"
3027
+ target_name = name
3028
+
3029
+ # check extension is not already installed
3030
+ extensions_path = get_extensions_path()
3031
+ target_path = os.path.join(extensions_path, target_name)
3032
+
3033
+ if os.path.exists(target_path):
3034
+ print(f"Extension {target_name} is already installed at {target_path}")
3035
+ return
3036
+
3037
+ print(f"Installing extension: {name}")
3038
+ print(f"Cloning from {git_url} to {target_path}...")
3039
+
3040
+ try:
3041
+ subprocess.run(["git", "clone", git_url, target_path], check=True)
3042
+
3043
+ # Check for requirements.txt
3044
+ requirements_path = os.path.join(target_path, "requirements.txt")
3045
+ if os.path.exists(requirements_path):
3046
+ print(f"Installing dependencies from {requirements_path}...")
3047
+
3048
+ # Check if uv is installed
3049
+ has_uv = False
3050
+ try:
3051
+ subprocess.run(
3052
+ ["uv", "--version"], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL, check=True
3053
+ )
3054
+ has_uv = True
3055
+ except (subprocess.CalledProcessError, FileNotFoundError):
3056
+ pass
3057
+
3058
+ if has_uv:
3059
+ subprocess.run(
3060
+ ["uv", "pip", "install", "-p", sys.executable, "-r", "requirements.txt"],
3061
+ cwd=target_path,
3062
+ check=True,
3063
+ )
3064
+ else:
3065
+ subprocess.run(
3066
+ [sys.executable, "-m", "pip", "install", "-r", "requirements.txt"],
3067
+ cwd=target_path,
3068
+ check=True,
3069
+ )
3070
+ print("Dependencies installed successfully.")
3071
+
3072
+ print(f"Extension {target_name} installed successfully.")
3073
+
3074
+ except subprocess.CalledProcessError as e:
3075
+ print(f"Failed to install extension: {e}")
3076
+ # cleanup if clone failed but directory was created (unlikely with simple git clone but good practice)
3077
+ if os.path.exists(target_path) and not os.listdir(target_path):
3078
+ os.rmdir(target_path)
3079
+
3080
+ asyncio.run(install_extension(cli_args.add))
3081
+ exit(0)
3082
+
3083
+ if cli_args.remove is not None:
3084
+ if cli_args.remove == "ls":
3085
+ # List installed extensions
3086
+ extensions_path = get_extensions_path()
3087
+ extensions = os.listdir(extensions_path)
3088
+ if len(extensions) == 0:
3089
+ print("No extensions installed.")
3090
+ exit(0)
3091
+ print("Installed extensions:")
3092
+ for extension in extensions:
3093
+ print(f" {extension}")
3094
+ exit(0)
3095
+ # Remove an extension
3096
+ extension_name = cli_args.remove
3097
+ extensions_path = get_extensions_path()
3098
+ target_path = os.path.join(extensions_path, extension_name)
3099
+
3100
+ if not os.path.exists(target_path):
3101
+ print(f"Extension {extension_name} not found at {target_path}")
3102
+ exit(1)
3103
+
3104
+ print(f"Removing extension: {extension_name}...")
3105
+ try:
3106
+ shutil.rmtree(target_path)
3107
+ print(f"Extension {extension_name} removed successfully.")
3108
+ except Exception as e:
3109
+ print(f"Failed to remove extension: {e}")
3110
+ exit(1)
3111
+
3112
+ exit(0)
3113
+
3114
+ if cli_args.update:
3115
+ if cli_args.update == "ls":
3116
+ # List installed extensions
3117
+ extensions_path = get_extensions_path()
3118
+ extensions = os.listdir(extensions_path)
3119
+ if len(extensions) == 0:
3120
+ print("No extensions installed.")
3121
+ exit(0)
3122
+ print("Installed extensions:")
3123
+ for extension in extensions:
3124
+ print(f" {extension}")
3125
+
3126
+ print("\nUsage:")
3127
+ print(" llms --update <extension>")
3128
+ print(" llms --update all")
3129
+ exit(0)
3130
+
3131
+ async def update_extensions(extension_name):
3132
+ extensions_path = get_extensions_path()
3133
+ for extension in os.listdir(extensions_path):
3134
+ extension_path = os.path.join(extensions_path, extension)
3135
+ if os.path.isdir(extension_path):
3136
+ if extension_name != "all" and extension != extension_name:
3137
+ continue
3138
+ result = subprocess.run(["git", "pull"], cwd=extension_path, capture_output=True)
3139
+ if result.returncode != 0:
3140
+ print(f"Failed to update extension {extension}: {result.stderr.decode('utf-8')}")
3141
+ continue
3142
+ print(f"Updated extension {extension}")
3143
+ _log(result.stdout.decode("utf-8"))
3144
+
3145
+ asyncio.run(update_extensions(cli_args.update))
3146
+ exit(0)
3147
+
3148
+ install_extensions()
1562
3149
 
1563
3150
  asyncio.run(reload_providers())
1564
3151
 
@@ -1568,7 +3155,7 @@ def main():
1568
3155
  filter_list = []
1569
3156
  if len(extra_args) > 0:
1570
3157
  arg = extra_args[0]
1571
- if arg == 'ls':
3158
+ if arg == "ls":
1572
3159
  cli_args.list = True
1573
3160
  if len(extra_args) > 1:
1574
3161
  filter_list = extra_args[1:]
@@ -1576,36 +3163,57 @@ def main():
1576
3163
  if cli_args.list:
1577
3164
  # Show list of enabled providers and their models
1578
3165
  enabled = []
3166
+ provider_count = 0
3167
+ model_count = 0
3168
+
3169
+ max_model_length = 0
1579
3170
  for name, provider in g_handlers.items():
1580
3171
  if len(filter_list) > 0 and name not in filter_list:
1581
3172
  continue
3173
+ for model in provider.models:
3174
+ max_model_length = max(max_model_length, len(model))
3175
+
3176
+ for name, provider in g_handlers.items():
3177
+ if len(filter_list) > 0 and name not in filter_list:
3178
+ continue
3179
+ provider_count += 1
1582
3180
  print(f"{name}:")
1583
3181
  enabled.append(name)
1584
3182
  for model in provider.models:
1585
- print(f" {model}")
3183
+ model_count += 1
3184
+ model_cost_info = None
3185
+ if "cost" in provider.models[model]:
3186
+ model_cost = provider.models[model]["cost"]
3187
+ if "input" in model_cost and "output" in model_cost:
3188
+ if model_cost["input"] == 0 and model_cost["output"] == 0:
3189
+ model_cost_info = " 0"
3190
+ else:
3191
+ model_cost_info = f"{model_cost['input']:5} / {model_cost['output']}"
3192
+ print(f" {model:{max_model_length}} {model_cost_info or ''}")
3193
+
3194
+ print(f"\n{model_count} models available from {provider_count} providers")
1586
3195
 
1587
3196
  print_status()
1588
- exit(0)
3197
+ g_app.exit(0)
1589
3198
 
1590
3199
  if cli_args.check is not None:
1591
3200
  # Check validity of models for a provider
1592
3201
  provider_name = cli_args.check
1593
3202
  model_names = extra_args if len(extra_args) > 0 else None
1594
3203
  asyncio.run(check_models(provider_name, model_names))
1595
- exit(0)
3204
+ g_app.exit(0)
1596
3205
 
1597
3206
  if cli_args.serve is not None:
1598
3207
  # Disable inactive providers and save to config before starting server
1599
- all_providers = g_config['providers'].keys()
3208
+ all_providers = g_config["providers"].keys()
1600
3209
  enabled_providers = list(g_handlers.keys())
1601
3210
  disable_providers = []
1602
3211
  for provider in all_providers:
1603
- provider_config = g_config['providers'][provider]
1604
- if provider not in enabled_providers:
1605
- if 'enabled' in provider_config and provider_config['enabled']:
1606
- provider_config['enabled'] = False
1607
- disable_providers.append(provider)
1608
-
3212
+ provider_config = g_config["providers"][provider]
3213
+ if provider not in enabled_providers and "enabled" in provider_config and provider_config["enabled"]:
3214
+ provider_config["enabled"] = False
3215
+ disable_providers.append(provider)
3216
+
1609
3217
  if len(disable_providers) > 0:
1610
3218
  _log(f"Disabled unavailable providers: {', '.join(disable_providers)}")
1611
3219
  save_config(g_config)
@@ -1613,24 +3221,28 @@ def main():
1613
3221
  # Start server
1614
3222
  port = int(cli_args.serve)
1615
3223
 
1616
- if not os.path.exists(g_ui_path):
1617
- print(f"UI not found at {g_ui_path}")
1618
- exit(1)
1619
-
1620
3224
  # Validate auth configuration if enabled
1621
- auth_enabled = g_config.get('auth', {}).get('enabled', False)
3225
+ auth_enabled = g_config.get("auth", {}).get("enabled", False)
1622
3226
  if auth_enabled:
1623
- github_config = g_config.get('auth', {}).get('github', {})
1624
- client_id = github_config.get('client_id', '')
1625
- client_secret = github_config.get('client_secret', '')
3227
+ github_config = g_config.get("auth", {}).get("github", {})
3228
+ client_id = github_config.get("client_id", "")
3229
+ client_secret = github_config.get("client_secret", "")
1626
3230
 
1627
3231
  # Expand environment variables
1628
- if client_id.startswith('$'):
1629
- client_id = os.environ.get(client_id[1:], '')
1630
- if client_secret.startswith('$'):
1631
- client_secret = os.environ.get(client_secret[1:], '')
1632
-
1633
- if not client_id or not client_secret:
3232
+ if client_id.startswith("$"):
3233
+ client_id = client_id[1:]
3234
+ if client_secret.startswith("$"):
3235
+ client_secret = client_secret[1:]
3236
+
3237
+ client_id = os.getenv(client_id, client_id)
3238
+ client_secret = os.getenv(client_secret, client_secret)
3239
+
3240
+ if (
3241
+ not client_id
3242
+ or not client_secret
3243
+ or client_id == "GITHUB_CLIENT_ID"
3244
+ or client_secret == "GITHUB_CLIENT_SECRET"
3245
+ ):
1634
3246
  print("ERROR: Authentication is enabled but GitHub OAuth is not properly configured.")
1635
3247
  print("Please set GITHUB_CLIENT_ID and GITHUB_CLIENT_SECRET environment variables,")
1636
3248
  print("or disable authentication by setting 'auth.enabled' to false in llms.json")
@@ -1638,157 +3250,290 @@ def main():
1638
3250
 
1639
3251
  _log("Authentication enabled - GitHub OAuth configured")
1640
3252
 
1641
- client_max_size = g_config.get('limits', {}).get('client_max_size', 20*1024*1024) # 20MB max request size (to handle base64 encoding overhead)
1642
- _log(f"client_max_size set to {client_max_size} bytes ({client_max_size/1024/1024:.1f}MB)")
3253
+ client_max_size = g_config.get("limits", {}).get(
3254
+ "client_max_size", 20 * 1024 * 1024
3255
+ ) # 20MB max request size (to handle base64 encoding overhead)
3256
+ _log(f"client_max_size set to {client_max_size} bytes ({client_max_size / 1024 / 1024:.1f}MB)")
1643
3257
  app = web.Application(client_max_size=client_max_size)
1644
3258
 
1645
- # Authentication middleware helper
1646
- def check_auth(request):
1647
- """Check if request is authenticated. Returns (is_authenticated, user_data)"""
1648
- if not auth_enabled:
1649
- return True, None
1650
-
1651
- # Check for OAuth session token
1652
- session_token = request.query.get('session') or request.headers.get('X-Session-Token')
1653
- if session_token and session_token in g_sessions:
1654
- return True, g_sessions[session_token]
1655
-
1656
- # Check for API key
1657
- auth_header = request.headers.get('Authorization', '')
1658
- if auth_header.startswith('Bearer '):
1659
- api_key = auth_header[7:]
1660
- if api_key:
1661
- return True, {"authProvider": "apikey"}
1662
-
1663
- return False, None
1664
-
1665
3259
  async def chat_handler(request):
1666
3260
  # Check authentication if enabled
1667
- is_authenticated, user_data = check_auth(request)
3261
+ is_authenticated, user_data = g_app.check_auth(request)
1668
3262
  if not is_authenticated:
1669
- return web.json_response({
1670
- "error": {
1671
- "message": "Authentication required",
1672
- "type": "authentication_error",
1673
- "code": "unauthorized"
1674
- }
1675
- }, status=401)
3263
+ return web.json_response(g_app.error_auth_required, status=401)
1676
3264
 
1677
3265
  try:
1678
3266
  chat = await request.json()
1679
- response = await chat_completion(chat)
3267
+ context = {"chat": chat, "request": request, "user": g_app.get_username(request)}
3268
+ metadata = chat.get("metadata", {})
3269
+ context["threadId"] = metadata.get("threadId", None)
3270
+ context["tools"] = metadata.get("tools", "all")
3271
+ response = await g_app.chat_completion(chat, context)
1680
3272
  return web.json_response(response)
1681
3273
  except Exception as e:
1682
- return web.json_response({"error": str(e)}, status=500)
1683
- app.router.add_post('/v1/chat/completions', chat_handler)
3274
+ return web.json_response(to_error_response(e), status=500)
1684
3275
 
1685
- async def models_handler(request):
1686
- return web.json_response(get_models())
1687
- app.router.add_get('/models/list', models_handler)
3276
+ app.router.add_post("/v1/chat/completions", chat_handler)
1688
3277
 
1689
3278
  async def active_models_handler(request):
1690
3279
  return web.json_response(get_active_models())
1691
- app.router.add_get('/models', active_models_handler)
3280
+
3281
+ app.router.add_get("/models", active_models_handler)
3282
+
3283
+ async def active_providers_handler(request):
3284
+ return web.json_response(api_providers())
3285
+
3286
+ app.router.add_get("/providers", active_providers_handler)
1692
3287
 
1693
3288
  async def status_handler(request):
1694
3289
  enabled, disabled = provider_status()
1695
- return web.json_response({
1696
- "all": list(g_config['providers'].keys()),
1697
- "enabled": enabled,
1698
- "disabled": disabled,
1699
- })
1700
- app.router.add_get('/status', status_handler)
3290
+ return web.json_response(
3291
+ {
3292
+ "all": list(g_config["providers"].keys()),
3293
+ "enabled": enabled,
3294
+ "disabled": disabled,
3295
+ }
3296
+ )
3297
+
3298
+ app.router.add_get("/status", status_handler)
1701
3299
 
1702
3300
  async def provider_handler(request):
1703
- provider = request.match_info.get('provider', "")
3301
+ provider = request.match_info.get("provider", "")
1704
3302
  data = await request.json()
1705
3303
  msg = None
1706
- if provider:
1707
- if data.get('enable', False):
3304
+ if provider:
3305
+ if data.get("enable", False):
1708
3306
  provider_config, msg = enable_provider(provider)
1709
- _log(f"Enabled provider {provider}")
1710
- await load_llms()
1711
- elif data.get('disable', False):
3307
+ _log(f"Enabled provider {provider} {msg}")
3308
+ if not msg:
3309
+ await load_llms()
3310
+ elif data.get("disable", False):
1712
3311
  disable_provider(provider)
1713
3312
  _log(f"Disabled provider {provider}")
1714
3313
  enabled, disabled = provider_status()
1715
- return web.json_response({
1716
- "enabled": enabled,
1717
- "disabled": disabled,
1718
- "feedback": msg or "",
1719
- })
1720
- app.router.add_post('/providers/{provider}', provider_handler)
3314
+ return web.json_response(
3315
+ {
3316
+ "enabled": enabled,
3317
+ "disabled": disabled,
3318
+ "feedback": msg or "",
3319
+ }
3320
+ )
3321
+
3322
+ app.router.add_post("/providers/{provider}", provider_handler)
3323
+
3324
+ async def upload_handler(request):
3325
+ # Check authentication if enabled
3326
+ is_authenticated, user_data = g_app.check_auth(request)
3327
+ if not is_authenticated:
3328
+ return web.json_response(g_app.error_auth_required, status=401)
3329
+
3330
+ reader = await request.multipart()
3331
+
3332
+ # Read first file field
3333
+ field = await reader.next()
3334
+ while field and field.name != "file":
3335
+ field = await reader.next()
3336
+
3337
+ if not field:
3338
+ return web.json_response(create_error_response("No file provided"), status=400)
3339
+
3340
+ filename = field.filename or "file"
3341
+ content = await field.read()
3342
+ mimetype = get_file_mime_type(filename)
3343
+
3344
+ # If image, resize if needed
3345
+ if mimetype.startswith("image/"):
3346
+ content, mimetype = convert_image_if_needed(content, mimetype)
3347
+
3348
+ # Calculate SHA256
3349
+ sha256_hash = hashlib.sha256(content).hexdigest()
3350
+ ext = filename.rsplit(".", 1)[1] if "." in filename else ""
3351
+ if not ext:
3352
+ ext = mimetypes.guess_extension(mimetype) or ""
3353
+ if ext.startswith("."):
3354
+ ext = ext[1:]
3355
+
3356
+ if not ext:
3357
+ ext = "bin"
3358
+
3359
+ save_filename = f"{sha256_hash}.{ext}" if ext else sha256_hash
3360
+
3361
+ # Use first 2 chars for subdir to avoid too many files in one dir
3362
+ subdir = sha256_hash[:2]
3363
+ relative_path = f"{subdir}/{save_filename}"
3364
+ full_path = get_cache_path(relative_path)
3365
+
3366
+ # if file and its .info.json already exists, return it
3367
+ info_path = os.path.splitext(full_path)[0] + ".info.json"
3368
+ if os.path.exists(full_path) and os.path.exists(info_path):
3369
+ return web.json_response(json.load(open(info_path)))
3370
+
3371
+ os.makedirs(os.path.dirname(full_path), exist_ok=True)
3372
+
3373
+ with open(full_path, "wb") as f:
3374
+ f.write(content)
3375
+
3376
+ url = f"/~cache/{relative_path}"
3377
+ response_data = {
3378
+ "date": int(time.time()),
3379
+ "url": url,
3380
+ "size": len(content),
3381
+ "type": mimetype,
3382
+ "name": filename,
3383
+ }
3384
+
3385
+ # If image, get dimensions
3386
+ if HAS_PIL and mimetype.startswith("image/"):
3387
+ try:
3388
+ with Image.open(BytesIO(content)) as img:
3389
+ response_data["width"] = img.width
3390
+ response_data["height"] = img.height
3391
+ except Exception:
3392
+ pass
3393
+
3394
+ # Save metadata
3395
+ info_path = os.path.splitext(full_path)[0] + ".info.json"
3396
+ with open(info_path, "w") as f:
3397
+ json.dump(response_data, f)
3398
+
3399
+ g_app.on_cache_saved_filters({"url": url, "info": response_data})
3400
+
3401
+ return web.json_response(response_data)
3402
+
3403
+ app.router.add_post("/upload", upload_handler)
3404
+
3405
+ async def extensions_handler(request):
3406
+ return web.json_response(g_app.ui_extensions)
3407
+
3408
+ app.router.add_get("/ext", extensions_handler)
3409
+
3410
+ async def tools_handler(request):
3411
+ return web.json_response(g_app.tool_definitions)
3412
+
3413
+ app.router.add_get("/ext/tools", tools_handler)
3414
+
3415
+ async def cache_handler(request):
3416
+ path = request.match_info["tail"]
3417
+ full_path = get_cache_path(path)
3418
+
3419
+ if "info" in request.query:
3420
+ info_path = os.path.splitext(full_path)[0] + ".info.json"
3421
+ if not os.path.exists(info_path):
3422
+ return web.Response(text="404: Not Found", status=404)
3423
+
3424
+ # Check for directory traversal for info path
3425
+ try:
3426
+ cache_root = Path(get_cache_path())
3427
+ requested_path = Path(info_path).resolve()
3428
+ if not str(requested_path).startswith(str(cache_root)):
3429
+ return web.Response(text="403: Forbidden", status=403)
3430
+ except Exception:
3431
+ return web.Response(text="403: Forbidden", status=403)
3432
+
3433
+ with open(info_path) as f:
3434
+ content = f.read()
3435
+ return web.Response(text=content, content_type="application/json")
3436
+
3437
+ if not os.path.exists(full_path):
3438
+ return web.Response(text="404: Not Found", status=404)
3439
+
3440
+ # Check for directory traversal
3441
+ try:
3442
+ cache_root = Path(get_cache_path())
3443
+ requested_path = Path(full_path).resolve()
3444
+ if not str(requested_path).startswith(str(cache_root)):
3445
+ return web.Response(text="403: Forbidden", status=403)
3446
+ except Exception:
3447
+ return web.Response(text="403: Forbidden", status=403)
3448
+
3449
+ with open(full_path, "rb") as f:
3450
+ content = f.read()
3451
+
3452
+ mimetype = get_file_mime_type(full_path)
3453
+ return web.Response(body=content, content_type=mimetype)
3454
+
3455
+ app.router.add_get("/~cache/{tail:.*}", cache_handler)
1721
3456
 
1722
3457
  # OAuth handlers
1723
3458
  async def github_auth_handler(request):
1724
3459
  """Initiate GitHub OAuth flow"""
1725
- if 'auth' not in g_config or 'github' not in g_config['auth']:
1726
- return web.json_response({"error": "GitHub OAuth not configured"}, status=500)
3460
+ if "auth" not in g_config or "github" not in g_config["auth"]:
3461
+ return web.json_response(create_error_response("GitHub OAuth not configured"), status=500)
1727
3462
 
1728
- auth_config = g_config['auth']['github']
1729
- client_id = auth_config.get('client_id', '')
1730
- redirect_uri = auth_config.get('redirect_uri', '')
3463
+ auth_config = g_config["auth"]["github"]
3464
+ client_id = auth_config.get("client_id", "")
3465
+ redirect_uri = auth_config.get("redirect_uri", "")
1731
3466
 
1732
3467
  # Expand environment variables
1733
- if client_id.startswith('$'):
1734
- client_id = os.environ.get(client_id[1:], '')
1735
- if redirect_uri.startswith('$'):
1736
- redirect_uri = os.environ.get(redirect_uri[1:], '')
3468
+ if client_id.startswith("$"):
3469
+ client_id = client_id[1:]
3470
+ if redirect_uri.startswith("$"):
3471
+ redirect_uri = redirect_uri[1:]
3472
+
3473
+ client_id = os.getenv(client_id, client_id)
3474
+ redirect_uri = os.getenv(redirect_uri, redirect_uri)
1737
3475
 
1738
3476
  if not client_id:
1739
- return web.json_response({"error": "GitHub client_id not configured"}, status=500)
3477
+ return web.json_response(create_error_response("GitHub client_id not configured"), status=500)
1740
3478
 
1741
3479
  # Generate CSRF state token
1742
3480
  state = secrets.token_urlsafe(32)
1743
- g_oauth_states[state] = {
1744
- 'created': time.time(),
1745
- 'redirect_uri': redirect_uri
1746
- }
3481
+ g_oauth_states[state] = {"created": time.time(), "redirect_uri": redirect_uri}
1747
3482
 
1748
3483
  # Clean up old states (older than 10 minutes)
1749
3484
  current_time = time.time()
1750
- expired_states = [s for s, data in g_oauth_states.items() if current_time - data['created'] > 600]
3485
+ expired_states = [s for s, data in g_oauth_states.items() if current_time - data["created"] > 600]
1751
3486
  for s in expired_states:
1752
3487
  del g_oauth_states[s]
1753
3488
 
1754
3489
  # Build GitHub authorization URL
1755
3490
  params = {
1756
- 'client_id': client_id,
1757
- 'redirect_uri': redirect_uri,
1758
- 'state': state,
1759
- 'scope': 'read:user user:email'
3491
+ "client_id": client_id,
3492
+ "redirect_uri": redirect_uri,
3493
+ "state": state,
3494
+ "scope": "read:user user:email",
1760
3495
  }
1761
3496
  auth_url = f"https://github.com/login/oauth/authorize?{urlencode(params)}"
1762
3497
 
1763
3498
  return web.HTTPFound(auth_url)
1764
-
3499
+
1765
3500
  def validate_user(github_username):
1766
- auth_config = g_config['auth']['github']
3501
+ auth_config = g_config["auth"]["github"]
1767
3502
  # Check if user is restricted
1768
- restrict_to = auth_config.get('restrict_to', '')
3503
+ restrict_to = auth_config.get("restrict_to", "")
1769
3504
 
1770
3505
  # Expand environment variables
1771
- if restrict_to.startswith('$'):
1772
- restrict_to = os.environ.get(restrict_to[1:], '')
3506
+ if restrict_to.startswith("$"):
3507
+ restrict_to = restrict_to[1:]
3508
+
3509
+ restrict_to = os.getenv(restrict_to, None if restrict_to == "GITHUB_USERS" else restrict_to)
1773
3510
 
1774
3511
  # If restrict_to is configured, validate the user
1775
3512
  if restrict_to:
1776
3513
  # Parse allowed users (comma or space delimited)
1777
- allowed_users = [u.strip() for u in re.split(r'[,\s]+', restrict_to) if u.strip()]
3514
+ allowed_users = [u.strip() for u in re.split(r"[,\s]+", restrict_to) if u.strip()]
1778
3515
 
1779
3516
  # Check if user is in the allowed list
1780
3517
  if not github_username or github_username not in allowed_users:
1781
3518
  _log(f"Access denied for user: {github_username}. Not in allowed list: {allowed_users}")
1782
3519
  return web.Response(
1783
3520
  text=f"Access denied. User '{github_username}' is not authorized to access this application.",
1784
- status=403
3521
+ status=403,
1785
3522
  )
1786
3523
  return None
1787
3524
 
1788
3525
  async def github_callback_handler(request):
1789
3526
  """Handle GitHub OAuth callback"""
1790
- code = request.query.get('code')
1791
- state = request.query.get('state')
3527
+ code = request.query.get("code")
3528
+ state = request.query.get("state")
3529
+
3530
+ # Handle malformed URLs where query params are appended with & instead of ?
3531
+ if not code and "tail" in request.match_info:
3532
+ tail = request.match_info["tail"]
3533
+ if tail.startswith("&"):
3534
+ params = parse_qs(tail[1:])
3535
+ code = params.get("code", [None])[0]
3536
+ state = params.get("state", [None])[0]
1792
3537
 
1793
3538
  if not code or not state:
1794
3539
  return web.Response(text="Missing code or state parameter", status=400)
@@ -1797,118 +3542,122 @@ def main():
1797
3542
  if state not in g_oauth_states:
1798
3543
  return web.Response(text="Invalid state parameter", status=400)
1799
3544
 
1800
- state_data = g_oauth_states.pop(state)
3545
+ g_oauth_states.pop(state)
1801
3546
 
1802
- if 'auth' not in g_config or 'github' not in g_config['auth']:
1803
- return web.json_response({"error": "GitHub OAuth not configured"}, status=500)
3547
+ if "auth" not in g_config or "github" not in g_config["auth"]:
3548
+ return web.json_response(create_error_response("GitHub OAuth not configured"), status=500)
1804
3549
 
1805
- auth_config = g_config['auth']['github']
1806
- client_id = auth_config.get('client_id', '')
1807
- client_secret = auth_config.get('client_secret', '')
1808
- redirect_uri = auth_config.get('redirect_uri', '')
3550
+ auth_config = g_config["auth"]["github"]
3551
+ client_id = auth_config.get("client_id", "")
3552
+ client_secret = auth_config.get("client_secret", "")
3553
+ redirect_uri = auth_config.get("redirect_uri", "")
1809
3554
 
1810
3555
  # Expand environment variables
1811
- if client_id.startswith('$'):
1812
- client_id = os.environ.get(client_id[1:], '')
1813
- if client_secret.startswith('$'):
1814
- client_secret = os.environ.get(client_secret[1:], '')
1815
- if redirect_uri.startswith('$'):
1816
- redirect_uri = os.environ.get(redirect_uri[1:], '')
3556
+ if client_id.startswith("$"):
3557
+ client_id = client_id[1:]
3558
+ if client_secret.startswith("$"):
3559
+ client_secret = client_secret[1:]
3560
+ if redirect_uri.startswith("$"):
3561
+ redirect_uri = redirect_uri[1:]
3562
+
3563
+ client_id = os.getenv(client_id, client_id)
3564
+ client_secret = os.getenv(client_secret, client_secret)
3565
+ redirect_uri = os.getenv(redirect_uri, redirect_uri)
1817
3566
 
1818
3567
  if not client_id or not client_secret:
1819
- return web.json_response({"error": "GitHub OAuth credentials not configured"}, status=500)
3568
+ return web.json_response(create_error_response("GitHub OAuth credentials not configured"), status=500)
1820
3569
 
1821
3570
  # Exchange code for access token
1822
3571
  async with aiohttp.ClientSession() as session:
1823
3572
  token_url = "https://github.com/login/oauth/access_token"
1824
3573
  token_data = {
1825
- 'client_id': client_id,
1826
- 'client_secret': client_secret,
1827
- 'code': code,
1828
- 'redirect_uri': redirect_uri
3574
+ "client_id": client_id,
3575
+ "client_secret": client_secret,
3576
+ "code": code,
3577
+ "redirect_uri": redirect_uri,
1829
3578
  }
1830
- headers = {'Accept': 'application/json'}
3579
+ headers = {"Accept": "application/json"}
1831
3580
 
1832
3581
  async with session.post(token_url, data=token_data, headers=headers) as resp:
1833
3582
  token_response = await resp.json()
1834
- access_token = token_response.get('access_token')
3583
+ access_token = token_response.get("access_token")
1835
3584
 
1836
3585
  if not access_token:
1837
- error = token_response.get('error_description', 'Failed to get access token')
1838
- return web.Response(text=f"OAuth error: {error}", status=400)
3586
+ error = token_response.get("error_description", "Failed to get access token")
3587
+ return web.json_response(create_error_response(f"OAuth error: {error}"), status=400)
1839
3588
 
1840
3589
  # Fetch user info
1841
3590
  user_url = "https://api.github.com/user"
1842
- headers = {
1843
- "Authorization": f"Bearer {access_token}",
1844
- "Accept": "application/json"
1845
- }
3591
+ headers = {"Authorization": f"Bearer {access_token}", "Accept": "application/json"}
1846
3592
 
1847
3593
  async with session.get(user_url, headers=headers) as resp:
1848
3594
  user_data = await resp.json()
1849
3595
 
1850
3596
  # Validate user
1851
- error_response = validate_user(user_data.get('login', ''))
3597
+ error_response = validate_user(user_data.get("login", ""))
1852
3598
  if error_response:
1853
3599
  return error_response
1854
3600
 
1855
3601
  # Create session
1856
3602
  session_token = secrets.token_urlsafe(32)
1857
3603
  g_sessions[session_token] = {
1858
- "userId": str(user_data.get('id', '')),
1859
- "userName": user_data.get('login', ''),
1860
- "displayName": user_data.get('name', ''),
1861
- "profileUrl": user_data.get('avatar_url', ''),
1862
- "email": user_data.get('email', ''),
1863
- "created": time.time()
3604
+ "userId": str(user_data.get("id", "")),
3605
+ "userName": user_data.get("login", ""),
3606
+ "displayName": user_data.get("name", ""),
3607
+ "profileUrl": user_data.get("avatar_url", ""),
3608
+ "email": user_data.get("email", ""),
3609
+ "created": time.time(),
1864
3610
  }
1865
3611
 
1866
3612
  # Redirect to UI with session token
1867
- return web.HTTPFound(f"/?session={session_token}")
3613
+ response = web.HTTPFound(f"/?session={session_token}")
3614
+ response.set_cookie("llms-token", session_token, httponly=True, path="/", max_age=86400)
3615
+ return response
1868
3616
 
1869
3617
  async def session_handler(request):
1870
3618
  """Validate and return session info"""
1871
- session_token = request.query.get('session') or request.headers.get('X-Session-Token')
3619
+ session_token = get_session_token(request)
1872
3620
 
1873
3621
  if not session_token or session_token not in g_sessions:
1874
- return web.json_response({"error": "Invalid or expired session"}, status=401)
3622
+ return web.json_response(create_error_response("Invalid or expired session"), status=401)
1875
3623
 
1876
3624
  session_data = g_sessions[session_token]
1877
3625
 
1878
3626
  # Clean up old sessions (older than 24 hours)
1879
3627
  current_time = time.time()
1880
- expired_sessions = [token for token, data in g_sessions.items() if current_time - data['created'] > 86400]
3628
+ expired_sessions = [token for token, data in g_sessions.items() if current_time - data["created"] > 86400]
1881
3629
  for token in expired_sessions:
1882
3630
  del g_sessions[token]
1883
3631
 
1884
- return web.json_response({
1885
- **session_data,
1886
- "sessionToken": session_token
1887
- })
3632
+ return web.json_response({**session_data, "sessionToken": session_token})
1888
3633
 
1889
3634
  async def logout_handler(request):
1890
3635
  """End OAuth session"""
1891
- session_token = request.query.get('session') or request.headers.get('X-Session-Token')
3636
+ session_token = get_session_token(request)
1892
3637
 
1893
3638
  if session_token and session_token in g_sessions:
1894
3639
  del g_sessions[session_token]
1895
3640
 
1896
- return web.json_response({"success": True})
3641
+ response = web.json_response({"success": True})
3642
+ response.del_cookie("llms-token")
3643
+ return response
1897
3644
 
1898
3645
  async def auth_handler(request):
1899
3646
  """Check authentication status and return user info"""
1900
3647
  # Check for OAuth session token
1901
- session_token = request.query.get('session') or request.headers.get('X-Session-Token')
3648
+ session_token = get_session_token(request)
1902
3649
 
1903
3650
  if session_token and session_token in g_sessions:
1904
3651
  session_data = g_sessions[session_token]
1905
- return web.json_response({
1906
- "userId": session_data.get("userId", ""),
1907
- "userName": session_data.get("userName", ""),
1908
- "displayName": session_data.get("displayName", ""),
1909
- "profileUrl": session_data.get("profileUrl", ""),
1910
- "authProvider": "github"
1911
- })
3652
+ return web.json_response(
3653
+ {
3654
+ "userId": session_data.get("userId", ""),
3655
+ "userName": session_data.get("userName", ""),
3656
+ "displayName": session_data.get("displayName", ""),
3657
+ "profileUrl": session_data.get("profileUrl", ""),
3658
+ "authProvider": "github",
3659
+ }
3660
+ )
1912
3661
 
1913
3662
  # Check for API key in Authorization header
1914
3663
  # auth_header = request.headers.get('Authorization', '')
@@ -1926,25 +3675,21 @@ def main():
1926
3675
  # })
1927
3676
 
1928
3677
  # Not authenticated - return error in expected format
1929
- return web.json_response({
1930
- "responseStatus": {
1931
- "errorCode": "Unauthorized",
1932
- "message": "Not authenticated"
1933
- }
1934
- }, status=401)
3678
+ return web.json_response(g_app.error_auth_required, status=401)
1935
3679
 
1936
- app.router.add_get('/auth', auth_handler)
1937
- app.router.add_get('/auth/github', github_auth_handler)
1938
- app.router.add_get('/auth/github/callback', github_callback_handler)
1939
- app.router.add_get('/auth/session', session_handler)
1940
- app.router.add_post('/auth/logout', logout_handler)
3680
+ app.router.add_get("/auth", auth_handler)
3681
+ app.router.add_get("/auth/github", github_auth_handler)
3682
+ app.router.add_get("/auth/github/callback", github_callback_handler)
3683
+ app.router.add_get("/auth/github/callback{tail:.*}", github_callback_handler)
3684
+ app.router.add_get("/auth/session", session_handler)
3685
+ app.router.add_post("/auth/logout", logout_handler)
1941
3686
 
1942
3687
  async def ui_static(request: web.Request) -> web.Response:
1943
3688
  path = Path(request.match_info["path"])
1944
3689
 
1945
3690
  try:
1946
3691
  # Handle both Path objects and importlib.resources Traversable objects
1947
- if hasattr(_ROOT, 'joinpath'):
3692
+ if hasattr(_ROOT, "joinpath"):
1948
3693
  # importlib.resources Traversable
1949
3694
  resource = _ROOT.joinpath("ui").joinpath(str(path))
1950
3695
  if not resource.is_file():
@@ -1957,82 +3702,154 @@ def main():
1957
3702
  raise web.HTTPNotFound
1958
3703
  try:
1959
3704
  resource.relative_to(Path(_ROOT)) # basic directory-traversal guard
1960
- except ValueError:
1961
- raise web.HTTPBadRequest(text="Invalid path")
3705
+ except ValueError as e:
3706
+ raise web.HTTPBadRequest(text="Invalid path") from e
1962
3707
  content = resource.read_bytes()
1963
3708
 
1964
3709
  content_type, _ = mimetypes.guess_type(str(path))
1965
3710
  if content_type is None:
1966
3711
  content_type = "application/octet-stream"
1967
3712
  return web.Response(body=content, content_type=content_type)
1968
- except (OSError, PermissionError, AttributeError):
1969
- raise web.HTTPNotFound
3713
+ except (OSError, PermissionError, AttributeError) as e:
3714
+ raise web.HTTPNotFound from e
1970
3715
 
1971
3716
  app.router.add_get("/ui/{path:.*}", ui_static, name="ui_static")
1972
-
1973
- async def ui_config_handler(request):
1974
- with open(g_ui_path, "r") as f:
1975
- ui = json.load(f)
1976
- if 'defaults' not in ui:
1977
- ui['defaults'] = g_config['defaults']
1978
- enabled, disabled = provider_status()
1979
- ui['status'] = {
1980
- "all": list(g_config['providers'].keys()),
1981
- "enabled": enabled,
1982
- "disabled": disabled
1983
- }
1984
- # Add auth configuration
1985
- ui['requiresAuth'] = auth_enabled
1986
- ui['authType'] = 'oauth' if auth_enabled else 'apikey'
1987
- return web.json_response(ui)
1988
- app.router.add_get('/config', ui_config_handler)
3717
+
3718
+ async def config_handler(request):
3719
+ ret = {}
3720
+ if "defaults" not in ret:
3721
+ ret["defaults"] = g_config["defaults"]
3722
+ enabled, disabled = provider_status()
3723
+ ret["status"] = {"all": list(g_config["providers"].keys()), "enabled": enabled, "disabled": disabled}
3724
+ # Add auth configuration
3725
+ ret["requiresAuth"] = auth_enabled
3726
+ ret["authType"] = "oauth" if auth_enabled else "apikey"
3727
+ return web.json_response(ret)
3728
+
3729
+ app.router.add_get("/config", config_handler)
1989
3730
 
1990
3731
  async def not_found_handler(request):
1991
3732
  return web.Response(text="404: Not Found", status=404)
1992
- app.router.add_get('/favicon.ico', not_found_handler)
3733
+
3734
+ app.router.add_get("/favicon.ico", not_found_handler)
3735
+
3736
+ # go through and register all g_app extensions
3737
+ for handler in g_app.server_add_get:
3738
+ handler_fn = handler[1]
3739
+
3740
+ async def managed_handler(request, handler_fn=handler_fn):
3741
+ try:
3742
+ return await handler_fn(request)
3743
+ except Exception as e:
3744
+ return web.json_response(to_error_response(e, stacktrace=g_verbose), status=500)
3745
+
3746
+ app.router.add_get(handler[0], managed_handler, **handler[2])
3747
+ for handler in g_app.server_add_post:
3748
+ handler_fn = handler[1]
3749
+
3750
+ async def managed_handler(request, handler_fn=handler_fn):
3751
+ try:
3752
+ return await handler_fn(request)
3753
+ except Exception as e:
3754
+ return web.json_response(to_error_response(e, stacktrace=g_verbose), status=500)
3755
+
3756
+ app.router.add_post(handler[0], managed_handler, **handler[2])
3757
+ for handler in g_app.server_add_put:
3758
+ handler_fn = handler[1]
3759
+
3760
+ async def managed_handler(request, handler_fn=handler_fn):
3761
+ try:
3762
+ return await handler_fn(request)
3763
+ except Exception as e:
3764
+ return web.json_response(to_error_response(e, stacktrace=g_verbose), status=500)
3765
+
3766
+ app.router.add_put(handler[0], managed_handler, **handler[2])
3767
+ for handler in g_app.server_add_delete:
3768
+ handler_fn = handler[1]
3769
+
3770
+ async def managed_handler(request, handler_fn=handler_fn):
3771
+ try:
3772
+ return await handler_fn(request)
3773
+ except Exception as e:
3774
+ return web.json_response(to_error_response(e, stacktrace=g_verbose), status=500)
3775
+
3776
+ app.router.add_delete(handler[0], managed_handler, **handler[2])
3777
+ for handler in g_app.server_add_patch:
3778
+ handler_fn = handler[1]
3779
+
3780
+ async def managed_handler(request, handler_fn=handler_fn):
3781
+ try:
3782
+ return await handler_fn(request)
3783
+ except Exception as e:
3784
+ return web.json_response(to_error_response(e, stacktrace=g_verbose), status=500)
3785
+
3786
+ app.router.add_patch(handler[0], managed_handler, **handler[2])
1993
3787
 
1994
3788
  # Serve index.html from root
1995
3789
  async def index_handler(request):
1996
3790
  index_content = read_resource_file_bytes("index.html")
1997
- if index_content is None:
1998
- raise web.HTTPNotFound
1999
- return web.Response(body=index_content, content_type='text/html')
2000
- app.router.add_get('/', index_handler)
3791
+
3792
+ importmaps = {"imports": g_app.import_maps}
3793
+ importmaps_script = '<script type="importmap">\n' + json.dumps(importmaps, indent=4) + "\n</script>"
3794
+ index_content = index_content.replace(
3795
+ b'<script type="importmap"></script>',
3796
+ importmaps_script.encode("utf-8"),
3797
+ )
3798
+
3799
+ if len(g_app.index_headers) > 0:
3800
+ html_header = ""
3801
+ for header in g_app.index_headers:
3802
+ html_header += header
3803
+ # replace </head> with html_header
3804
+ index_content = index_content.replace(b"</head>", html_header.encode("utf-8") + b"\n</head>")
3805
+
3806
+ if len(g_app.index_footers) > 0:
3807
+ html_footer = ""
3808
+ for footer in g_app.index_footers:
3809
+ html_footer += footer
3810
+ # replace </body> with html_footer
3811
+ index_content = index_content.replace(b"</body>", html_footer.encode("utf-8") + b"\n</body>")
3812
+
3813
+ return web.Response(body=index_content, content_type="text/html")
3814
+
3815
+ app.router.add_get("/", index_handler)
2001
3816
 
2002
3817
  # Serve index.html as fallback route (SPA routing)
2003
- app.router.add_route('*', '/{tail:.*}', index_handler)
3818
+ app.router.add_route("*", "/{tail:.*}", index_handler)
2004
3819
 
2005
3820
  # Setup file watcher for config files
2006
3821
  async def start_background_tasks(app):
2007
3822
  """Start background tasks when the app starts"""
2008
3823
  # Start watching config files in the background
2009
- asyncio.create_task(watch_config_files(g_config_path, g_ui_path))
3824
+ asyncio.create_task(watch_config_files(g_config_path, home_providers_path))
2010
3825
 
2011
3826
  app.on_startup.append(start_background_tasks)
2012
3827
 
3828
+ # go through and register all g_app extensions
3829
+
2013
3830
  print(f"Starting server on port {port}...")
2014
- web.run_app(app, host='0.0.0.0', port=port, print=_log)
2015
- exit(0)
3831
+ web.run_app(app, host="0.0.0.0", port=port, print=_log)
3832
+ g_app.exit(0)
2016
3833
 
2017
3834
  if cli_args.enable is not None:
2018
- if cli_args.enable.endswith(','):
3835
+ if cli_args.enable.endswith(","):
2019
3836
  cli_args.enable = cli_args.enable[:-1].strip()
2020
3837
  enable_providers = [cli_args.enable]
2021
- all_providers = g_config['providers'].keys()
3838
+ all_providers = g_config["providers"].keys()
2022
3839
  msgs = []
2023
3840
  if len(extra_args) > 0:
2024
3841
  for arg in extra_args:
2025
- if arg.endswith(','):
3842
+ if arg.endswith(","):
2026
3843
  arg = arg[:-1].strip()
2027
3844
  if arg in all_providers:
2028
3845
  enable_providers.append(arg)
2029
3846
 
2030
3847
  for provider in enable_providers:
2031
- if provider not in g_config['providers']:
2032
- print(f"Provider {provider} not found")
3848
+ if provider not in g_config["providers"]:
3849
+ print(f"Provider '{provider}' not found")
2033
3850
  print(f"Available providers: {', '.join(g_config['providers'].keys())}")
2034
3851
  exit(1)
2035
- if provider in g_config['providers']:
3852
+ if provider in g_config["providers"]:
2036
3853
  provider_config, msg = enable_provider(provider)
2037
3854
  print(f"\nEnabled provider {provider}:")
2038
3855
  printdump(provider_config)
@@ -2042,22 +3859,22 @@ def main():
2042
3859
  print_status()
2043
3860
  if len(msgs) > 0:
2044
3861
  print("\n" + "\n".join(msgs))
2045
- exit(0)
3862
+ g_app.exit(0)
2046
3863
 
2047
3864
  if cli_args.disable is not None:
2048
- if cli_args.disable.endswith(','):
3865
+ if cli_args.disable.endswith(","):
2049
3866
  cli_args.disable = cli_args.disable[:-1].strip()
2050
3867
  disable_providers = [cli_args.disable]
2051
- all_providers = g_config['providers'].keys()
3868
+ all_providers = g_config["providers"].keys()
2052
3869
  if len(extra_args) > 0:
2053
3870
  for arg in extra_args:
2054
- if arg.endswith(','):
3871
+ if arg.endswith(","):
2055
3872
  arg = arg[:-1].strip()
2056
3873
  if arg in all_providers:
2057
3874
  disable_providers.append(arg)
2058
3875
 
2059
3876
  for provider in disable_providers:
2060
- if provider not in g_config['providers']:
3877
+ if provider not in g_config["providers"]:
2061
3878
  print(f"Provider {provider} not found")
2062
3879
  print(f"Available providers: {', '.join(g_config['providers'].keys())}")
2063
3880
  exit(1)
@@ -2065,30 +3882,42 @@ def main():
2065
3882
  print(f"\nDisabled provider {provider}")
2066
3883
 
2067
3884
  print_status()
2068
- exit(0)
3885
+ g_app.exit(0)
2069
3886
 
2070
3887
  if cli_args.default is not None:
2071
3888
  default_model = cli_args.default
2072
- all_models = get_models()
2073
- if default_model not in all_models:
3889
+ provider_model = get_provider_model(default_model)
3890
+ if provider_model is None:
2074
3891
  print(f"Model {default_model} not found")
2075
- print(f"Available models: {', '.join(all_models)}")
2076
3892
  exit(1)
2077
- default_text = g_config['defaults']['text']
2078
- default_text['model'] = default_model
3893
+ default_text = g_config["defaults"]["text"]
3894
+ default_text["model"] = default_model
2079
3895
  save_config(g_config)
2080
3896
  print(f"\nDefault model set to: {default_model}")
2081
- exit(0)
2082
-
2083
- if cli_args.chat is not None or cli_args.image is not None or cli_args.audio is not None or cli_args.file is not None or len(extra_args) > 0:
3897
+ g_app.exit(0)
3898
+
3899
+ if (
3900
+ cli_args.chat is not None
3901
+ or cli_args.image is not None
3902
+ or cli_args.audio is not None
3903
+ or cli_args.file is not None
3904
+ or cli_args.out is not None
3905
+ or len(extra_args) > 0
3906
+ ):
2084
3907
  try:
2085
- chat = g_config['defaults']['text']
3908
+ chat = g_config["defaults"]["text"]
2086
3909
  if cli_args.image is not None:
2087
- chat = g_config['defaults']['image']
3910
+ chat = g_config["defaults"]["image"]
2088
3911
  elif cli_args.audio is not None:
2089
- chat = g_config['defaults']['audio']
3912
+ chat = g_config["defaults"]["audio"]
2090
3913
  elif cli_args.file is not None:
2091
- chat = g_config['defaults']['file']
3914
+ chat = g_config["defaults"]["file"]
3915
+ elif cli_args.out is not None:
3916
+ template = f"out:{cli_args.out}"
3917
+ if template not in g_config["defaults"]:
3918
+ print(f"Template for output modality '{cli_args.out}' not found")
3919
+ exit(1)
3920
+ chat = g_config["defaults"][template]
2092
3921
  if cli_args.chat is not None:
2093
3922
  chat_path = os.path.join(os.path.dirname(__file__), cli_args.chat)
2094
3923
  if not os.path.exists(chat_path):
@@ -2096,41 +3925,60 @@ def main():
2096
3925
  exit(1)
2097
3926
  _log(f"Using chat: {chat_path}")
2098
3927
 
2099
- with open (chat_path, "r") as f:
3928
+ with open(chat_path) as f:
2100
3929
  chat_json = f.read()
2101
3930
  chat = json.loads(chat_json)
2102
3931
 
2103
3932
  if cli_args.system is not None:
2104
- chat['messages'].insert(0, {'role': 'system', 'content': cli_args.system})
3933
+ chat["messages"].insert(0, {"role": "system", "content": cli_args.system})
2105
3934
 
2106
3935
  if len(extra_args) > 0:
2107
- prompt = ' '.join(extra_args)
3936
+ prompt = " ".join(extra_args)
3937
+ if not chat["messages"] or len(chat["messages"]) == 0:
3938
+ chat["messages"] = [{"role": "user", "content": [{"type": "text", "text": ""}]}]
3939
+
2108
3940
  # replace content of last message if exists, else add
2109
- last_msg = chat['messages'][-1] if 'messages' in chat else None
2110
- if last_msg and last_msg['role'] == 'user':
2111
- if isinstance(last_msg['content'], list):
2112
- last_msg['content'][-1]['text'] = prompt
3941
+ last_msg = chat["messages"][-1] if "messages" in chat else None
3942
+ if last_msg and last_msg["role"] == "user":
3943
+ if isinstance(last_msg["content"], list):
3944
+ last_msg["content"][-1]["text"] = prompt
2113
3945
  else:
2114
- last_msg['content'] = prompt
3946
+ last_msg["content"] = prompt
2115
3947
  else:
2116
- chat['messages'].append({'role': 'user', 'content': prompt})
3948
+ chat["messages"].append({"role": "user", "content": prompt})
2117
3949
 
2118
3950
  # Parse args parameters if provided
2119
3951
  args = None
2120
3952
  if cli_args.args is not None:
2121
3953
  args = parse_args_params(cli_args.args)
2122
3954
 
2123
- asyncio.run(cli_chat(chat, image=cli_args.image, audio=cli_args.audio, file=cli_args.file, args=args, raw=cli_args.raw))
2124
- exit(0)
3955
+ asyncio.run(
3956
+ cli_chat(
3957
+ chat,
3958
+ tools=cli_args.tools,
3959
+ image=cli_args.image,
3960
+ audio=cli_args.audio,
3961
+ file=cli_args.file,
3962
+ args=args,
3963
+ raw=cli_args.raw,
3964
+ )
3965
+ )
3966
+ g_app.exit(0)
2125
3967
  except Exception as e:
2126
3968
  print(f"{cli_args.logprefix}Error: {e}")
2127
3969
  if cli_args.verbose:
2128
3970
  traceback.print_exc()
2129
- exit(1)
3971
+ g_app.exit(1)
3972
+
3973
+ handled = run_extension_cli()
2130
3974
 
2131
- # show usage from ArgumentParser
2132
- parser.print_help()
3975
+ if not handled:
3976
+ # show usage from ArgumentParser
3977
+ parser.print_help()
3978
+ g_app.exit(0)
2133
3979
 
2134
3980
 
2135
- if __name__ == "__main__":
3981
+ if __name__ == "__main__":
3982
+ if MOCK or DEBUG:
3983
+ print(f"MOCK={MOCK} or DEBUG={DEBUG}")
2136
3984
  main()