sunholo 0.61.5__py3-none-any.whl → 0.61.7__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.
@@ -16,5 +16,6 @@ def create_app(name):
16
16
  # Initialize Flask app
17
17
  app = Flask(name)
18
18
  app.config['TRAP_HTTP_EXCEPTIONS'] = True
19
+ app.config['PROPAGATE_EXCEPTIONS'] = True
19
20
 
20
21
  return app
@@ -27,13 +27,12 @@ from ...utils.config import load_config
27
27
  try:
28
28
  from flask import request, jsonify, Response
29
29
  except ImportError:
30
- print("No flask installed for agents.flask.register_qna_routes, install via `pip install sunholo[http]`")
30
+ pass
31
31
 
32
32
  try:
33
33
  from langfuse.decorators import langfuse_context, observe
34
- except ImportError as err:
35
- print(f"No langfuse installed for agents.flask.register_qna_routes, install via `pip install sunholo[http]` - {str(err)}")
36
-
34
+ except ImportError:
35
+ pass
37
36
 
38
37
  def register_qna_routes(app, stream_interpreter, vac_interpreter):
39
38
 
@@ -25,6 +25,10 @@ def data_to_embed_pubsub(data: dict):
25
25
 
26
26
  message_data, metadata, vector_name = process_pubsub_message(data)
27
27
 
28
+ return process_chunker_data(message_data, metadata, vector_name)
29
+
30
+ def process_chunker_data(message_data, metadata, vector_name):
31
+
28
32
  if metadata:
29
33
  metadata["vector_name"] = vector_name
30
34
 
