Compare commits
1 Commits
dev
...
d5f56d8b0c
| Author | SHA1 | Date | |
|---|---|---|---|
| d5f56d8b0c |
@@ -1,13 +0,0 @@
|
|||||||
---
|
|
||||||
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
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
---
|
|
||||||
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
|
|
||||||
@@ -1,14 +1,14 @@
|
|||||||
name: SonarQube
|
name: SonarQube
|
||||||
on:
|
on:
|
||||||
pull_request:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- master
|
- master
|
||||||
|
pull_request:
|
||||||
types: [opened, synchronize, reopened]
|
types: [opened, synchronize, reopened]
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
name: Build and analyze
|
name: Build and analyze
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
timeout-minutes: 20
|
|
||||||
steps:
|
steps:
|
||||||
- name: Set up JDK 17
|
- name: Set up JDK 17
|
||||||
uses: actions/setup-java@v4
|
uses: actions/setup-java@v4
|
||||||
|
|||||||
@@ -1,49 +0,0 @@
|
|||||||
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 }}"
|
|
||||||
@@ -3,7 +3,7 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- master
|
- master
|
||||||
- dev
|
- develop
|
||||||
pull_request:
|
pull_request:
|
||||||
types: [opened, synchronize, reopened]
|
types: [opened, synchronize, reopened]
|
||||||
|
|
||||||
@@ -11,7 +11,6 @@ jobs:
|
|||||||
test:
|
test:
|
||||||
name: Run Tests
|
name: Run Tests
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
timeout-minutes: 10
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
|
|||||||
@@ -1,240 +0,0 @@
|
|||||||
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");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -359,7 +359,7 @@ public class ChatSessionTests
|
|||||||
var session = new ChatSession();
|
var session = new ChatSession();
|
||||||
session.AddUserMessage("Test", "user");
|
session.AddUserMessage("Test", "user");
|
||||||
var lastUpdated = session.LastUpdatedAt;
|
var lastUpdated = session.LastUpdatedAt;
|
||||||
await Task.Delay(10, CancellationToken.None); // Small delay
|
await Task.Delay(10); // Small delay
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
session.ClearHistory();
|
session.ClearHistory();
|
||||||
|
|||||||
@@ -232,7 +232,7 @@ public class InMemorySessionStorageTests
|
|||||||
var originalTime = session.LastUpdatedAt;
|
var originalTime = session.LastUpdatedAt;
|
||||||
|
|
||||||
// Wait a bit to ensure time difference
|
// Wait a bit to ensure time difference
|
||||||
await Task.Delay(10, CancellationToken.None);
|
await Task.Delay(10);
|
||||||
|
|
||||||
session.ChatTitle = "Updated Title";
|
session.ChatTitle = "Updated Title";
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user