2023-04-08 19:55:33 +02:00
|
|
|
|
|
|
|
import sqlite3
|
2023-06-10 15:49:41 +02:00
|
|
|
from pathlib import Path
|
2023-06-15 14:19:25 +02:00
|
|
|
from datetime import datetime
|
2023-06-15 18:49:47 +02:00
|
|
|
from lollms.helpers import ASCIIColors
|
2023-08-03 01:07:29 +02:00
|
|
|
import json
|
2023-06-15 18:49:47 +02:00
|
|
|
|
2023-04-20 19:30:03 +02:00
|
|
|
__author__ = "parisneo"
|
2023-06-08 08:58:02 +02:00
|
|
|
__github__ = "https://github.com/ParisNeo/lollms-webui"
|
2023-04-20 19:30:03 +02:00
|
|
|
__copyright__ = "Copyright 2023, "
|
|
|
|
__license__ = "Apache 2.0"
|
|
|
|
|
|
|
|
|
2023-04-08 19:55:33 +02:00
|
|
|
# =================================== Database ==================================================================
|
2023-04-10 10:27:25 +02:00
|
|
|
class DiscussionsDB:
|
2023-07-06 20:56:21 +02:00
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
def __init__(self, db_path="database.db"):
|
2023-06-10 15:49:41 +02:00
|
|
|
self.db_path = Path(db_path)
|
|
|
|
self.db_path .parent.mkdir(exist_ok=True, parents= True)
|
2023-04-08 19:55:33 +02:00
|
|
|
|
2023-04-13 17:24:52 +02:00
|
|
|
|
2023-06-15 18:49:47 +02:00
|
|
|
def create_tables(self):
|
2024-01-02 02:16:48 +01:00
|
|
|
db_version = 10
|
2023-04-08 19:55:33 +02:00
|
|
|
with sqlite3.connect(self.db_path) as conn:
|
2023-04-10 10:27:25 +02:00
|
|
|
cursor = conn.cursor()
|
2023-04-10 16:14:39 +02:00
|
|
|
|
2023-06-15 18:49:47 +02:00
|
|
|
cursor.execute("""
|
|
|
|
CREATE TABLE IF NOT EXISTS schema_version (
|
|
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
|
|
version INTEGER NOT NULL
|
|
|
|
)
|
|
|
|
""")
|
|
|
|
|
|
|
|
cursor.execute("""
|
|
|
|
CREATE TABLE IF NOT EXISTS discussion (
|
|
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
|
|
title TEXT,
|
|
|
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
|
|
|
)
|
|
|
|
""")
|
|
|
|
|
|
|
|
cursor.execute("""
|
|
|
|
CREATE TABLE IF NOT EXISTS message (
|
|
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
|
|
binding TEXT,
|
|
|
|
model TEXT,
|
|
|
|
personality TEXT,
|
|
|
|
sender TEXT NOT NULL,
|
|
|
|
content TEXT NOT NULL,
|
2023-10-09 01:18:21 +02:00
|
|
|
message_type INT NOT NULL,
|
2023-08-03 01:07:29 +02:00
|
|
|
sender_type INT DEFAULT 0,
|
|
|
|
rank INT NOT NULL DEFAULT 0,
|
|
|
|
parent_message_id INT,
|
2023-06-17 19:47:49 +02:00
|
|
|
created_at TIMESTAMP,
|
|
|
|
finished_generating_at TIMESTAMP,
|
2023-06-15 18:49:47 +02:00
|
|
|
discussion_id INTEGER NOT NULL,
|
2023-08-03 01:07:29 +02:00
|
|
|
metadata TEXT,
|
2023-08-30 02:35:26 +02:00
|
|
|
ui TEXT,
|
2023-06-15 18:49:47 +02:00
|
|
|
FOREIGN KEY (discussion_id) REFERENCES discussion(id),
|
2023-08-03 01:07:29 +02:00
|
|
|
FOREIGN KEY (parent_message_id) REFERENCES message(id)
|
2023-06-15 18:49:47 +02:00
|
|
|
)
|
|
|
|
""")
|
|
|
|
|
|
|
|
cursor.execute("SELECT * FROM schema_version")
|
2023-04-10 16:14:39 +02:00
|
|
|
row = cursor.fetchone()
|
2023-06-15 18:49:47 +02:00
|
|
|
|
2023-04-10 16:14:39 +02:00
|
|
|
if row is None:
|
2023-06-15 18:49:47 +02:00
|
|
|
cursor.execute("INSERT INTO schema_version (version) VALUES (?)", (db_version,))
|
2023-04-10 16:14:39 +02:00
|
|
|
else:
|
2023-06-15 18:49:47 +02:00
|
|
|
cursor.execute("UPDATE schema_version SET version = ?", (db_version,))
|
|
|
|
|
|
|
|
conn.commit()
|
|
|
|
|
|
|
|
def add_missing_columns(self):
|
|
|
|
with sqlite3.connect(self.db_path) as conn:
|
|
|
|
cursor = conn.cursor()
|
|
|
|
|
|
|
|
table_columns = {
|
|
|
|
'discussion': [
|
|
|
|
'id',
|
|
|
|
'title',
|
|
|
|
'created_at'
|
|
|
|
],
|
|
|
|
'message': [
|
|
|
|
'id',
|
|
|
|
'binding',
|
|
|
|
'model',
|
|
|
|
'personality',
|
|
|
|
'sender',
|
|
|
|
'content',
|
2023-08-03 01:07:29 +02:00
|
|
|
'message_type',
|
|
|
|
'sender_type',
|
2023-06-15 18:49:47 +02:00
|
|
|
'rank',
|
2023-08-03 01:07:29 +02:00
|
|
|
'parent_message_id',
|
2023-06-15 18:49:47 +02:00
|
|
|
'created_at',
|
2023-07-03 00:17:01 +02:00
|
|
|
'metadata',
|
2023-08-30 02:35:26 +02:00
|
|
|
'ui',
|
2023-06-15 18:49:47 +02:00
|
|
|
'finished_generating_at',
|
|
|
|
'discussion_id'
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
for table, columns in table_columns.items():
|
|
|
|
cursor.execute(f"PRAGMA table_info({table})")
|
|
|
|
existing_columns = [column[1] for column in cursor.fetchall()]
|
|
|
|
|
|
|
|
for column in columns:
|
|
|
|
if column not in existing_columns:
|
|
|
|
if column == 'id':
|
|
|
|
cursor.execute(f"ALTER TABLE {table} ADD COLUMN {column} INTEGER PRIMARY KEY AUTOINCREMENT")
|
|
|
|
elif column.endswith('_at'):
|
|
|
|
cursor.execute(f"ALTER TABLE {table} ADD COLUMN {column} TIMESTAMP")
|
2023-07-03 00:17:01 +02:00
|
|
|
elif column=='metadata':
|
2023-08-03 01:07:29 +02:00
|
|
|
cursor.execute(f"ALTER TABLE {table} ADD COLUMN {column} TEXT")
|
|
|
|
elif column=='message_type':
|
|
|
|
cursor.execute(f"ALTER TABLE {table} RENAME COLUMN type TO {column}")
|
|
|
|
elif column=='sender_type':
|
|
|
|
cursor.execute(f"ALTER TABLE {table} ADD COLUMN {column} INT DEFAULT 0")
|
|
|
|
elif column=='parent_message_id':
|
|
|
|
cursor.execute(f"ALTER TABLE {table} RENAME COLUMN parent TO {column}")
|
2023-06-15 18:49:47 +02:00
|
|
|
else:
|
|
|
|
cursor.execute(f"ALTER TABLE {table} ADD COLUMN {column} TEXT")
|
2023-07-03 00:17:01 +02:00
|
|
|
ASCIIColors.yellow(f"Added column :{column}")
|
2023-04-08 19:55:33 +02:00
|
|
|
conn.commit()
|
|
|
|
|
2023-06-15 18:49:47 +02:00
|
|
|
|
2023-04-10 16:14:39 +02:00
|
|
|
def select(self, query, params=None, fetch_all=True):
|
2023-04-10 10:27:25 +02:00
|
|
|
"""
|
|
|
|
Execute the specified SQL select query on the database,
|
|
|
|
with optional parameters.
|
|
|
|
Returns the cursor object for further processing.
|
|
|
|
"""
|
|
|
|
with sqlite3.connect(self.db_path) as conn:
|
2023-04-10 16:14:39 +02:00
|
|
|
if params is None:
|
|
|
|
cursor = conn.execute(query)
|
|
|
|
else:
|
|
|
|
cursor = conn.execute(query, params)
|
2023-04-10 10:27:25 +02:00
|
|
|
if fetch_all:
|
|
|
|
return cursor.fetchall()
|
|
|
|
else:
|
|
|
|
return cursor.fetchone()
|
|
|
|
|
|
|
|
|
2023-04-13 12:31:48 +02:00
|
|
|
def delete(self, query, params=None):
|
2023-04-10 10:27:25 +02:00
|
|
|
"""
|
|
|
|
Execute the specified SQL delete query on the database,
|
|
|
|
with optional parameters.
|
|
|
|
Returns the cursor object for further processing.
|
|
|
|
"""
|
|
|
|
with sqlite3.connect(self.db_path) as conn:
|
2023-04-08 19:55:33 +02:00
|
|
|
cursor = conn.cursor()
|
2023-04-13 12:31:48 +02:00
|
|
|
if params is None:
|
|
|
|
cursor.execute(query)
|
|
|
|
else:
|
|
|
|
cursor.execute(query, params)
|
2023-04-08 19:55:33 +02:00
|
|
|
conn.commit()
|
2023-04-10 10:27:25 +02:00
|
|
|
|
|
|
|
def insert(self, query, params=None):
|
|
|
|
"""
|
|
|
|
Execute the specified INSERT SQL query on the database,
|
|
|
|
with optional parameters.
|
|
|
|
Returns the ID of the newly inserted row.
|
|
|
|
"""
|
|
|
|
|
2023-04-08 19:55:33 +02:00
|
|
|
with sqlite3.connect(self.db_path) as conn:
|
2023-04-10 11:02:00 +02:00
|
|
|
cursor = conn.execute(query, params)
|
2023-04-10 10:27:25 +02:00
|
|
|
rowid = cursor.lastrowid
|
2023-04-08 19:55:33 +02:00
|
|
|
conn.commit()
|
2023-04-10 10:27:25 +02:00
|
|
|
self.conn = None
|
|
|
|
return rowid
|
2023-04-08 19:55:33 +02:00
|
|
|
|
2023-08-30 02:35:26 +02:00
|
|
|
def update(self, query, params:tuple=None):
|
2023-04-10 11:02:00 +02:00
|
|
|
"""
|
|
|
|
Execute the specified Update SQL query on the database,
|
|
|
|
with optional parameters.
|
|
|
|
Returns the ID of the newly inserted row.
|
|
|
|
"""
|
|
|
|
|
|
|
|
with sqlite3.connect(self.db_path) as conn:
|
|
|
|
conn.execute(query, params)
|
|
|
|
conn.commit()
|
2023-04-10 17:36:26 +02:00
|
|
|
|
|
|
|
def load_last_discussion(self):
|
2023-04-10 21:17:09 +02:00
|
|
|
last_discussion_id = self.select("SELECT id FROM discussion ORDER BY id DESC LIMIT 1", fetch_all=False)
|
2023-04-10 20:58:53 +02:00
|
|
|
if last_discussion_id is None:
|
2023-04-13 20:06:23 -05:00
|
|
|
last_discussion = self.create_discussion()
|
|
|
|
last_discussion_id = last_discussion.discussion_id
|
2023-04-10 20:58:53 +02:00
|
|
|
else:
|
2023-04-13 20:06:23 -05:00
|
|
|
last_discussion_id = last_discussion_id[0]
|
2023-04-13 19:02:20 +02:00
|
|
|
self.current_message_id = self.select("SELECT id FROM message WHERE discussion_id=? ORDER BY id DESC LIMIT 1", (last_discussion_id,), fetch_all=False)
|
2023-04-10 17:36:26 +02:00
|
|
|
return Discussion(last_discussion_id, self)
|
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
def create_discussion(self, title="untitled"):
|
|
|
|
"""Creates a new discussion
|
2023-04-08 19:55:33 +02:00
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
Args:
|
|
|
|
title (str, optional): The title of the discussion. Defaults to "untitled".
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
Discussion: A Discussion instance
|
|
|
|
"""
|
2023-04-10 11:02:00 +02:00
|
|
|
discussion_id = self.insert(f"INSERT INTO discussion (title) VALUES (?)",(title,))
|
2023-04-10 10:27:25 +02:00
|
|
|
return Discussion(discussion_id, self)
|
|
|
|
|
|
|
|
def build_discussion(self, discussion_id=0):
|
|
|
|
return Discussion(discussion_id, self)
|
2023-04-08 19:55:33 +02:00
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
def get_discussions(self):
|
2023-04-10 16:14:39 +02:00
|
|
|
rows = self.select("SELECT * FROM discussion")
|
2023-04-10 10:27:25 +02:00
|
|
|
return [{"id": row[0], "title": row[1]} for row in rows]
|
2023-04-08 19:55:33 +02:00
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
def does_last_discussion_have_messages(self):
|
2023-04-10 21:17:09 +02:00
|
|
|
last_discussion_id = self.select("SELECT id FROM discussion ORDER BY id DESC LIMIT 1", fetch_all=False)
|
|
|
|
if last_discussion_id is None:
|
2023-04-13 20:06:23 -05:00
|
|
|
last_discussion = self.create_discussion()
|
|
|
|
last_discussion_id = last_discussion.discussion_id
|
2023-04-10 21:17:09 +02:00
|
|
|
else:
|
2023-04-13 20:06:23 -05:00
|
|
|
last_discussion_id = last_discussion_id[0]
|
2023-04-10 17:36:26 +02:00
|
|
|
last_message = self.select("SELECT * FROM message WHERE discussion_id=?", (last_discussion_id,), fetch_all=False)
|
2023-04-10 10:27:25 +02:00
|
|
|
return last_message is not None
|
|
|
|
|
|
|
|
def remove_discussions(self):
|
|
|
|
self.delete("DELETE FROM message")
|
|
|
|
self.delete("DELETE FROM discussion")
|
2023-04-08 19:55:33 +02:00
|
|
|
|
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
def export_to_json(self):
|
2023-10-25 23:20:16 +02:00
|
|
|
"""
|
|
|
|
Export all discussions and their messages from the database to a JSON format.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
list: A list of dictionaries representing discussions and their messages.
|
|
|
|
Each dictionary contains the discussion ID, title, and a list of messages.
|
|
|
|
Each message dictionary contains the sender, content, message type, rank,
|
|
|
|
parent message ID, binding, model, personality, created at, and finished
|
|
|
|
generating at fields.
|
|
|
|
"""
|
2023-04-10 11:02:00 +02:00
|
|
|
db_discussions = self.select("SELECT * FROM discussion")
|
2023-04-08 19:55:33 +02:00
|
|
|
discussions = []
|
2023-04-10 11:02:00 +02:00
|
|
|
for row in db_discussions:
|
2023-04-08 19:55:33 +02:00
|
|
|
discussion_id = row[0]
|
2023-04-21 11:48:35 +02:00
|
|
|
discussion_title = row[1]
|
|
|
|
discussion = {"id": discussion_id, "title":discussion_title, "messages": []}
|
2023-10-09 01:18:21 +02:00
|
|
|
rows = self.select(f"SELECT sender, content, message_type, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at FROM message WHERE discussion_id=?",(discussion_id,))
|
2023-04-10 11:02:00 +02:00
|
|
|
for message_row in rows:
|
2023-11-09 02:00:49 +01:00
|
|
|
sender = message_row[0]
|
|
|
|
content = message_row[1]
|
|
|
|
content_type = message_row[2]
|
|
|
|
rank = message_row[3]
|
|
|
|
parent_message_id = message_row[4]
|
|
|
|
binding = message_row[5]
|
|
|
|
model = message_row[6]
|
|
|
|
personality = message_row[7]
|
|
|
|
created_at = message_row[8]
|
|
|
|
finished_generating_at = message_row[9]
|
2023-06-15 14:19:25 +02:00
|
|
|
|
2023-04-08 19:55:33 +02:00
|
|
|
discussion["messages"].append(
|
2023-10-09 01:18:21 +02:00
|
|
|
{"sender": sender, "content": content, "message_type": content_type, "rank": rank, "parent_message_id": parent_message_id, "binding": binding, "model":model, "personality":personality, "created_at":created_at, "finished_generating_at":finished_generating_at}
|
2023-04-08 19:55:33 +02:00
|
|
|
)
|
|
|
|
discussions.append(discussion)
|
|
|
|
return discussions
|
2023-10-10 01:47:49 +02:00
|
|
|
|
|
|
|
def export_all_as_markdown_list_for_vectorization(self):
|
2023-10-25 23:20:16 +02:00
|
|
|
"""
|
|
|
|
Export all discussions and their messages from the database to a Markdown list format.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
list: A list of lists representing discussions and their messages in a Markdown format.
|
|
|
|
Each inner list contains the discussion title and a string representing all
|
|
|
|
messages in the discussion in a Markdown format.
|
|
|
|
"""
|
2023-10-10 01:47:49 +02:00
|
|
|
data = self.export_all_discussions_to_json()
|
|
|
|
# Initialize an empty result string
|
|
|
|
discussions = []
|
|
|
|
# Iterate through discussions in the JSON data
|
|
|
|
for discussion in data:
|
|
|
|
# Extract the title
|
|
|
|
title = discussion['title']
|
|
|
|
messages = ""
|
|
|
|
# Iterate through messages in the discussion
|
|
|
|
for message in discussion['messages']:
|
|
|
|
sender = message['sender']
|
|
|
|
content = message['content']
|
|
|
|
# Append the sender and content in a Markdown format
|
|
|
|
messages += f'{sender}: {content}\n'
|
|
|
|
discussions.append([title, messages])
|
|
|
|
return discussions
|
|
|
|
|
2023-10-07 23:16:27 +02:00
|
|
|
def export_all_as_markdown(self):
|
2023-10-25 23:20:16 +02:00
|
|
|
"""
|
|
|
|
Export all discussions and their messages from the database to a Markdown format.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
str: A string representing all discussions and their messages in a Markdown format.
|
|
|
|
Each discussion is represented as a Markdown heading, and each message is
|
|
|
|
represented with the sender and content in a Markdown format.
|
|
|
|
"""
|
2023-10-07 23:16:27 +02:00
|
|
|
data = self.export_all_discussions_to_json()
|
|
|
|
|
|
|
|
# Initialize an empty result string
|
|
|
|
result = ''
|
|
|
|
|
|
|
|
# Iterate through discussions in the JSON data
|
|
|
|
for discussion in data:
|
|
|
|
# Extract the title
|
|
|
|
title = discussion['title']
|
|
|
|
# Append the title with '#' as Markdown heading
|
|
|
|
result += f'#{title}\n'
|
|
|
|
|
|
|
|
# Iterate through messages in the discussion
|
|
|
|
for message in discussion['messages']:
|
|
|
|
sender = message['sender']
|
|
|
|
content = message['content']
|
|
|
|
# Append the sender and content in a Markdown format
|
|
|
|
result += f'{sender}: {content}\n'
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
def export_all_discussions_to_json(self):
|
|
|
|
# Convert the list of discussion IDs to a tuple
|
|
|
|
db_discussions = self.select(
|
|
|
|
f"SELECT * FROM discussion"
|
|
|
|
)
|
|
|
|
discussions = []
|
|
|
|
for row in db_discussions:
|
|
|
|
discussion_id = row[0]
|
|
|
|
discussion_title = row[1]
|
|
|
|
discussion = {"id": discussion_id, "title":discussion_title, "messages": []}
|
|
|
|
rows = self.select(f"SELECT sender, content, message_type, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at FROM message WHERE discussion_id=?",(discussion_id,))
|
|
|
|
for message_row in rows:
|
|
|
|
sender = message_row[0]
|
|
|
|
content = message_row[1]
|
|
|
|
content_type = message_row[2]
|
|
|
|
rank = message_row[3]
|
|
|
|
parent_message_id = message_row[4]
|
|
|
|
binding = message_row[5]
|
|
|
|
model = message_row[6]
|
|
|
|
personality = message_row[7]
|
|
|
|
created_at = message_row[8]
|
|
|
|
finished_generating_at = message_row[9]
|
|
|
|
|
|
|
|
discussion["messages"].append(
|
2023-10-09 01:18:21 +02:00
|
|
|
{"sender": sender, "content": content, "message_type": content_type, "rank": rank, "parent_message_id": parent_message_id, "binding": binding, "model":model, "personality":personality, "created_at":created_at, "finished_generating_at": finished_generating_at}
|
2023-10-07 23:16:27 +02:00
|
|
|
)
|
|
|
|
discussions.append(discussion)
|
|
|
|
return discussions
|
2023-04-08 19:55:33 +02:00
|
|
|
|
2023-05-24 16:08:42 +02:00
|
|
|
def export_discussions_to_json(self, discussions_ids:list):
|
|
|
|
# Convert the list of discussion IDs to a tuple
|
|
|
|
discussions_ids_tuple = tuple(discussions_ids)
|
2023-05-24 17:28:22 +02:00
|
|
|
txt = ','.join(['?'] * len(discussions_ids_tuple))
|
|
|
|
db_discussions = self.select(
|
|
|
|
f"SELECT * FROM discussion WHERE id IN ({txt})",
|
|
|
|
discussions_ids_tuple
|
|
|
|
)
|
2023-05-24 16:08:42 +02:00
|
|
|
discussions = []
|
|
|
|
for row in db_discussions:
|
|
|
|
discussion_id = row[0]
|
|
|
|
discussion_title = row[1]
|
|
|
|
discussion = {"id": discussion_id, "title":discussion_title, "messages": []}
|
2023-10-04 15:56:41 +02:00
|
|
|
rows = self.select(f"SELECT sender, content, message_type, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at FROM message WHERE discussion_id=?",(discussion_id,))
|
2023-05-24 16:08:42 +02:00
|
|
|
for message_row in rows:
|
2023-07-05 02:19:11 +02:00
|
|
|
sender = message_row[0]
|
|
|
|
content = message_row[1]
|
|
|
|
content_type = message_row[2]
|
|
|
|
rank = message_row[3]
|
2023-08-03 01:07:29 +02:00
|
|
|
parent_message_id = message_row[4]
|
2023-07-05 02:19:11 +02:00
|
|
|
binding = message_row[5]
|
|
|
|
model = message_row[6]
|
|
|
|
personality = message_row[7]
|
|
|
|
created_at = message_row[8]
|
|
|
|
finished_generating_at = message_row[9]
|
2023-06-15 14:19:25 +02:00
|
|
|
|
2023-05-24 16:08:42 +02:00
|
|
|
discussion["messages"].append(
|
2023-10-09 01:18:21 +02:00
|
|
|
{"sender": sender, "content": content, "message_type": content_type, "rank": rank, "parent_message_id": parent_message_id, "binding": binding, "model":model, "personality":personality, "created_at":created_at, "finished_generating_at": finished_generating_at}
|
2023-05-24 16:08:42 +02:00
|
|
|
)
|
|
|
|
discussions.append(discussion)
|
|
|
|
return discussions
|
2023-05-29 20:24:25 +02:00
|
|
|
|
|
|
|
def import_from_json(self, json_data):
|
|
|
|
discussions = []
|
|
|
|
data = json_data
|
|
|
|
for discussion_data in data:
|
|
|
|
discussion_id = discussion_data.get("id")
|
|
|
|
discussion_title = discussion_data.get("title")
|
|
|
|
messages_data = discussion_data.get("messages", [])
|
|
|
|
discussion = {"id": discussion_id, "title": discussion_title, "messages": []}
|
|
|
|
|
|
|
|
# Insert discussion into the database
|
2023-05-30 14:11:30 +02:00
|
|
|
discussion_id = self.insert("INSERT INTO discussion (title) VALUES (?)", (discussion_title,))
|
2023-05-29 20:24:25 +02:00
|
|
|
|
|
|
|
for message_data in messages_data:
|
|
|
|
sender = message_data.get("sender")
|
|
|
|
content = message_data.get("content")
|
2023-10-09 01:18:21 +02:00
|
|
|
content_type = message_data.get("message_type",message_data.get("type"))
|
2023-05-29 20:24:25 +02:00
|
|
|
rank = message_data.get("rank")
|
2023-08-03 01:07:29 +02:00
|
|
|
parent_message_id = message_data.get("parent_message_id")
|
2023-06-15 14:19:25 +02:00
|
|
|
binding = message_data.get("binding","")
|
|
|
|
model = message_data.get("model","")
|
|
|
|
personality = message_data.get("personality","")
|
|
|
|
created_at = message_data.get("created_at",datetime.now().strftime('%Y-%m-%d %H:%M:%S'))
|
2023-06-15 16:56:08 +02:00
|
|
|
finished_generating_at = message_data.get("finished_generating_at",datetime.now().strftime('%Y-%m-%d %H:%M:%S'))
|
2023-05-29 20:24:25 +02:00
|
|
|
discussion["messages"].append(
|
2023-10-09 01:18:21 +02:00
|
|
|
{"sender": sender, "content": content, "message_type": content_type, "rank": rank, "binding": binding, "model": model, "personality": personality, "created_at": created_at, "finished_generating_at": finished_generating_at}
|
2023-05-29 20:24:25 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
# Insert message into the database
|
2023-10-09 01:18:21 +02:00
|
|
|
self.insert("INSERT INTO message (sender, content, message_type, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at, discussion_id) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
2023-08-03 01:07:29 +02:00
|
|
|
(sender, content, content_type, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at, discussion_id))
|
2023-05-29 20:24:25 +02:00
|
|
|
|
|
|
|
discussions.append(discussion)
|
|
|
|
|
|
|
|
return discussions
|
2023-05-24 16:08:42 +02:00
|
|
|
|
2023-11-22 20:49:54 +01:00
|
|
|
def export_discussions_to_markdown(self, discussions_ids:list, title = ""):
|
|
|
|
# Convert the list of discussion IDs to a tuple
|
|
|
|
discussions_ids_tuple = tuple(discussions_ids)
|
|
|
|
txt = ','.join(['?'] * len(discussions_ids_tuple))
|
|
|
|
db_discussions = self.select(
|
|
|
|
f"SELECT * FROM discussion WHERE id IN ({txt})",
|
|
|
|
discussions_ids_tuple
|
|
|
|
)
|
|
|
|
discussions = f"# {title}" if title!="" else ""
|
|
|
|
for row in db_discussions:
|
|
|
|
discussion_id = row[0]
|
|
|
|
discussion_title = row[1]
|
|
|
|
discussions += f"## {discussion_title}\n"
|
|
|
|
rows = self.select(f"SELECT sender, content, message_type, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at FROM message WHERE discussion_id=?",(discussion_id,))
|
|
|
|
for message_row in rows:
|
|
|
|
sender = message_row[0]
|
|
|
|
content = message_row[1]
|
|
|
|
content_type = message_row[2]
|
|
|
|
rank = message_row[3]
|
|
|
|
parent_message_id = message_row[4]
|
|
|
|
binding = message_row[5]
|
|
|
|
model = message_row[6]
|
|
|
|
personality = message_row[7]
|
|
|
|
created_at = message_row[8]
|
|
|
|
finished_generating_at = message_row[9]
|
|
|
|
|
|
|
|
discussions +=f"### {sender}:\n{content}\n"
|
|
|
|
discussions +=f"\n"
|
|
|
|
return discussions
|
|
|
|
|
2023-08-03 01:07:29 +02:00
|
|
|
|
|
|
|
class Message:
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
discussion_id,
|
|
|
|
discussions_db,
|
|
|
|
message_type,
|
|
|
|
sender_type,
|
|
|
|
sender,
|
|
|
|
content,
|
|
|
|
metadata = None,
|
2023-08-30 02:35:26 +02:00
|
|
|
ui = None,
|
2023-08-03 01:07:29 +02:00
|
|
|
rank = 0,
|
|
|
|
parent_message_id = 0,
|
|
|
|
binding = "",
|
|
|
|
model = "",
|
|
|
|
personality = "",
|
|
|
|
created_at = None,
|
|
|
|
finished_generating_at = None,
|
|
|
|
id = None,
|
|
|
|
insert_into_db = False
|
|
|
|
):
|
|
|
|
|
|
|
|
self.discussion_id = discussion_id
|
|
|
|
self.discussions_db = discussions_db
|
|
|
|
self.self = self
|
|
|
|
self.sender = sender
|
|
|
|
self.sender_type = sender_type
|
|
|
|
self.content = content
|
|
|
|
self.message_type = message_type
|
|
|
|
self.rank = rank
|
|
|
|
self.parent_message_id = parent_message_id
|
|
|
|
self.binding = binding
|
|
|
|
self.model = model
|
|
|
|
self.metadata = json.dumps(metadata, indent=4) if metadata is not None and type(metadata)== dict else metadata
|
2023-08-30 02:35:26 +02:00
|
|
|
self.ui = ui
|
2023-08-03 01:07:29 +02:00
|
|
|
self.personality = personality
|
|
|
|
self.created_at = created_at
|
|
|
|
self.finished_generating_at = finished_generating_at
|
|
|
|
|
|
|
|
if insert_into_db:
|
|
|
|
self.id = self.discussions_db.insert(
|
2023-09-04 02:05:46 +02:00
|
|
|
"INSERT INTO message (sender, message_type, sender_type, sender, content, metadata, ui, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at, discussion_id) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
|
|
|
(sender, message_type, sender_type, sender, content, metadata, ui, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at, discussion_id)
|
2023-08-03 01:07:29 +02:00
|
|
|
)
|
|
|
|
else:
|
|
|
|
self.id = id
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_fields():
|
|
|
|
return [
|
|
|
|
"id",
|
|
|
|
"message_type",
|
|
|
|
"sender_type",
|
|
|
|
"sender",
|
|
|
|
"content",
|
|
|
|
"metadata",
|
2023-08-30 02:35:26 +02:00
|
|
|
"ui",
|
2023-08-03 01:07:29 +02:00
|
|
|
"rank",
|
|
|
|
"parent_message_id",
|
|
|
|
"binding",
|
|
|
|
"model",
|
|
|
|
"personality",
|
|
|
|
"created_at",
|
|
|
|
"finished_generating_at",
|
|
|
|
"discussion_id"
|
|
|
|
]
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def from_db(discussions_db, message_id):
|
|
|
|
columns = Message.get_fields()
|
|
|
|
rows = discussions_db.select(
|
2023-08-04 19:39:05 +02:00
|
|
|
f"SELECT {','.join(columns)} FROM message WHERE id=?", (message_id,)
|
2023-08-03 01:07:29 +02:00
|
|
|
)
|
|
|
|
data_dict={
|
2023-08-04 19:40:33 +02:00
|
|
|
col:rows[0][i]
|
2023-08-03 01:07:29 +02:00
|
|
|
for i,col in enumerate(columns)
|
|
|
|
}
|
|
|
|
data_dict["discussions_db"]=discussions_db
|
|
|
|
return Message(
|
|
|
|
**data_dict
|
|
|
|
)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def from_dict(discussions_db,data_dict):
|
|
|
|
data_dict["discussions_db"]=discussions_db
|
|
|
|
return Message(
|
|
|
|
**data_dict
|
|
|
|
)
|
|
|
|
|
|
|
|
def insert_into_db(self):
|
|
|
|
self.message_id = self.discussions_db.insert(
|
2023-10-09 01:18:21 +02:00
|
|
|
"INSERT INTO message (sender, content, metadata, ui, message_type, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at, discussion_id) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
2023-08-30 02:35:26 +02:00
|
|
|
(self.sender, self.content, self.metadata, self.ui, self.message_type, self.rank, self.parent_message_id, self.binding, self.model, self.personality, self.created_at, self.finished_generating_at, self.discussion_id)
|
2023-08-03 01:07:29 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
def update_db(self):
|
|
|
|
self.message_id = self.discussions_db.insert(
|
2023-10-09 01:18:21 +02:00
|
|
|
"INSERT INTO message (sender, content, metadata, ui, message_type, rank, parent_message_id, binding, model, personality, created_at, finished_generating_at, discussion_id) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
2023-08-30 02:35:26 +02:00
|
|
|
(self.sender, self.content, self.metadata, self.ui, self.message_type, self.rank, self.parent_message_id, self.binding, self.model, self.personality, self.created_at, self.finished_generating_at, self.discussion_id)
|
2023-08-03 01:07:29 +02:00
|
|
|
)
|
2023-08-30 02:35:26 +02:00
|
|
|
def update(self, new_content, new_metadata=None, new_ui=None, commit=True):
|
2023-08-03 01:07:29 +02:00
|
|
|
self.finished_generating_at = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
|
2023-08-30 02:35:26 +02:00
|
|
|
text = f"UPDATE message SET content = ?"
|
|
|
|
params = [new_content]
|
|
|
|
if new_metadata is not None:
|
|
|
|
text+=", metadata = ?"
|
|
|
|
params.append(new_metadata)
|
|
|
|
if new_ui is not None:
|
|
|
|
text+=", ui = ?"
|
|
|
|
params.append(new_ui)
|
|
|
|
|
|
|
|
text +=", finished_generating_at = ? WHERE id = ?"
|
|
|
|
params.append(self.finished_generating_at)
|
|
|
|
params.append(self.id)
|
|
|
|
self.discussions_db.update(
|
|
|
|
text, tuple(params)
|
|
|
|
)
|
|
|
|
|
2023-08-03 01:07:29 +02:00
|
|
|
def to_json(self):
|
|
|
|
attributes = Message.get_fields()
|
|
|
|
msgJson = {}
|
|
|
|
for attribute_name in attributes:
|
|
|
|
attribute_value = getattr(self, attribute_name, None)
|
2023-08-18 01:29:53 +02:00
|
|
|
if attribute_name=="metadata":
|
|
|
|
if type(attribute_value) == str:
|
|
|
|
msgJson[attribute_name] = json.loads(attribute_value)
|
|
|
|
else:
|
|
|
|
msgJson[attribute_name] = attribute_value
|
|
|
|
else:
|
|
|
|
msgJson[attribute_name] = attribute_value
|
2023-08-03 01:07:29 +02:00
|
|
|
return msgJson
|
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
class Discussion:
|
|
|
|
def __init__(self, discussion_id, discussions_db:DiscussionsDB):
|
|
|
|
self.discussion_id = discussion_id
|
|
|
|
self.discussions_db = discussions_db
|
2023-09-14 10:38:33 +02:00
|
|
|
self.messages = self.get_messages()
|
2023-09-14 13:30:15 +02:00
|
|
|
if len(self.messages)>0:
|
|
|
|
self.current_message = self.messages[-1]
|
2023-04-10 10:27:25 +02:00
|
|
|
|
2023-07-14 01:42:29 +02:00
|
|
|
def load_message(self, id):
|
|
|
|
"""Gets a list of messages information
|
|
|
|
|
|
|
|
Returns:
|
2023-10-09 01:18:21 +02:00
|
|
|
list: List of entries in the format {"id":message id, "sender":sender name, "content":message content, "message_type":message type, "rank": message rank}
|
2023-07-14 01:42:29 +02:00
|
|
|
"""
|
2023-08-03 01:07:29 +02:00
|
|
|
self.current_message = Message.from_db(self.discussions_db, id)
|
|
|
|
return self.current_message
|
2023-07-14 01:42:29 +02:00
|
|
|
|
2023-08-03 01:07:29 +02:00
|
|
|
def add_message(
|
|
|
|
self,
|
|
|
|
message_type,
|
|
|
|
sender_type,
|
|
|
|
sender,
|
|
|
|
content,
|
|
|
|
metadata=None,
|
2023-08-30 02:35:26 +02:00
|
|
|
ui=None,
|
2023-08-03 01:07:29 +02:00
|
|
|
rank=0,
|
|
|
|
parent_message_id=0,
|
|
|
|
binding="",
|
|
|
|
model ="",
|
|
|
|
personality="",
|
|
|
|
created_at=None,
|
|
|
|
finished_generating_at=None
|
|
|
|
):
|
2023-04-10 10:27:25 +02:00
|
|
|
"""Adds a new message to the discussion
|
|
|
|
|
|
|
|
Args:
|
|
|
|
sender (str): The sender name
|
|
|
|
content (str): The text sent by the sender
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
int: The added message id
|
2023-04-08 19:55:33 +02:00
|
|
|
"""
|
2023-06-20 14:21:49 +02:00
|
|
|
if created_at is None:
|
|
|
|
created_at = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
|
|
|
|
|
|
|
|
if finished_generating_at is None:
|
|
|
|
finished_generating_at = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
|
2023-06-18 09:06:52 +02:00
|
|
|
|
2023-08-03 01:07:29 +02:00
|
|
|
self.current_message = Message(
|
|
|
|
self.discussion_id,
|
|
|
|
self.discussions_db,
|
|
|
|
message_type,
|
|
|
|
sender_type,
|
|
|
|
sender,
|
|
|
|
content,
|
|
|
|
metadata,
|
2023-08-30 02:35:26 +02:00
|
|
|
ui,
|
2023-08-03 01:07:29 +02:00
|
|
|
rank,
|
|
|
|
parent_message_id,
|
|
|
|
binding,
|
|
|
|
model,
|
|
|
|
personality,
|
|
|
|
created_at,
|
|
|
|
finished_generating_at,
|
|
|
|
insert_into_db=True
|
2023-04-08 19:55:33 +02:00
|
|
|
)
|
2023-08-03 01:07:29 +02:00
|
|
|
|
|
|
|
self.messages.append(self.current_message)
|
|
|
|
return self.current_message
|
2023-04-10 10:27:25 +02:00
|
|
|
|
|
|
|
def rename(self, new_title):
|
|
|
|
"""Renames the discussion
|
|
|
|
|
|
|
|
Args:
|
|
|
|
new_title (str): The nex discussion name
|
2023-04-08 19:55:33 +02:00
|
|
|
"""
|
2023-04-10 11:02:00 +02:00
|
|
|
self.discussions_db.update(
|
2023-04-10 16:18:01 +02:00
|
|
|
f"UPDATE discussion SET title=? WHERE id=?",(new_title,self.discussion_id)
|
2023-04-08 19:55:33 +02:00
|
|
|
)
|
|
|
|
|
2023-11-26 02:33:25 +01:00
|
|
|
def title(self):
|
|
|
|
"""Renames the discussion
|
|
|
|
|
|
|
|
Args:
|
|
|
|
new_title (str): The nex discussion name
|
|
|
|
"""
|
|
|
|
rows = self.discussions_db.select(
|
|
|
|
f"Select title from discussion WHERE id={self.discussion_id}"
|
|
|
|
)
|
|
|
|
return rows[0][0]
|
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
def delete_discussion(self):
|
|
|
|
"""Deletes the discussion
|
|
|
|
"""
|
2023-04-10 11:02:00 +02:00
|
|
|
self.discussions_db.delete(
|
2023-04-10 10:27:25 +02:00
|
|
|
f"DELETE FROM message WHERE discussion_id={self.discussion_id}"
|
|
|
|
)
|
2023-04-10 11:02:00 +02:00
|
|
|
self.discussions_db.delete(
|
2023-04-10 10:27:25 +02:00
|
|
|
f"DELETE FROM discussion WHERE id={self.discussion_id}"
|
|
|
|
)
|
|
|
|
|
|
|
|
def get_messages(self):
|
|
|
|
"""Gets a list of messages information
|
|
|
|
|
|
|
|
Returns:
|
2023-10-09 01:18:21 +02:00
|
|
|
list: List of entries in the format {"id":message id, "sender":sender name, "content":message content, "message_type":message type, "rank": message rank}
|
2023-04-10 10:27:25 +02:00
|
|
|
"""
|
2023-08-03 01:07:29 +02:00
|
|
|
columns = Message.get_fields()
|
|
|
|
|
2023-04-10 10:27:25 +02:00
|
|
|
rows = self.discussions_db.select(
|
2023-08-03 01:07:29 +02:00
|
|
|
f"SELECT {','.join(columns)} FROM message WHERE discussion_id=?", (self.discussion_id,)
|
2023-04-10 10:27:25 +02:00
|
|
|
)
|
2023-08-03 01:07:29 +02:00
|
|
|
msg_dict = [{ c:row[i] for i,c in enumerate(columns)} for row in rows]
|
|
|
|
self.messages=[]
|
|
|
|
for msg in msg_dict:
|
|
|
|
self.messages.append(Message.from_dict(self.discussions_db, msg))
|
|
|
|
|
|
|
|
if len(self.messages)>0:
|
|
|
|
self.current_message = self.messages[-1]
|
2023-04-13 21:40:46 +02:00
|
|
|
|
2023-08-03 01:07:29 +02:00
|
|
|
return self.messages
|
|
|
|
|
2023-08-23 04:21:58 +02:00
|
|
|
def get_message(self, message_id):
|
2023-08-03 01:07:29 +02:00
|
|
|
for message in self.messages:
|
2023-08-23 04:21:58 +02:00
|
|
|
if message.id == int(message_id):
|
2023-08-03 01:07:29 +02:00
|
|
|
self.current_message = message
|
|
|
|
return message
|
|
|
|
return None
|
|
|
|
|
2023-08-23 04:21:58 +02:00
|
|
|
def select_message(self, message_id):
|
|
|
|
msg = self.get_message(message_id)
|
|
|
|
if msg is not None:
|
|
|
|
self.current_message = msg
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2023-08-30 02:35:26 +02:00
|
|
|
def update_message(self, new_content, new_metadata=None, new_ui=None):
|
2023-04-10 10:27:25 +02:00
|
|
|
"""Updates the content of a message
|
|
|
|
|
|
|
|
Args:
|
|
|
|
message_id (int): The id of the message to be changed
|
|
|
|
new_content (str): The nex message content
|
|
|
|
"""
|
2023-08-30 02:35:26 +02:00
|
|
|
self.current_message.update(new_content, new_metadata, new_ui)
|
2023-08-23 04:21:58 +02:00
|
|
|
|
2023-08-30 02:35:26 +02:00
|
|
|
def edit_message(self, message_id, new_content, new_metadata=None, new_ui=None):
|
2023-08-23 04:21:58 +02:00
|
|
|
"""Edits the content of a message
|
|
|
|
|
|
|
|
Args:
|
|
|
|
message_id (int): The id of the message to be changed
|
|
|
|
new_content (str): The nex message content
|
|
|
|
"""
|
|
|
|
msg = self.get_message(message_id)
|
|
|
|
if msg:
|
2023-08-30 02:35:26 +02:00
|
|
|
msg.update(new_content, new_metadata, new_ui)
|
2023-08-23 04:21:58 +02:00
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2023-04-10 16:14:39 +02:00
|
|
|
def message_rank_up(self, message_id):
|
|
|
|
"""Increments the rank of the message
|
2023-04-08 19:55:33 +02:00
|
|
|
|
2023-04-10 16:14:39 +02:00
|
|
|
Args:
|
|
|
|
message_id (int): The id of the message to be changed
|
|
|
|
"""
|
|
|
|
# Retrieve current rank value for message_id
|
|
|
|
current_rank = self.discussions_db.select("SELECT rank FROM message WHERE id=?", (message_id,),False)[0]
|
|
|
|
|
|
|
|
# Increment current rank value by 1
|
|
|
|
new_rank = current_rank + 1
|
|
|
|
self.discussions_db.update(
|
|
|
|
f"UPDATE message SET rank = ? WHERE id = ?",(new_rank,message_id)
|
|
|
|
)
|
|
|
|
return new_rank
|
|
|
|
|
|
|
|
def message_rank_down(self, message_id):
|
|
|
|
"""Increments the rank of the message
|
|
|
|
|
|
|
|
Args:
|
|
|
|
message_id (int): The id of the message to be changed
|
|
|
|
"""
|
|
|
|
# Retrieve current rank value for message_id
|
|
|
|
current_rank = self.discussions_db.select("SELECT rank FROM message WHERE id=?", (message_id,),False)[0]
|
|
|
|
|
|
|
|
# Increment current rank value by 1
|
|
|
|
new_rank = current_rank - 1
|
|
|
|
self.discussions_db.update(
|
|
|
|
f"UPDATE message SET rank = ? WHERE id = ?",(new_rank,message_id)
|
|
|
|
)
|
|
|
|
return new_rank
|
2023-04-13 12:31:48 +02:00
|
|
|
|
|
|
|
def delete_message(self, message_id):
|
|
|
|
"""Delete the message
|
|
|
|
|
|
|
|
Args:
|
|
|
|
message_id (int): The id of the message to be deleted
|
|
|
|
"""
|
|
|
|
# Retrieve current rank value for message_id
|
|
|
|
self.discussions_db.delete("DELETE FROM message WHERE id=?", (message_id,))
|
2023-04-08 19:55:33 +02:00
|
|
|
|
2023-12-04 01:40:36 +01:00
|
|
|
def export_for_vectorization(self):
|
|
|
|
"""
|
|
|
|
Export all discussions and their messages from the database to a Markdown list format.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
list: A list of lists representing discussions and their messages in a Markdown format.
|
|
|
|
Each inner list contains the discussion title and a string representing all
|
|
|
|
messages in the discussion in a Markdown format.
|
|
|
|
"""
|
|
|
|
# Extract the title
|
|
|
|
title = self.title()
|
|
|
|
messages = ""
|
|
|
|
# Iterate through messages in the discussion
|
|
|
|
for message in self.messages:
|
|
|
|
sender = message.sender
|
|
|
|
content = message.content
|
|
|
|
# Append the sender and content in a Markdown format
|
|
|
|
messages += f'{sender}: {content}\n'
|
|
|
|
return title, messages
|
2023-04-08 19:55:33 +02:00
|
|
|
# ========================================================================================================================
|