@@ -186,7 +186,7 @@ def handle_json_content_message(message_data: str, metadata: dict, vector_name:
186
186
 
187
187
  if the_content is None:
188
188
  log.info("No content found")
189
- return {"metadata": "No content found"}
189
+ return {"metadata": "No content found in 'page_content' JSON field"}
190
190
 
191
191
  docs = [Document(page_content=the_content, metadata=metadata)]
192
192
 
sunholo/cli/chat_vac.py CHANGED
@@ -20,14 +20,19 @@ from rich.text import Text
20
20
  from rich.table import Table
21
21
 
22
22
 
23
- def get_service_url(vac_name, project, region):
24
- agent_name = load_config_key("agent", vac_name, kind="vacConfig")
23
+ def get_service_url(vac_name, project, region, no_config=False):
24
+
25
+ if no_config:
26
+ agent_name = vac_name
27
+ else:
28
+ agent_name = load_config_key("agent", vac_name, kind="vacConfig")
29
+
25
30
  proxies = clean_proxy_list()
26
31
  if agent_name in proxies:
27
32
  port = proxies[agent_name]['port']
28
33
  url = f"http://127.0.0.1:{port}"
29
34
  else:
30
- print(f"No proxy found running for service: {agent_name} required for {vac_name} - attempting to connect")
35
+ console.print(f"No proxy found running for service: [bold orange]'{agent_name}[/bold orange] required for [bold orange]{vac_name}[/bold orange] - attempting to connect")
31
36
  url = start_proxy(agent_name, region, project)
32
37
 
33
38
  return url
@@ -144,33 +149,32 @@ def headless_mode(service_url, service_name, user_input, chat_history=None):
144
149
 
145
150
  return chat_history
146
151
 
147
- def resolve_service_url(args):
148
-
152
+ def resolve_service_url(args, no_config=False):
153
+ """
154
+ no_config: some VACs do not have an entry in the config file e.g. chunker, embedder etc.
155
+ """
149
156
  if args.url_override:
150
157
 
151
158
  return args.url_override
152
159
 
153
- if not args.no_proxy:
154
- try:
155
- service_url = get_service_url(args.vac_name, args.project, args.region)
156
- except ValueError as e:
157
- console.print(f"[bold red]ERROR: Could not start {args.vac_name} proxy URL: {str(e)}[/bold red]")
158
- sys.exit(1)
159
- else:
160
- console.print(f"Not using a proxy, connecting directly to {service_url}")
161
-
160
+ if args.no_proxy:
162
161
  agent_url = load_config_key("agent_url", args.vac_name, "vacConfig")
163
162
  if agent_url:
164
163
  console.print("Found agent_url within vacConfig: {agent_url}")
165
164
 
166
165
  service_url = agent_url or get_cloud_run_service_url(args.project, args.region, args.vac_name)
166
+ console.print(f"No proxy, connecting directly to {service_url}")
167
+ else:
168
+ try:
169
+ service_url = get_service_url(args.vac_name, args.project, args.region, no_config=no_config)
170
+ except ValueError as e:
171
+ console.print(f"[bold red]ERROR: Could not start {args.vac_name} proxy URL: {str(e)}[/bold red]")
172
+ sys.exit(1)
167
173
 
168
174
  return service_url
169
175
 
170
176
  def vac_command(args):
171
177
 
172
- service_url = resolve_service_url(args)
173
-
174
178
  if args.action == 'list':
175
179
 
176
180
  list_cloud_run_services(args.project, args.region)
@@ -178,13 +182,13 @@ def vac_command(args):
178
182
  return
179
183
 
180
184
  elif args.action == 'get-url':
181
-
185
+ service_url = resolve_service_url(args)
182
186
  console.print(service_url)
183
187
 
184
188
  return
185
189
 
186
190
  elif args.action == 'chat':
187
-
191
+ service_url = resolve_service_url(args)
188
192
  agent_name = load_config_key("agent", args.vac_name, kind="vacConfig")
189
193
 
190
194
  if args.headless:
@@ -209,6 +213,7 @@ def vac_command(args):
209
213
  stop_proxy(agent_name, stop_local=False)
210
214
 
211
215
  elif args.action == 'invoke':
216
+ service_url = resolve_service_url(args, no_config=True)
212
217
  try:
213
218
  json_data = json.loads(args.data)
214
219
  except json.JSONDecodeError as err:
@@ -222,7 +227,12 @@ def invoke_vac(service_url, data):
222
227
  headers = {"Content-Type": "application/json"}
223
228
  response = requests.post(service_url, headers=headers, data=json.dumps(data))
224
229
  response.raise_for_status()
225
- print(response.json())
230
+
231
+ the_data = response.json()
232
+ console.print(the_data)
233
+
234
+ return the_data
235
+
226
236
  except requests.exceptions.RequestException as e:
227
237
  console.print(f"[bold red]ERROR: Failed to invoke VAC: {e}[/bold red]")
228
238
 
sunholo/cli/cli.py CHANGED
@@ -7,12 +7,15 @@ from .cli_init import setup_init_subparser
7
7
  from .merge_texts import setup_merge_text_subparser
8
8
  from .run_proxy import setup_proxy_subparser
9
9
  from .chat_vac import setup_vac_subparser
10
+ from .embedder import setup_embedder_subparser
11
+
10
12
  from ..utils.config import load_config_key
11
13
 
12
14
  from ..logging import log
13
15
 
14
16
  from .sun_rich import console
15
17
  import sys
18
+ from rich.panel import Panel
16
19
 
17
20
 
18
21
  def load_default_gcp_config():
@@ -29,6 +32,12 @@ def load_default_gcp_config():
29
32
 
30
33
  def main(args=None):
31
34
 
35
+ console.print(
36
+ Panel("Welcome to Sunholo Command Line Interface, your assistant to deploy GenAI Virtual Agent Computers (VACs) to Multivac or your own Cloud.",
37
+ title="Sunholo GenAIOps Assistant CLI",
38
+ subtitle="Documentation at https://dev.sunholo.com/")
39
+ )
40
+ console.rule()
32
41
 
33
42
  """
34
43
  Entry point for the sunholo console script. This function parses command line arguments
@@ -64,6 +73,8 @@ def main(args=None):
64
73
  setup_proxy_subparser(subparsers)
65
74
  # vac command
66
75
  setup_vac_subparser(subparsers)
76
+ # embed command
77
+ setup_embedder_subparser(subparsers)
67
78
 
68
79
  args = parser.parse_args(args)
69
80
 
@@ -0,0 +1,148 @@
1
+ import json
2
+ import uuid
3
+ import base64
4
+ from datetime import datetime, timezone
5
+ from argparse import Namespace
6
+
7
+ from .sun_rich import console
8
+ from rich.progress import Progress
9
+
10
+ from .chat_vac import resolve_service_url, invoke_vac
11
+
12
+ def encode_data(vac, content, metadata=None, local_chunks=False):
13
+ # Current time in UTC
14
+ now_utc = datetime.now(timezone.utc)
15
+ formatted_time = now_utc.strftime("%Y-%m-%dT%H:%M:%SZ")
16
+
17
+ # Default metadata if none provided
18
+ default_metadata = {"vector_name": vac, "source": "sunholo-cli", "eventTime": formatted_time}
19
+
20
+ # Merge default metadata with provided metadata
21
+ if metadata:
22
+ if not isinstance(metadata, dict):
23
+ metadata = json.loads(metadata)
24
+ else:
25
+ metadata = {}
26
+
27
+ # Update metadata with default values if not present
28
+ metadata.update(default_metadata)
29
+
30
+ # Encode the content (URL)
31
+ if isinstance(content, str):
32
+ message_data = base64.b64encode(content.encode('utf-8')).decode('utf-8')
33
+ else:
34
+ raise ValueError(f"Unsupported content type: {type(content)}")
35
+
36
+ # Construct the message dictionary
37
+ messageId = str(uuid.uuid4())
38
+ message = {
39
+ "message": {
40
+ "data": message_data,
41
+ "messageId": messageId,
42
+ "publishTime": formatted_time,
43
+ "attributes": {
44
+ "namespace": vac,
45
+ "return_chunks": str(local_chunks).lower()
46
+ },
47
+ }
48
+ }
49
+
50
+ # Merge metadata with attributes
51
+ message["message"]["attributes"].update(metadata)
52
+
53
+ #console.print()
54
+ #console.print(f"Sending message: {messageId} with metadata:")
55
+ #console.print(f"{message['message']['attributes']}")
56
+
57
+ return message
58
+
59
+ def embed_command(args):
60
+ chunk_args = vars(args).copy()
61
+ embed_args = vars(args).copy()
62
+
63
+ console.rule("Sending data for chunking")
64
+
65
+ if args.chunk_override:
66
+ chunk_args["url_override"] = args.chunk_override
67
+ else:
68
+ chunk_args["vac_name"] = "chunker"
69
+ chunk_args["url_override"] = ""
70
+ chunk_args = Namespace(**chunk_args)
71
+ chunk_url = resolve_service_url(chunk_args, no_config=True)
72
+
73
+ json_data = encode_data(args.vac_name, args.data, args.metadata, args.local_chunks)
74
+
75
+ with console.status(f"[bold orange]Sending {args.data} to chunk via {chunk_url}[/bold orange]", spinner="star"):
76
+ chunk_res = invoke_vac(f"{chunk_url}/pubsub_to_store", json_data)
77
+
78
+ if not args.local_chunks:
79
+ console.rule(f"Chunks sent for processing in cloud: {chunk_res}")
80
+
81
+ return
82
+
83
+ console.rule("Processing chunks locally")
84
+
85
+ if args.embed_override:
86
+ embed_args["url_override"] = args.embed_override
87
+ else:
88
+ embed_args["vac_name"] = "embedder"
89
+ embed_args["url_override"] = ""
90
+ embed_args = Namespace(**embed_args)
91
+ embed_url = resolve_service_url(embed_args, no_config=True)
92
+
93
+ if not chunk_res:
94
+ console.print(f"[bold red]ERROR: Did not get any chunks from {chunk_url} for {json_data}")
95
+
96
+ return
97
+
98
+ chunks = chunk_res.get('chunks')
99
+ if not chunks:
100
+ console.print(f"[bold red]ERROR: No chunks found within json data: {str(chunk_res)} [/bold red]")
101
+
102
+ return
103
+
104
+ embeds = []
105
+ with Progress() as progress:
106
+ task = progress.add_task(f"Embedding [{len(chunks)}] chunks via {embed_url}", total=len(chunks))
107
+ for chunk in chunks:
108
+ progress.console.print(f"Working on chunk {chunk['metadata']}")
109
+
110
+ # do this async?
111
+ content = chunk.get("page_content")
112
+ now_utc = datetime.now(timezone.utc)
113
+ formatted_time = now_utc.strftime("%Y-%m-%dT%H:%M:%SZ")
114
+ chunk["metadata"]["eventTime"] = formatted_time
115
+ if not content:
116
+ progress.console.print("[bold red]No content chunk found, skipping.[/bold red]")
117
+ progress.advance(task)
118
+ continue
119
+ progress.console.print(f"Sending chunk length {len(content)} to embedder")
120
+ processed_chunk = encode_data(vac = args.vac_name,
121
+ content = json.dumps(chunk))
122
+
123
+ embed_res = invoke_vac(f"{embed_url}/embed_chunk", processed_chunk)
124
+ embeds.append(embed_res)
125
+ progress.advance(task)
126
+
127
+ console.rule("Embedding pipeline finished")
128
+
129
+ return embed_res
130
+
131
+
132
+ def setup_embedder_subparser(subparsers):
133
+ """
134
+ Sets up an argparse subparser for the 'embed' command.
135
+
136
+ Args:
137
+ subparsers: The subparsers object from argparse.ArgumentParser().
138
+ """
139
+ embed_parser = subparsers.add_parser('embed', help='Send data for embedding to a VAC vector store')
140
+ embed_parser.add_argument('--embed-override', help='Override the embed VAC service URL.')
141
+ embed_parser.add_argument('--chunk-override', help='Override the chunk VAC service URL.')
142
+ embed_parser.add_argument('--no-proxy', action='store_true', help='Do not use the proxy and connect directly to the VAC service.')
143
+ embed_parser.add_argument('-m', '--metadata', default=None, help='Metadata to send with the embedding (as JSON string).')
144
+ embed_parser.add_argument('--local-chunks', action='store_true', help='Whether to process chunks to embed locally, or via the cloud.')
145
+ embed_parser.add_argument('vac_name', help='VAC service to embed the data for')
146
+ embed_parser.add_argument('data', help='String content to send for embedding')
147
+
148
+ embed_parser.set_defaults(func=embed_command)
sunholo/cli/run_proxy.py CHANGED
@@ -1,6 +1,5 @@
1
1
  import subprocess
2
2
  import os
3
- import sys
4
3
  import signal
5
4
  import json
6
5
 
@@ -11,19 +10,6 @@ from rich import print
11
10
  PROXY_TRACKER_FILE = '.vac_proxy_tracker.json'
12
11
  DEFAULT_PORT = 8080
13
12
 
14
- def create_hyperlink(url, text):
15
- """
16
- Creates a hyperlink for the console.
17
-
18
- Args:
19
- url (str): The URL for the hyperlink.
20
- text (str): The text to display for the hyperlink.
21
-
22
- Returns:
23
- str: The formatted hyperlink.
24
- """
25
- return f"\033]8;;{url}\033\\{text}\033]8;;\033\\"
26
-
27
13
 
28
14
  def get_next_available_port(proxies, default_port):
29
15
  """
@@ -94,9 +80,6 @@ def save_proxies(proxies):
94
80
  with open(PROXY_TRACKER_FILE, 'w') as file:
95
81
  json.dump(proxies, file, indent=4)
96
82
 
97
-
98
-
99
-
100
83
  def start_proxy(service_name, region, project, port=None, local=False, app_type=None, app_folder=None, log_file=None):
101
84
  """
102
85
  Starts the gcloud proxy to the Cloud Run service and stores the PID.
@@ -116,8 +99,6 @@ def start_proxy(service_name, region, project, port=None, local=False, app_type=
116
99
  if not port:
117
100
  port = get_next_available_port(proxies, DEFAULT_PORT)
118
101
 
119
-
120
-
121
102
  if local:
122
103
  start_local(service_name, port, app_type, app_folder, log_file)
123
104
  else:
@@ -160,6 +141,9 @@ def stop_proxy(service_name, stop_local=True):
160
141
  """
161
142
  proxies = clean_proxy_list()
162
143
 
144
+ # ps aux | grep gcloud
145
+ # kill PID
146
+
163
147
  if service_name not in proxies:
164
148
  print(f"No proxy found for service: {service_name}")
165
149
  return
@@ -248,7 +232,7 @@ def start_local(service_name, port, app_type, app_folder, log_file):
248
232
  return
249
233
 
250
234
  if app_type == 'flask':
251
- command = [sys.executable, 'app.py']
235
+ command = ["gunicorn", "--bind", f"0.0.0.0:{port}", "--workers", "4", "app:app"]
252
236
  elif app_type == 'fastapi':
253
237
  command = ["uvicorn", "app:app", f"--port={port}"]
254
238
  else:
@@ -304,11 +288,11 @@ def setup_proxy_subparser(subparsers):
304
288
  args.app_folder,
305
289
  args.log_file))
