Compare commits

...

16 Commits

Author SHA1 Message Date
29c5174f0a Merge pull request 'dev' (#9) from dev into master
Some checks failed
Publish Docker Image / Build and Publish to Harbor (push) Failing after 18s
Tests / Run Tests (push) Successful in 2m51s
Reviewed-on: #9
2025-10-23 11:25:20 +00:00
Leonid Pershin
b188afd9ab Update workflow to trigger on 'dev' branch instead of 'develop'
All checks were successful
Tests / Run Tests (push) Successful in 3m10s
Tests / Run Tests (pull_request) Successful in 2m52s
SonarQube / Build and analyze (pull_request) Successful in 3m55s
2025-10-23 09:43:15 +03:00
Leonid Pershin
3e3df20d84 Update build workflow to trigger on pull requests instead of pushes 2025-10-23 09:41:27 +03:00
445b96af21 Merge pull request 'Enhance workflow configurations by adding timeouts for build, publish, and test jobs' (#8) from dev into master
Some checks failed
SonarQube / Build and analyze (push) Failing after 13m20s
Publish Docker Image / Build and Publish to Harbor (push) Failing after 16s
Tests / Run Tests (push) Successful in 3m51s
Reviewed-on: #8
2025-10-23 06:20:54 +00:00
Leonid Pershin
e54d44b581 Enhance workflow configurations by adding timeouts for build, publish, and test jobs
All checks were successful
SonarQube / Build and analyze (pull_request) Successful in 3m27s
Tests / Run Tests (pull_request) Successful in 2m34s
2025-10-23 09:14:15 +03:00
4d56b29e67 Merge pull request 'fix pub' (#7) from dev into master
Some checks failed
SonarQube / Build and analyze (push) Successful in 17m44s
Publish Docker Image / Run Tests (push) Successful in 3m36s
Publish Docker Image / SonarQube Analysis (push) Failing after 21m20s
Publish Docker Image / Build and Publish to Harbor (push) Has been skipped
Tests / Run Tests (push) Successful in 3m56s
Reviewed-on: #7
2025-10-23 05:13:19 +00:00
Leonid Pershin
738ae73ebd fix pub
All checks were successful
SonarQube / Build and analyze (pull_request) Successful in 12m33s
Tests / Run Tests (pull_request) Successful in 3m36s
2025-10-23 07:56:42 +03:00
a57108cf21 Merge pull request 'add' (#6) from dev into master
All checks were successful
SonarQube / Build and analyze (push) Successful in 3m28s
Tests / Run Tests (push) Successful in 2m34s
Reviewed-on: #6
2025-10-22 09:45:19 +00:00
Leonid Pershin
3adbc189eb add
All checks were successful
SonarQube / Build and analyze (pull_request) Successful in 3m10s
Tests / Run Tests (pull_request) Successful in 2m37s
2025-10-22 12:38:16 +03:00
a9569b6985 Merge pull request 'add docker pub' (#5) from dev into master
Some checks failed
SonarQube / Build and analyze (push) Successful in 3m2s
Publish Docker Image / Build and Publish to Harbor (push) Failing after 2m11s
Tests / Run Tests (push) Successful in 2m26s
Reviewed-on: #5
2025-10-22 04:15:55 +00:00
Leonid Pershin
a4bcb78295 add docker pub
All checks were successful
SonarQube / Build and analyze (pull_request) Successful in 3m7s
Tests / Run Tests (pull_request) Successful in 2m27s
2025-10-22 07:09:46 +03:00
0c97d0bef0 Merge pull request 'fix issues' (#4) from dev into master
All checks were successful
SonarQube / Build and analyze (push) Successful in 3m7s
Tests / Run Tests (push) Successful in 2m51s
Reviewed-on: #4
2025-10-22 02:50:26 +00:00
Leonid Pershin
9063ddb881 fix issues
All checks were successful
SonarQube / Build and analyze (pull_request) Successful in 3m3s
Tests / Run Tests (pull_request) Successful in 2m29s
2025-10-22 05:42:11 +03:00
98f0a0c154 Merge pull request 'Add tests' (#3) from dev into master
Some checks failed
SonarQube / Build and analyze (push) Failing after 3m51s
Tests / Run Tests (push) Successful in 2m27s
Reviewed-on: #3
2025-10-22 02:23:16 +00:00
Leonid Pershin
594e4a1782 fix tests
All checks were successful
SonarQube / Build and analyze (pull_request) Successful in 3m40s
Tests / Run Tests (pull_request) Successful in 2m30s
2025-10-22 05:16:58 +03:00
Leonid Pershin
c03de646cc Add tests
Some checks failed
SonarQube / Build and analyze (pull_request) Failing after 1m40s
Tests / Run Tests (pull_request) Failing after 1m11s
2025-10-22 04:41:56 +03:00
8 changed files with 355 additions and 5 deletions

13
.cursor/rules/default.mdc Normal file
View File

@@ -0,0 +1,13 @@
---
alwaysApply: true
---
MCP предоставляет ассистенту доступ к данным SonarQube. Используй инструменты для:
Поиска проблем: search_sonar_issues_in_projects
Проверки статуса: get_project_quality_gate_status, get_system_status, get_system_health
Анализа кода: analyze_code_snippet, get_raw_source
Работы с задачами: change_sonar_issue_status
Получения метрик: get_component_measures, search_metrics
Получение документации по библиотекам: use context7
Не гадай — запрашивай данные. Уточняй ключи проектов и issue. Действуй точно, опираясь на информацию из SonarQube.
Текущий проект ChatBot

View File

@@ -0,0 +1,47 @@
---
description: SonarQube MCP Server usage guidelines
globs:
alwaysApply: true
---
These are some guidelines when using the SonarQube MCP server.
# Important Tool Guidelines
## Basic usage
- When starting a new task, disable automatic analysis with the `toggle_automatic_analysis` tool if it exists.
- When you are done generating code at the very end of the task, re-enable automatic analysis with the `toggle_automatic_analysis` tool if it exists.
Then call the `analyze_file_list` tool if it exists.
## Project Keys
- When a user mentions a project key, use `search_my_sonarqube_projects` first to find the exact project key
- Don't guess project keys - always look them up
## Code Language Detection
- When analyzing code snippets, try to detect the programming language from the code syntax
- If unclear, ask the user or make an educated guess based on syntax
## Branch and Pull Request Context
- Many operations support branch-specific analysis
- If user mentions working on a feature branch, include the branch parameter
- Pull request analysis is available for PR-specific insights
## Code Issues and Violations
- After fixing issues, do not attempt to verify them using `search_sonar_issues_in_projects`, as the server will not yet reflect the updates
# Common Troubleshooting
## Authentication Issues
- SonarQube requires USER tokens (not project tokens)
- When the error `SonarQube answered with Not authorized` occurs, verify the token type
## Project Not Found
- Use `search_my_sonarqube_projects` to confirm available projects
- Check if user has access to the specific project
- Verify project key spelling and format
## Code Analysis Issues
- Ensure programming language is correctly specified
- Remind users that snippet analysis doesn't replace full project scans
- Provide full file content for better analysis results
- Mention that code snippet analysis tool has limited capabilities compared to full SonarQube scans

View File

@@ -1,14 +1,14 @@
name: SonarQube
on:
push:
pull_request:
branches:
- master
pull_request:
types: [opened, synchronize, reopened]
jobs:
build:
name: Build and analyze
runs-on: ubuntu-latest
timeout-minutes: 20
steps:
- name: Set up JDK 17
uses: actions/setup-java@v4

View File

@@ -0,0 +1,49 @@
name: Publish Docker Image
on:
push:
branches:
- master
jobs:
publish:
name: Build and Publish to Harbor
runs-on: ubuntu-latest
timeout-minutes: 15
steps:
- name: Checkout code
uses: actions/checkout@v4
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3
- name: Log in to Harbor
uses: docker/login-action@v3
with:
registry: harbor.home
username: robot$chatbot
password: ${{ secrets.HARBOR_TOKEN }}
- name: Extract metadata
id: meta
uses: docker/metadata-action@v5
with:
images: harbor.home/chatbot/chatbot
tags: |
type=ref,event=branch
type=sha,prefix={{branch}}-
type=raw,value=latest,enable={{is_default_branch}}
- name: Build and push Docker image
uses: docker/build-push-action@v5
with:
context: ./ChatBot
file: ./ChatBot/Dockerfile
push: true
tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }}
cache-from: type=registry,ref=harbor.home/chatbot/chatbot:buildcache
cache-to: type=registry,ref=harbor.home/chatbot/chatbot:buildcache,mode=max
- name: Image digest
run: echo "Image published with digest ${{ steps.build.outputs.digest }}"

View File

@@ -3,7 +3,7 @@ on:
push:
branches:
- master
- develop
- dev
pull_request:
types: [opened, synchronize, reopened]
@@ -11,6 +11,7 @@ jobs:
test:
name: Run Tests
runs-on: ubuntu-latest
timeout-minutes: 10
steps:
- uses: actions/checkout@v4
with:

View File

@@ -0,0 +1,240 @@
using ChatBot.Models;
using ChatBot.Models.Dto;
using ChatBot.Services.Interfaces;
using FluentAssertions;
using Moq;
using OllamaSharp.Models.Chat;
using System.Collections.Concurrent;
namespace ChatBot.Tests.Models;
public class ChatSessionCompressionTests
{
[Fact]
public async Task CompressHistoryAsync_ShouldCompressMessages_WhenCompressionServiceAvailable()
{
// Arrange
var session = new ChatSession();
var compressionServiceMock = new Mock<IHistoryCompressionService>();
session.SetCompressionService(compressionServiceMock.Object);
// Setup compression service to return compressed messages
var compressedMessages = new List<ChatMessage>
{
new ChatMessage { Role = ChatRole.System.ToString(), Content = "System prompt" },
new ChatMessage { Role = ChatRole.User.ToString(), Content = "Compressed user message" }
};
compressionServiceMock
.Setup(x => x.CompressHistoryAsync(It.IsAny<List<ChatMessage>>(), It.IsAny<int>(), It.IsAny<CancellationToken>()))
.ReturnsAsync(compressedMessages);
compressionServiceMock
.Setup(x => x.ShouldCompress(It.IsAny<int>(), It.IsAny<int>()))
.Returns(true);
// Add messages to session
for (int i = 0; i < 10; i++)
{
session.AddMessage(new ChatMessage { Role = ChatRole.User, Content = $"Message {i}" });
}
// Act
await session.AddMessageWithCompressionAsync(
new ChatMessage { Role = ChatRole.User, Content = "New message" },
compressionThreshold: 5,
compressionTarget: 2
);
// Assert
var messages = session.GetAllMessages();
messages.Should().HaveCount(2);
messages[0].Role.Should().Be(ChatRole.System);
messages[1].Role.Should().Be(ChatRole.User);
messages[1].Content.Should().Be("Compressed user message");
}
[Fact]
public async Task CompressHistoryAsync_ShouldFallbackToTrimming_WhenCompressionFails()
{
// Arrange
var session = new ChatSession { MaxHistoryLength = 3 };
var compressionServiceMock = new Mock<IHistoryCompressionService>();
session.SetCompressionService(compressionServiceMock.Object);
// Setup compression service to throw an exception
var exception = new Exception("Compression failed");
compressionServiceMock
.Setup(x => x.CompressHistoryAsync(It.IsAny<List<ChatMessage>>(), It.IsAny<int>(), It.IsAny<CancellationToken>()))
.ThrowsAsync(exception);
compressionServiceMock
.Setup(x => x.ShouldCompress(It.IsAny<int>(), It.IsAny<int>()))
.Returns(true);
// Add messages to session
for (int i = 0; i < 5; i++)
{
session.AddMessage(new ChatMessage { Role = ChatRole.User, Content = $"Message {i}" });
}
// Act
await session.AddMessageWithCompressionAsync(
new ChatMessage { Role = ChatRole.User, Content = "New message" },
compressionThreshold: 3,
compressionTarget: 2
);
// Assert - Should fall back to simple trimming
var messages = session.GetAllMessages();
messages.Should().HaveCount(3);
}
[Fact]
public async Task AddMessageWithCompressionAsync_ShouldNotCompress_WhenBelowThreshold()
{
// Arrange
var session = new ChatSession();
var compressionServiceMock = new Mock<IHistoryCompressionService>();
session.SetCompressionService(compressionServiceMock.Object);
// Setup compression service to return false for ShouldCompress when count is below threshold
compressionServiceMock
.Setup(x => x.ShouldCompress(It.Is<int>(c => c < 5), It.Is<int>(t => t == 5)))
.Returns(false);
// Add messages to session (below threshold)
session.AddMessage(new ChatMessage { Role = ChatRole.User, Content = "Message 1" });
session.AddMessage(new ChatMessage { Role = ChatRole.Assistant, Content = "Response 1" });
// Act - Set threshold higher than current message count
await session.AddMessageWithCompressionAsync(
new ChatMessage { Role = ChatRole.User, Content = "Message 2" },
compressionThreshold: 5,
compressionTarget: 2
);
// Assert - Should not call compression service
compressionServiceMock.Verify(
x => x.CompressHistoryAsync(It.IsAny<List<ChatMessage>>(), It.IsAny<int>(), It.IsAny<CancellationToken>()),
Times.Never
);
var messages = session.GetAllMessages();
messages.Should().HaveCount(3);
}
[Fact]
public async Task AddMessageWithCompressionAsync_ShouldHandleConcurrentAccess()
{
// Arrange
var session = new ChatSession();
var compressionServiceMock = new Mock<IHistoryCompressionService>();
session.SetCompressionService(compressionServiceMock.Object);
// Setup compression service to simulate processing time
var delayedResult = new List<ChatMessage>
{
new ChatMessage { Role = ChatRole.System.ToString(), Content = "Compressed" }
};
compressionServiceMock
.Setup(x => x.CompressHistoryAsync(It.IsAny<List<ChatMessage>>(), It.IsAny<int>(), It.IsAny<CancellationToken>()))
.Returns(async (List<ChatMessage> messages, int target, CancellationToken ct) =>
{
await Task.Delay(50, ct);
return delayedResult;
});
compressionServiceMock
.Setup(x => x.ShouldCompress(It.IsAny<int>(), It.IsAny<int>()))
.Returns(true);
var tasks = new List<Task>();
int messageCount = 5;
// Act - Start multiple concurrent operations
for (int i = 0; i < messageCount; i++)
{
tasks.Add(
session.AddMessageWithCompressionAsync(
new ChatMessage { Role = ChatRole.User, Content = $"Message {i}" },
compressionThreshold: 2,
compressionTarget: 1
)
);
}
// Wait for all operations to complete
await Task.WhenAll(tasks);
// Assert - Should handle concurrent access without exceptions
// and maintain thread safety
session.GetMessageCount().Should().Be(1);
}
[Fact]
public void SetCompressionService_ShouldNotThrow_WhenCalledMultipleTimes()
{
// Arrange
var session = new ChatSession();
var compressionService1 = new Mock<IHistoryCompressionService>().Object;
var compressionService2 = new Mock<IHistoryCompressionService>().Object;
// Act & Assert
session.Invoking(s => s.SetCompressionService(compressionService1)).Should().NotThrow();
// Should not throw when setting a different service
session.Invoking(s => s.SetCompressionService(compressionService2)).Should().NotThrow();
}
[Fact]
public async Task CompressHistoryAsync_ShouldPreserveSystemMessage_WhenCompressing()
{
// Arrange
var session = new ChatSession();
var compressionServiceMock = new Mock<IHistoryCompressionService>();
session.SetCompressionService(compressionServiceMock.Object);
// Setup compression service to preserve system message
compressionServiceMock
.Setup(x => x.CompressHistoryAsync(It.IsAny<List<ChatMessage>>(), It.IsAny<int>(), It.IsAny<CancellationToken>()))
.Returns((List<ChatMessage> messages, int target, CancellationToken ct) =>
{
var systemMessage = messages.FirstOrDefault(m => m.Role == ChatRole.System.ToString());
var compressed = new List<ChatMessage>();
if (systemMessage != null)
{
compressed.Add(systemMessage);
}
compressed.Add(new ChatMessage
{
Role = ChatRole.User.ToString(),
Content = "Compressed user messages"
});
return Task.FromResult(compressed);
});
compressionServiceMock
.Setup(x => x.ShouldCompress(It.IsAny<int>(), It.IsAny<int>()))
.Returns(true);
// Add system message and some user messages
session.AddMessage(new ChatMessage { Role = ChatRole.System, Content = "System prompt" });
for (int i = 0; i < 10; i++)
{
session.AddMessage(new ChatMessage { Role = ChatRole.User, Content = $"Message {i}" });
}
// Act
await session.AddMessageWithCompressionAsync(
new ChatMessage { Role = ChatRole.User, Content = "New message" },
compressionThreshold: 5,
compressionTarget: 2
);
// Assert - System message should be preserved
var messages = session.GetAllMessages();
messages.Should().HaveCount(2);
messages[0].Role.Should().Be(ChatRole.System);
messages[0].Content.Should().Be("System prompt");
messages[1].Content.Should().Be("Compressed user messages");
}
}

View File

@@ -359,7 +359,7 @@ public class ChatSessionTests
var session = new ChatSession();
session.AddUserMessage("Test", "user");
var lastUpdated = session.LastUpdatedAt;
await Task.Delay(10); // Small delay
await Task.Delay(10, CancellationToken.None); // Small delay
// Act
session.ClearHistory();

View File

@@ -232,7 +232,7 @@ public class InMemorySessionStorageTests
var originalTime = session.LastUpdatedAt;
// Wait a bit to ensure time difference
await Task.Delay(10);
await Task.Delay(10, CancellationToken.None);
session.ChatTitle = "Updated Title";