Skip to content

Commit 719065e

Browse files
authored
refactor: Extract document reading to separate class (#127)
This decouples LLMClientInterface from Qt Creator text editor implementation and allows to write tests
1 parent a218064 commit 719065e

6 files changed

+99
-7
lines changed

LLMClientInterface.cpp

+5-7
Original file line numberDiff line numberDiff line change
@@ -23,8 +23,6 @@
2323
#include <QNetworkAccessManager>
2424
#include <QNetworkReply>
2525

26-
#include <texteditor/textdocument.h>
27-
2826
#include "CodeHandler.hpp"
2927
#include "context/ContextManager.hpp"
3028
#include "context/DocumentContextReader.hpp"
@@ -44,12 +42,14 @@ LLMClientInterface::LLMClientInterface(
4442
LLMCore::IProviderRegistry &providerRegistry,
4543
LLMCore::IPromptProvider *promptProvider,
4644
LLMCore::RequestHandlerBase &requestHandler,
45+
Context::IDocumentReader &documentReader,
4746
IRequestPerformanceLogger &performanceLogger)
4847
: m_generalSettings(generalSettings)
4948
, m_completeSettings(completeSettings)
5049
, m_providerRegistry(providerRegistry)
5150
, m_promptProvider(promptProvider)
5251
, m_requestHandler(requestHandler)
52+
, m_documentReader(documentReader)
5353
, m_performanceLogger(performanceLogger)
5454
{
5555
connect(
@@ -261,12 +261,10 @@ LLMCore::ContextData LLMClientInterface::prepareContext(
261261
QJsonObject params = request["params"].toObject();
262262
QJsonObject doc = params["doc"].toObject();
263263
QJsonObject position = doc["position"].toObject();
264-
265264
auto filePath = Context::extractFilePathFromRequest(request);
266-
TextEditor::TextDocument *textDocument = TextEditor::TextDocument::textDocumentForFilePath(
267-
Utils::FilePath::fromString(filePath));
268265

269-
if (!textDocument) {
266+
auto documentInfo = m_documentReader.readDocument(filePath);
267+
if (!documentInfo.document) {
270268
LOG_MESSAGE("Error: Document is not available for" + filePath);
271269
return LLMCore::ContextData{};
272270
}
@@ -275,7 +273,7 @@ LLMCore::ContextData LLMClientInterface::prepareContext(
275273
int lineNumber = position["line"].toInt();
276274

277275
Context::DocumentContextReader
278-
reader(textDocument->document(), textDocument->mimeType(), filePath);
276+
reader(documentInfo.document, documentInfo.mimeType, documentInfo.filePath);
279277
return reader.prepareContext(lineNumber, cursorPosition, m_completeSettings);
280278
}
281279

LLMClientInterface.hpp

+3
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
#include <languageclient/languageclientinterface.h>
2323
#include <texteditor/texteditor.h>
2424

25+
#include <context/IDocumentReader.hpp>
2526
#include <context/ProgrammingLanguage.hpp>
2627
#include <llmcore/ContextData.hpp>
2728
#include <llmcore/IPromptProvider.hpp>
@@ -47,6 +48,7 @@ class LLMClientInterface : public LanguageClient::BaseClientInterface
4748
LLMCore::IProviderRegistry &providerRegistry,
4849
LLMCore::IPromptProvider *promptProvider,
4950
LLMCore::RequestHandlerBase &requestHandler,
51+
Context::IDocumentReader &documentReader,
5052
IRequestPerformanceLogger &performanceLogger);
5153

5254
Utils::FilePath serverDeviceTemplate() const override;
@@ -77,6 +79,7 @@ class LLMClientInterface : public LanguageClient::BaseClientInterface
7779
LLMCore::IPromptProvider *m_promptProvider = nullptr;
7880
LLMCore::IProviderRegistry &m_providerRegistry;
7981
LLMCore::RequestHandlerBase &m_requestHandler;
82+
Context::IDocumentReader &m_documentReader;
8083
IRequestPerformanceLogger &m_performanceLogger;
8184
QElapsedTimer m_completionTimer;
8285
};

context/CMakeLists.txt

+2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@ add_library(Context STATIC
33
ChangesManager.h ChangesManager.cpp
44
ContextManager.hpp ContextManager.cpp
55
ContentFile.hpp
6+
DocumentReaderQtCreator.hpp
7+
IDocumentReader.hpp
68
TokenUtils.hpp TokenUtils.cpp
79
ProgrammingLanguage.hpp ProgrammingLanguage.cpp
810
)

context/DocumentReaderQtCreator.hpp

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
/*
2+
* Copyright (C) 2025 Povilas Kanapickas
3+
*
4+
* This file is part of QodeAssist.
5+
*
6+
* QodeAssist is free software: you can redistribute it and/or modify
7+
* it under the terms of the GNU General Public License as published by
8+
* the Free Software Foundation, either version 3 of the License, or
9+
* (at your option) any later version.
10+
*
11+
* QodeAssist is distributed in the hope that it will be useful,
12+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14+
* GNU General Public License for more details.
15+
*
16+
* You should have received a copy of the GNU General Public License
17+
* along with QodeAssist. If not, see <https://www.gnu.org/licenses/>.
18+
*/
19+
20+
#pragma once
21+
22+
#include "IDocumentReader.hpp"
23+
24+
#include <texteditor/textdocument.h>
25+
26+
namespace QodeAssist::Context {
27+
28+
class DocumentReaderQtCreator : public IDocumentReader
29+
{
30+
public:
31+
DocumentInfo readDocument(const QString &path) const override
32+
{
33+
auto *textDocument = TextEditor::TextDocument::textDocumentForFilePath(
34+
Utils::FilePath::fromString(path));
35+
if (!textDocument) {
36+
return {};
37+
}
38+
return {
39+
.document = textDocument->document(),
40+
.mimeType = textDocument->mimeType(),
41+
.filePath = path};
42+
}
43+
};
44+
45+
} // namespace QodeAssist::Context

context/IDocumentReader.hpp

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
/*
2+
* Copyright (C) 2025 Povilas Kanapickas
3+
*
4+
* This file is part of QodeAssist.
5+
*
6+
* QodeAssist is free software: you can redistribute it and/or modify
7+
* it under the terms of the GNU General Public License as published by
8+
* the Free Software Foundation, either version 3 of the License, or
9+
* (at your option) any later version.
10+
*
11+
* QodeAssist is distributed in the hope that it will be useful,
12+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14+
* GNU General Public License for more details.
15+
*
16+
* You should have received a copy of the GNU General Public License
17+
* along with QodeAssist. If not, see <https://www.gnu.org/licenses/>.
18+
*/
19+
20+
#pragma once
21+
22+
#include <QTextDocument>
23+
24+
namespace QodeAssist::Context {
25+
26+
struct DocumentInfo
27+
{
28+
QTextDocument *document = nullptr; // not owned
29+
QString mimeType;
30+
QString filePath;
31+
};
32+
33+
class IDocumentReader
34+
{
35+
public:
36+
virtual ~IDocumentReader() = default;
37+
38+
virtual DocumentInfo readDocument(const QString &path) const = 0;
39+
};
40+
41+
} // namespace QodeAssist::Context

qodeassist.cpp

+3
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@
4646
#include "Version.hpp"
4747
#include "chat/ChatOutputPane.h"
4848
#include "chat/NavigationPanel.hpp"
49+
#include "context/DocumentReaderQtCreator.hpp"
4950
#include "llmcore/PromptProviderFim.hpp"
5051
#include "llmcore/ProvidersManager.hpp"
5152
#include "logger/RequestPerformanceLogger.hpp"
@@ -146,6 +147,7 @@ class QodeAssistPlugin final : public ExtensionSystem::IPlugin
146147
LLMCore::ProvidersManager::instance(),
147148
&m_promptProvider,
148149
m_requestHandler,
150+
m_documentReader,
149151
m_performanceLogger));
150152
}
151153

@@ -189,6 +191,7 @@ class QodeAssistPlugin final : public ExtensionSystem::IPlugin
189191
QPointer<QodeAssistClient> m_qodeAssistClient;
190192
LLMCore::PromptProviderFim m_promptProvider;
191193
LLMCore::RequestHandler m_requestHandler{this};
194+
Context::DocumentReaderQtCreator m_documentReader;
192195
RequestPerformanceLogger m_performanceLogger;
193196
QPointer<Chat::ChatOutputPane> m_chatOutputPane;
194197
QPointer<Chat::NavigationPanel> m_navigationPanel;

0 commit comments

Comments
 (0)