306
290
 
307
- stop_parser = proxy_subparsers.add_parser('stop', help='Stop the proxy to the Cloud Run service.')
291
+ stop_parser = proxy_subparsers.add_parser('stop', help='Stop the proxy to the Cloud Run service. Backup: `kill -9 PID`')
308
292
  stop_parser.add_argument('service_name', help='Name of the Cloud Run service.')
309
293
  stop_parser.set_defaults(func=lambda args: stop_proxy(args.service_name))
310
294
 
311
- list_parser = proxy_subparsers.add_parser('list', help='List all running proxies.')
295
+ list_parser = proxy_subparsers.add_parser('list', help='List all running proxies. Examine port via `lsof-i :PORT`')
312
296
  list_parser.set_defaults(func=lambda args: list_proxies())
313
297
 
314
298
  stop_all_parser = proxy_subparsers.add_parser('stop-all', help='Stop all running proxies.')
@@ -8,7 +8,7 @@ try:
8
8
  from langchain_google_alloydb_pg import AlloyDBEngine, Column, AlloyDBLoader, AlloyDBDocumentSaver
9
9
  from google.cloud.alloydb.connector import IPTypes
10
10
  except ImportError:
11
- print("No database.alloydb functions - install via `pip install sunholo[database,gcp]`")
11
+ pass
12
12
 
