mirror of
https://github.com/khoaliber/khoj.git
synced 2026-03-02 21:19:12 +00:00
Use placeholder for newline in json object values until json parsed and values extracted. This is useful when research mode models outputs multi-line codeblocks in queries etc.
226 lines
7.8 KiB
Python
226 lines
7.8 KiB
Python
import json
|
|
import logging
|
|
from datetime import datetime, timedelta
|
|
from typing import Dict, Optional
|
|
|
|
from langchain.schema import ChatMessage
|
|
|
|
from khoj.database.models import Agent, ChatModelOptions, KhojUser
|
|
from khoj.processor.conversation import prompts
|
|
from khoj.processor.conversation.openai.utils import (
|
|
chat_completion_with_backoff,
|
|
completion_with_backoff,
|
|
)
|
|
from khoj.processor.conversation.utils import (
|
|
clean_json,
|
|
construct_structured_message,
|
|
generate_chatml_messages_with_context,
|
|
)
|
|
from khoj.utils.helpers import ConversationCommand, is_none_or_empty
|
|
from khoj.utils.rawconfig import LocationData
|
|
from khoj.utils.yaml import yaml_dump
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def extract_questions(
|
|
text,
|
|
model: Optional[str] = "gpt-4o-mini",
|
|
conversation_log={},
|
|
api_key=None,
|
|
api_base_url=None,
|
|
location_data: LocationData = None,
|
|
user: KhojUser = None,
|
|
query_images: Optional[list[str]] = None,
|
|
vision_enabled: bool = False,
|
|
personality_context: Optional[str] = None,
|
|
tracer: dict = {},
|
|
):
|
|
"""
|
|
Infer search queries to retrieve relevant notes to answer user query
|
|
"""
|
|
location = f"{location_data}" if location_data else "Unknown"
|
|
username = prompts.user_name.format(name=user.get_full_name()) if user and user.get_full_name() else ""
|
|
|
|
# Extract Past User Message and Inferred Questions from Conversation Log
|
|
chat_history = "".join(
|
|
[
|
|
f'Q: {chat["intent"]["query"]}\nKhoj: {{"queries": {chat["intent"].get("inferred-queries") or list([chat["intent"]["query"]])}}}\nA: {chat["message"]}\n\n'
|
|
for chat in conversation_log.get("chat", [])[-4:]
|
|
if chat["by"] == "khoj" and "to-image" not in chat["intent"].get("type")
|
|
]
|
|
)
|
|
|
|
# Get dates relative to today for prompt creation
|
|
today = datetime.today()
|
|
current_new_year = today.replace(month=1, day=1)
|
|
last_new_year = current_new_year.replace(year=today.year - 1)
|
|
temperature = 0.7
|
|
|
|
prompt = prompts.extract_questions.format(
|
|
current_date=today.strftime("%Y-%m-%d"),
|
|
day_of_week=today.strftime("%A"),
|
|
current_month=today.strftime("%Y-%m"),
|
|
last_new_year=last_new_year.strftime("%Y"),
|
|
last_new_year_date=last_new_year.strftime("%Y-%m-%d"),
|
|
current_new_year_date=current_new_year.strftime("%Y-%m-%d"),
|
|
bob_tom_age_difference={current_new_year.year - 1984 - 30},
|
|
bob_age={current_new_year.year - 1984},
|
|
chat_history=chat_history,
|
|
text=text,
|
|
yesterday_date=(today - timedelta(days=1)).strftime("%Y-%m-%d"),
|
|
location=location,
|
|
username=username,
|
|
personality_context=personality_context,
|
|
)
|
|
|
|
prompt = construct_structured_message(
|
|
message=prompt,
|
|
images=query_images,
|
|
model_type=ChatModelOptions.ModelType.OPENAI,
|
|
vision_enabled=vision_enabled,
|
|
)
|
|
|
|
messages = [ChatMessage(content=prompt, role="user")]
|
|
|
|
response = send_message_to_model(
|
|
messages,
|
|
api_key,
|
|
model,
|
|
response_type="json_object",
|
|
api_base_url=api_base_url,
|
|
temperature=temperature,
|
|
tracer=tracer,
|
|
)
|
|
|
|
# Extract, Clean Message from GPT's Response
|
|
try:
|
|
response = clean_json(response)
|
|
response = json.loads(response)
|
|
response = [q.strip() for q in response["queries"] if q.strip()]
|
|
if not isinstance(response, list) or not response:
|
|
logger.error(f"Invalid response for constructing subqueries: {response}")
|
|
return [text]
|
|
return response
|
|
except:
|
|
logger.warning(f"GPT returned invalid JSON. Falling back to using user message as search query.\n{response}")
|
|
questions = [text]
|
|
|
|
logger.debug(f"Extracted Questions by GPT: {questions}")
|
|
return questions
|
|
|
|
|
|
def send_message_to_model(
|
|
messages, api_key, model, response_type="text", api_base_url=None, temperature=0, tracer: dict = {}
|
|
):
|
|
"""
|
|
Send message to model
|
|
"""
|
|
|
|
# Get Response from GPT
|
|
return completion_with_backoff(
|
|
messages=messages,
|
|
model=model,
|
|
openai_api_key=api_key,
|
|
temperature=temperature,
|
|
api_base_url=api_base_url,
|
|
model_kwargs={"response_format": {"type": response_type}},
|
|
tracer=tracer,
|
|
)
|
|
|
|
|
|
def converse(
|
|
references,
|
|
user_query,
|
|
online_results: Optional[Dict[str, Dict]] = None,
|
|
code_results: Optional[Dict[str, Dict]] = None,
|
|
conversation_log={},
|
|
model: str = "gpt-4o-mini",
|
|
api_key: Optional[str] = None,
|
|
api_base_url: Optional[str] = None,
|
|
temperature: float = 0.2,
|
|
completion_func=None,
|
|
conversation_commands=[ConversationCommand.Default],
|
|
max_prompt_size=None,
|
|
tokenizer_name=None,
|
|
location_data: LocationData = None,
|
|
user_name: str = None,
|
|
agent: Agent = None,
|
|
query_images: Optional[list[str]] = None,
|
|
vision_available: bool = False,
|
|
tracer: dict = {},
|
|
):
|
|
"""
|
|
Converse with user using OpenAI's ChatGPT
|
|
"""
|
|
# Initialize Variables
|
|
current_date = datetime.now()
|
|
|
|
if agent and agent.personality:
|
|
system_prompt = prompts.custom_personality.format(
|
|
name=agent.name,
|
|
bio=agent.personality,
|
|
current_date=current_date.strftime("%Y-%m-%d"),
|
|
day_of_week=current_date.strftime("%A"),
|
|
)
|
|
else:
|
|
system_prompt = prompts.personality.format(
|
|
current_date=current_date.strftime("%Y-%m-%d"),
|
|
day_of_week=current_date.strftime("%A"),
|
|
)
|
|
|
|
if location_data:
|
|
location_prompt = prompts.user_location.format(location=f"{location_data}")
|
|
system_prompt = f"{system_prompt}\n{location_prompt}"
|
|
|
|
if user_name:
|
|
user_name_prompt = prompts.user_name.format(name=user_name)
|
|
system_prompt = f"{system_prompt}\n{user_name_prompt}"
|
|
|
|
# Get Conversation Primer appropriate to Conversation Type
|
|
if conversation_commands == [ConversationCommand.Notes] and is_none_or_empty(references):
|
|
completion_func(chat_response=prompts.no_notes_found.format())
|
|
return iter([prompts.no_notes_found.format()])
|
|
elif conversation_commands == [ConversationCommand.Online] and is_none_or_empty(online_results):
|
|
completion_func(chat_response=prompts.no_online_results_found.format())
|
|
return iter([prompts.no_online_results_found.format()])
|
|
|
|
context_message = ""
|
|
if not is_none_or_empty(references):
|
|
context_message = f"{prompts.notes_conversation.format(references=yaml_dump(references))}\n\n"
|
|
if not is_none_or_empty(online_results):
|
|
context_message += f"{prompts.online_search_conversation.format(online_results=yaml_dump(online_results))}\n\n"
|
|
if not is_none_or_empty(code_results):
|
|
context_message += f"{prompts.code_executed_context.format(code_results=yaml_dump(code_results))}\n\n"
|
|
context_message = context_message.strip()
|
|
|
|
# Setup Prompt with Primer or Conversation History
|
|
messages = generate_chatml_messages_with_context(
|
|
user_query,
|
|
system_prompt,
|
|
conversation_log,
|
|
context_message=context_message,
|
|
model_name=model,
|
|
max_prompt_size=max_prompt_size,
|
|
tokenizer_name=tokenizer_name,
|
|
query_images=query_images,
|
|
vision_enabled=vision_available,
|
|
model_type=ChatModelOptions.ModelType.OPENAI,
|
|
)
|
|
truncated_messages = "\n".join({f"{message.content[:70]}..." for message in messages})
|
|
logger.debug(f"Conversation Context for GPT: {truncated_messages}")
|
|
|
|
# Get Response from GPT
|
|
return chat_completion_with_backoff(
|
|
messages=messages,
|
|
compiled_references=references,
|
|
online_results=online_results,
|
|
model_name=model,
|
|
temperature=temperature,
|
|
openai_api_key=api_key,
|
|
api_base_url=api_base_url,
|
|
completion_func=completion_func,
|
|
model_kwargs={"stop": ["Notes:\n["]},
|
|
tracer=tracer,
|
|
)
|