94 lines
4.3 KiB
Python
94 lines
4.3 KiB
Python
from datetime import datetime, timezone
|
|
from sqlalchemy.dialects.mysql import LONGTEXT
|
|
from backend import db
|
|
|
|
|
|
class User(db.Model):
|
|
__tablename__ = "users"
|
|
|
|
id = db.Column(db.BigInteger, primary_key=True, autoincrement=True)
|
|
username = db.Column(db.String(50), unique=True, nullable=False)
|
|
password = db.Column(db.String(255), nullable=True) # Allow NULL for third-party login
|
|
phone = db.Column(db.String(20))
|
|
|
|
conversations = db.relationship("Conversation", backref="user", lazy="dynamic",
|
|
cascade="all, delete-orphan",
|
|
order_by="Conversation.updated_at.desc()")
|
|
|
|
|
|
class Conversation(db.Model):
|
|
__tablename__ = "conversations"
|
|
|
|
id = db.Column(db.String(64), primary_key=True)
|
|
user_id = db.Column(db.BigInteger, db.ForeignKey("users.id"), nullable=False, index=True)
|
|
title = db.Column(db.String(255), nullable=False, default="")
|
|
model = db.Column(db.String(64), nullable=False, default="glm-5")
|
|
system_prompt = db.Column(db.Text, default="")
|
|
temperature = db.Column(db.Float, nullable=False, default=1.0)
|
|
max_tokens = db.Column(db.Integer, nullable=False, default=65536)
|
|
thinking_enabled = db.Column(db.Boolean, default=False)
|
|
created_at = db.Column(db.DateTime, default=lambda: datetime.now(timezone.utc))
|
|
updated_at = db.Column(db.DateTime, default=lambda: datetime.now(timezone.utc),
|
|
onupdate=lambda: datetime.now(timezone.utc), index=True)
|
|
|
|
messages = db.relationship("Message", backref="conversation", lazy="dynamic",
|
|
cascade="all, delete-orphan",
|
|
order_by="Message.created_at.asc()")
|
|
|
|
|
|
class Message(db.Model):
|
|
__tablename__ = "messages"
|
|
|
|
id = db.Column(db.String(64), primary_key=True)
|
|
conversation_id = db.Column(db.String(64), db.ForeignKey("conversations.id"),
|
|
nullable=False, index=True)
|
|
role = db.Column(db.String(16), nullable=False) # user, assistant, system, tool
|
|
content = db.Column(LONGTEXT, default="") # LONGTEXT for long conversations
|
|
token_count = db.Column(db.Integer, default=0)
|
|
thinking_content = db.Column(LONGTEXT, default="")
|
|
created_at = db.Column(db.DateTime, default=lambda: datetime.now(timezone.utc), index=True)
|
|
|
|
# Tool call support - relation to ToolCall table
|
|
tool_calls = db.relationship("ToolCall", backref="message", lazy="dynamic",
|
|
cascade="all, delete-orphan",
|
|
order_by="ToolCall.call_index.asc()")
|
|
|
|
|
|
class ToolCall(db.Model):
|
|
"""Tool call record - separate table, follows database normalization"""
|
|
__tablename__ = "tool_calls"
|
|
|
|
id = db.Column(db.BigInteger, primary_key=True, autoincrement=True)
|
|
message_id = db.Column(db.String(64), db.ForeignKey("messages.id"),
|
|
nullable=False, index=True)
|
|
call_id = db.Column(db.String(64), nullable=False) # Tool call ID
|
|
call_index = db.Column(db.Integer, nullable=False, default=0) # Call order
|
|
tool_name = db.Column(db.String(64), nullable=False) # Tool name
|
|
arguments = db.Column(LONGTEXT, nullable=False) # Call arguments JSON
|
|
result = db.Column(LONGTEXT) # Execution result JSON
|
|
execution_time = db.Column(db.Float, default=0) # Execution time (seconds)
|
|
created_at = db.Column(db.DateTime, default=lambda: datetime.now(timezone.utc))
|
|
|
|
__table_args__ = (
|
|
db.Index("ix_tool_calls_message_call", "message_id", "call_index"),
|
|
)
|
|
|
|
|
|
class TokenUsage(db.Model):
|
|
__tablename__ = "token_usage"
|
|
|
|
id = db.Column(db.BigInteger, primary_key=True, autoincrement=True)
|
|
user_id = db.Column(db.BigInteger, db.ForeignKey("users.id"),
|
|
nullable=False, index=True)
|
|
date = db.Column(db.Date, nullable=False, index=True)
|
|
model = db.Column(db.String(64), nullable=False)
|
|
prompt_tokens = db.Column(db.Integer, default=0)
|
|
completion_tokens = db.Column(db.Integer, default=0)
|
|
total_tokens = db.Column(db.Integer, default=0)
|
|
created_at = db.Column(db.DateTime, default=lambda: datetime.now(timezone.utc))
|
|
|
|
__table_args__ = (
|
|
db.UniqueConstraint("user_id", "date", "model", name="uq_user_date_model"),
|
|
db.Index("ix_token_usage_date_model", "date", "model"), # Composite index
|
|
)
|