13
13
  from .database import get_vector_size
14
14
  from ..logging import log
@@ -1,9 +1,9 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: sunholo
3
- Version: 0.61.5
3
+ Version: 0.61.7
4
4
  Summary: Large Language Model DevOps - a package to help deploy LLMs to the Cloud.
5
5
  Home-page: https://github.com/sunholo-data/sunholo-py
6
- Download-URL: https://github.com/sunholo-data/sunholo-py/archive/refs/tags/v0.61.5.tar.gz
6
+ Download-URL: https://github.com/sunholo-data/sunholo-py/archive/refs/tags/v0.61.7.tar.gz
7
7
  Author: Holosun ApS
8
8
  Author-email: multivac@sunholo.com
9
9
  License: Apache License, Version 2.0
@@ -11,8 +11,8 @@ sunholo/agents/fastapi/__init__.py,sha256=S_pj4_bTUmDGoq_exaREHlOKThi0zTuGT0VZY0
11
11
  sunholo/agents/fastapi/base.py,sha256=clk76cHbUAvU0OYJrRfCWX_5f0ACbhDsIzYBhI3wyoE,2514
12
12
  sunholo/agents/fastapi/qna_routes.py,sha256=DgK4Btu5XriOC1JaRQ4G_nWEjJfnQ0J5pyLanF6eF1g,3857
