/* * Copyright (C) 2024-2025 Petr Mironychev * * This file is part of QodeAssist. * * QodeAssist is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * QodeAssist is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with QodeAssist. If not, see . */ #include "ChatModel.hpp" #include #include #include #include #include "ChatAssistantSettings.hpp" #include "Logger.hpp" namespace QodeAssist::Chat { ChatModel::ChatModel(QObject *parent) : QAbstractListModel(parent) { auto &settings = Settings::chatAssistantSettings(); connect( &settings.chatTokensThreshold, &Utils::BaseAspect::changed, this, &ChatModel::tokensThresholdChanged); } int ChatModel::rowCount(const QModelIndex &parent) const { return m_messages.size(); } QVariant ChatModel::data(const QModelIndex &index, int role) const { if (!index.isValid() || index.row() >= m_messages.size()) return QVariant(); const Message &message = m_messages[index.row()]; switch (static_cast(role)) { case Roles::RoleType: return QVariant::fromValue(message.role); case Roles::Content: { return message.content; } case Roles::Attachments: { QStringList filenames; for (const auto &attachment : message.attachments) { filenames << attachment.filename; } return filenames; } default: return QVariant(); } } QHash ChatModel::roleNames() const { QHash roles; roles[Roles::RoleType] = "roleType"; roles[Roles::Content] = "content"; roles[Roles::Attachments] = "attachments"; return roles; } void ChatModel::addMessage( const QString &content, ChatRole role, const QString &id, const QList &attachments) { QString fullContent = content; if (!attachments.isEmpty()) { fullContent += "\n\nAttached files list:"; for (const auto &attachment : attachments) { fullContent += QString("\nname: %1\nfile content:\n%2") .arg(attachment.filename, attachment.content); } } if (!m_messages.isEmpty() && !id.isEmpty() && m_messages.last().id == id && m_messages.last().role == role) { Message &lastMessage = m_messages.last(); lastMessage.content = content; lastMessage.attachments = attachments; emit dataChanged(index(m_messages.size() - 1), index(m_messages.size() - 1)); } else { beginInsertRows(QModelIndex(), m_messages.size(), m_messages.size()); Message newMessage{role, content, id}; newMessage.attachments = attachments; m_messages.append(newMessage); endInsertRows(); } } QVector ChatModel::getChatHistory() const { return m_messages; } void ChatModel::clear() { beginResetModel(); m_messages.clear(); endResetModel(); emit modelReseted(); } QList ChatModel::processMessageContent(const QString &content) const { QList parts; QRegularExpression codeBlockRegex("```(\\w*)\\n?([\\s\\S]*?)```"); int lastIndex = 0; auto blockMatches = codeBlockRegex.globalMatch(content); while (blockMatches.hasNext()) { auto match = blockMatches.next(); if (match.capturedStart() > lastIndex) { QString textBetween = content.mid(lastIndex, match.capturedStart() - lastIndex).trimmed(); if (!textBetween.isEmpty()) { MessagePart part; part.type = MessagePartType::Text; part.text = textBetween; parts.append(part); } } MessagePart codePart; codePart.type = MessagePartType::Code; codePart.text = match.captured(2).trimmed(); codePart.language = match.captured(1); parts.append(codePart); lastIndex = match.capturedEnd(); } if (lastIndex < content.length()) { QString remainingText = content.mid(lastIndex).trimmed(); QRegularExpression unclosedBlockRegex("```(\\w*)\\n?([\\s\\S]*)$"); auto unclosedMatch = unclosedBlockRegex.match(remainingText); if (unclosedMatch.hasMatch()) { QString beforeCodeBlock = remainingText.left(unclosedMatch.capturedStart()).trimmed(); if (!beforeCodeBlock.isEmpty()) { MessagePart part; part.type = MessagePartType::Text; part.text = beforeCodeBlock; parts.append(part); } MessagePart codePart; codePart.type = MessagePartType::Code; codePart.text = unclosedMatch.captured(2).trimmed(); codePart.language = unclosedMatch.captured(1); parts.append(codePart); } else if (!remainingText.isEmpty()) { MessagePart part; part.type = MessagePartType::Text; part.text = remainingText; parts.append(part); } } return parts; } QJsonArray ChatModel::prepareMessagesForRequest(const QString &systemPrompt) const { QJsonArray messages; messages.append(QJsonObject{{"role", "system"}, {"content", systemPrompt}}); for (const auto &message : m_messages) { QString role; switch (message.role) { case ChatRole::User: role = "user"; break; case ChatRole::Assistant: role = "assistant"; break; case ChatRole::Tool: case ChatRole::FileEdit: // Skip Tool and FileEdit messages - they are UI-only continue; default: continue; } QString content = message.attachments.isEmpty() ? message.content : message.content + "\n\nAttached files list:" + std::accumulate( message.attachments.begin(), message.attachments.end(), QString(), [](QString acc, const Context::ContentFile &attachment) { return acc + QString("\nname: %1\nfile content:\n%2") .arg(attachment.filename, attachment.content); }); messages.append(QJsonObject{{"role", role}, {"content", content}}); } return messages; } int ChatModel::tokensThreshold() const { auto &settings = Settings::chatAssistantSettings(); return settings.chatTokensThreshold(); } QString ChatModel::lastMessageId() const { return !m_messages.isEmpty() ? m_messages.last().id : ""; } void ChatModel::resetModelTo(int index) { if (index < 0 || index >= m_messages.size()) return; if (index < m_messages.size()) { beginRemoveRows(QModelIndex(), index, m_messages.size() - 1); m_messages.remove(index, m_messages.size() - index); endRemoveRows(); } } void ChatModel::addToolExecutionStatus( const QString &requestId, const QString &toolId, const QString &toolName) { QString content = toolName; LOG_MESSAGE(QString("Adding tool execution status: requestId=%1, toolId=%2, toolName=%3") .arg(requestId, toolId, toolName)); if (!m_messages.isEmpty() && !toolId.isEmpty() && m_messages.last().id == toolId && m_messages.last().role == ChatRole::Tool) { Message &lastMessage = m_messages.last(); lastMessage.content = content; LOG_MESSAGE(QString("Updated existing tool message at index %1").arg(m_messages.size() - 1)); emit dataChanged(index(m_messages.size() - 1), index(m_messages.size() - 1)); } else { beginInsertRows(QModelIndex(), m_messages.size(), m_messages.size()); Message newMessage{ChatRole::Tool, content, toolId}; m_messages.append(newMessage); endInsertRows(); LOG_MESSAGE(QString("Created new tool message at index %1 with toolId=%2") .arg(m_messages.size() - 1) .arg(toolId)); } } void ChatModel::updateToolResult( const QString &requestId, const QString &toolId, const QString &toolName, const QString &result) { if (m_messages.isEmpty() || toolId.isEmpty()) { LOG_MESSAGE(QString("Cannot update tool result: messages empty=%1, toolId empty=%2") .arg(m_messages.isEmpty()) .arg(toolId.isEmpty())); return; } LOG_MESSAGE( QString("Updating tool result: requestId=%1, toolId=%2, toolName=%3, result length=%4") .arg(requestId, toolId, toolName) .arg(result.length())); bool toolMessageFound = false; for (int i = m_messages.size() - 1; i >= 0; --i) { if (m_messages[i].id == toolId && m_messages[i].role == ChatRole::Tool) { m_messages[i].content = toolName + "\n" + result; emit dataChanged(index(i), index(i)); toolMessageFound = true; LOG_MESSAGE(QString("Updated tool result at index %1").arg(i)); break; } } if (!toolMessageFound) { LOG_MESSAGE(QString("WARNING: Tool message with requestId=%1 toolId=%2 not found!") .arg(requestId, toolId)); } const QString marker = "QODEASSIST_FILE_EDIT:"; if (result.contains(marker)) { LOG_MESSAGE(QString("File edit marker detected in tool result")); int markerPos = result.indexOf(marker); int jsonStart = markerPos + marker.length(); if (jsonStart < result.length()) { QString jsonStr = result.mid(jsonStart); QJsonParseError parseError; QJsonDocument doc = QJsonDocument::fromJson(jsonStr.toUtf8(), &parseError); if (parseError.error != QJsonParseError::NoError) { LOG_MESSAGE(QString("ERROR: Failed to parse file edit JSON at offset %1: %2") .arg(parseError.offset) .arg(parseError.errorString())); } else if (!doc.isObject()) { LOG_MESSAGE( QString("ERROR: Parsed JSON is not an object, is array=%1").arg(doc.isArray())); } else { QJsonObject editData = doc.object(); QString editId = editData["edit_id"].toString(); LOG_MESSAGE(QString("Adding FileEdit message, editId=%1").arg(editId)); beginInsertRows(QModelIndex(), m_messages.size(), m_messages.size()); Message fileEditMsg; fileEditMsg.role = ChatRole::FileEdit; fileEditMsg.content = result; fileEditMsg.id = editId.isEmpty() ? QString("edit_%1").arg(requestId) : editId; m_messages.append(fileEditMsg); endInsertRows(); LOG_MESSAGE(QString("Added FileEdit message with editId=%1").arg(editId)); } } } } } // namespace QodeAssist::Chat