13
13
  sunholo/agents/flask/__init__.py,sha256=uqfHNw2Ru3EJ4dJEcbp86h_lkquBQPMxZbjhV_xe3rs,72
14
- sunholo/agents/flask/base.py,sha256=RUGWBYWeV60FatYF5sMRrxD-INU97Vodsi6JaB6i93s,763
15
- sunholo/agents/flask/qna_routes.py,sha256=9xQUIadDtQIgTRa-nXIYJbzwQokqtVjeLp-tcSxGy9Y,8788
14
+ sunholo/agents/flask/base.py,sha256=FgSaCODyoTtlstJtsqlLPScdgRUtv9_plxftdzHdVFo,809
15
+ sunholo/agents/flask/qna_routes.py,sha256=t5BabGKjlK3sMcaXZywja47gtQRoROXXz6NMm8H-SSc,8561
16
16
  sunholo/archive/__init__.py,sha256=qNHWm5rGPVOlxZBZCpA1wTYPbalizRT7f8X4rs2t290,31
17
17
  sunholo/archive/archive.py,sha256=C-UhG5x-XtZ8VheQp92IYJqgD0V3NFQjniqlit94t18,1197
18
18
  sunholo/auth/__init__.py,sha256=4owDjSaWYkbTlPK47UHTOC0gCWbZsqn4ZIEw5NWZTlg,28
@@ -22,22 +22,23 @@ sunholo/bots/discord.py,sha256=cCFae5K1BCa6JVkWGLh_iZ9qFO1JpXb6K4eJrlDfEro,2442
22
22
  sunholo/bots/github_webhook.py,sha256=5pQPRLM_wxxcILVaIzUDV8Kt7Arcm2dL1r1kMMHA524,9629
23
23
  sunholo/bots/webapp.py,sha256=EIMxdAJ_xtufwJmvnn7N_Fb_1hZ9DjhJ0Kf_hp02vEU,1926
24
24
  sunholo/chunker/__init__.py,sha256=UhQBZTKwDfBXm0TPv4LvsGc5pdUGCbYzi3qPTOkU4gw,55
25
- sunholo/chunker/data_to_embed_pubsub.py,sha256=t-pWNYv2mnwVAkMcIOK2CrIb3yr2aS9iAdtryk7hT8o,2931
25
+ sunholo/chunker/data_to_embed_pubsub.py,sha256=IY9SBRA7IO77QJBEgQuO1FiSCd6Dfm-TMEf1Ey-pLoo,3065
26
26
  sunholo/chunker/doc_handling.py,sha256=rIyknpzDyj5A0u_DqSQVD_CXLRNZPOU6TCL4bhCdjOI,8563
27
27
  sunholo/chunker/images.py,sha256=Xmh1vwHrVhoXm5iH2dhCc52O8YgdzE8KrDSdL-pGnp8,1861
28
28
  sunholo/chunker/loaders.py,sha256=xiToUVgPz2ZzcqpUAq7aNP3PTenb_rBUAFzu0JPycIg,10268
29
- sunholo/chunker/message_data.py,sha256=iDP94dySU3Xct-gWGnB4NNRSh2luQmgJeCfQb7ktt3U,6760
29
+ sunholo/chunker/message_data.py,sha256=X6aA4yX5aGN_mEvsDPWvdYRqqn5GO1BU9QhT9w5A0ec,6789
30
30
  sunholo/chunker/pdfs.py,sha256=daCZ1xjn1YvxlifIyxskWNpLJLe-Q9D_Jq12MWx3tZo,2473
31
31
  sunholo/chunker/publish.py,sha256=PoT8q3XJeFCg10WrLkYhuaaXIrGVkvUD3-R9IfoWoH4,2703
32
32
  sunholo/chunker/splitter.py,sha256=FLkDhkePkg_zGQpFBK13Cznw575D-Rf9pcaCpc1HUxY,6726
33
33
  sunholo/cli/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
34
- sunholo/cli/chat_vac.py,sha256=hroQtcf12nYaHaYObE4EeNlHmcB4BZl6LBs_yhSnwYI,12713
35
- sunholo/cli/cli.py,sha256=cogY1F5rcIGFYpZVFtbDNlAIElpfyPSCvSLC1ZIpHXg,2666
34
+ sunholo/cli/chat_vac.py,sha256=A_xavw034dE_Pr6dCRDp_OJrO3xs_F7S-W_QkK3r1Og,13165
35
+ sunholo/cli/cli.py,sha256=ya5gCK9z4c4tF3XhK9gga1TAC7DnDJES1voDV2TV5xQ,3128
36
36
  sunholo/cli/cli_init.py,sha256=JMZ9AX2cPDZ-_mv3adiv2ToFVNyRPtjk9Biszl1kiR0,2358
37
37
  sunholo/cli/configs.py,sha256=QUM9DvKOdZmEQRM5uI3Nh887T0YDiSMr7O240zTLqws,4546
38
38
  sunholo/cli/deploy.py,sha256=zxdwUsRTRMC8U5vyRv0JiKBLFn84Ug_Tc88-_h9hJSs,1609
39
+ sunholo/cli/embedder.py,sha256=hqIfqGCeV5UI_0dllNFsjdyjVWgC0Kmnw8kAKhN4jCI,5482
39
40
  sunholo/cli/merge_texts.py,sha256=U9vdMwKmcPoc6iPOWX5MKSxn49dNGbNzVLw8ui5PhEU,1823
40
- sunholo/cli/run_proxy.py,sha256=9ILCxSVHPzS-cSBvjdHhfZFlwsJ4Ttmu0vLtNoPCRgo,11469
41
+ sunholo/cli/run_proxy.py,sha256=0ft_9FUL9d1NKM2ABMMPpvXRTRZd9f-lR2FCahiGzlU,11281
41
42
  sunholo/cli/sun_rich.py,sha256=UpMqeJ0C8i0pkue1AHnnyyX0bFJ9zZeJ7HBR6yhuA8A,54
42
43
  sunholo/components/__init__.py,sha256=RJGNEihwvRIiDScKis04RHJv4yZGI1UpXlOmuCptNZI,208
43
44
  sunholo/components/llm.py,sha256=T4we3tGmqUj4tPwxQr9M6AXv_BALqZV_dRSvINan-oU,10374
@@ -45,7 +46,7 @@ sunholo/components/prompt.py,sha256=eZSghXkIlRzXiSrzgkG7e5ytUYq6R6LV-qjHU8jStig,
45
46
  sunholo/components/retriever.py,sha256=_Lyt9RIgb2PD-rhV6oKAadiUs3ukT5uAYGW197tEskw,3755
46
47
  sunholo/components/vectorstore.py,sha256=lB8vx_N6eBA44orNeVo1WRn0Q8GCIjvPPT9AfiPWBWE,5620
47
48
  sunholo/database/__init__.py,sha256=Zz0Shcq-CtStf9rJGIYB_Ybzb8rY_Q9mfSj-nviM490,241
48
- sunholo/database/alloydb.py,sha256=0zRLyeC9nACzj3v36ET9NqLeuzdwBJ2bE09CzgVTTFM,17098
49
+ sunholo/database/alloydb.py,sha256=zvT50Df7r-jJPo5lEEWbjlXzVr0KuqN6WINgRtiSyxo,17014
49
50
  sunholo/database/database.py,sha256=doY05kG8BZBLL-arh4hq5ef1ouWOtGHqdsDc6M2YHgk,7345
50
51
  sunholo/database/lancedb.py,sha256=2rAbJVusMrm5TPtVTsUtmwn0z1iZ_wvbKhc6eyT6ClE,708
51
52
  sunholo/database/static_dbs.py,sha256=aOyU3AJ-Dzz3qSNjbuN2293cfYw5PhkcQuQxdwPMJ4w,435
@@ -95,9 +96,9 @@ sunholo/utils/parsers.py,sha256=OrHmASqIbI45atVOhiGodgLvnfrzkvVzyHnSvAXD89I,3841
95
96
  sunholo/utils/user_ids.py,sha256=SQd5_H7FE7vcTZp9AQuQDWBXd4FEEd7TeVMQe1H4Ny8,292
96
97
  sunholo/vertex/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
97
98
  sunholo/vertex/init_vertex.py,sha256=JDMUaBRdednzbKF-5p33qqLit2LMsvgvWW-NRz0AqO0,1801
98
- sunholo-0.61.5.dist-info/LICENSE.txt,sha256=SdE3QjnD3GEmqqg9EX3TM9f7WmtOzqS1KJve8rhbYmU,11345
99
- sunholo-0.61.5.dist-info/METADATA,sha256=tJ2thVA-C3VH57sGLxsCI1dIVZ-H9gmxX-x8bzvaeQY,8057
100
- sunholo-0.61.5.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
101
- sunholo-0.61.5.dist-info/entry_points.txt,sha256=bZuN5AIHingMPt4Ro1b_T-FnQvZ3teBes-3OyO0asl4,49
102
- sunholo-0.61.5.dist-info/top_level.txt,sha256=wt5tadn5--5JrZsjJz2LceoUvcrIvxjHJe-RxuudxAk,8
103
- sunholo-0.61.5.dist-info/RECORD,,
99
+ sunholo-0.61.7.dist-info/LICENSE.txt,sha256=SdE3QjnD3GEmqqg9EX3TM9f7WmtOzqS1KJve8rhbYmU,11345
100
+ sunholo-0.61.7.dist-info/METADATA,sha256=49tEfz5t9jCW7XjVFyy-4Ip7_JbDUdTxxZviBqtBVaA,8057
101
+ sunholo-0.61.7.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
102
+ sunholo-0.61.7.dist-info/entry_points.txt,sha256=bZuN5AIHingMPt4Ro1b_T-FnQvZ3teBes-3OyO0asl4,49
103
+ sunholo-0.61.7.dist-info/top_level.txt,sha256=wt5tadn5--5JrZsjJz2LceoUvcrIvxjHJe-RxuudxAk,8
104
+ sunholo-0.61.7.dist-info/RECORD,,