| | | 1 | | namespace UIBlazor.Services; |
| | | 2 | | |
| | 9 | 3 | | public class SkillService(IVsBridge vsBridge) : ISkillService |
| | | 4 | | { |
| | | 5 | | private List<SkillMetadata>? _skillsCache; |
| | 9 | 6 | | private readonly Dictionary<string, SkillContent> _contentCache = new(); |
| | 9 | 7 | | private DateTime _lastCacheUpdate = DateTime.MinValue; |
| | | 8 | | |
| | | 9 | | /// <summary> |
| | | 10 | | /// Получить метаданные всех скиллов (кешируется) |
| | | 11 | | /// Вызывается при старте и при изменении файлов |
| | | 12 | | /// </summary> |
| | | 13 | | public async Task<List<SkillMetadata>> GetSkillsMetadataAsync(CancellationToken cancellationToken) |
| | | 14 | | { |
| | | 15 | | // Проверяем кеш (обновляем раз в 2 минуты или по запросу) |
| | 6 | 16 | | if (_skillsCache != null && (DateTime.UtcNow - _lastCacheUpdate).TotalMinutes < 2) |
| | | 17 | | { |
| | 1 | 18 | | return _skillsCache; |
| | | 19 | | } |
| | | 20 | | |
| | 5 | 21 | | var result = await vsBridge.ExecuteToolAsync(BasicEnum.GetSkillsMetadata, cancellationToken: cancellationToken); |
| | 5 | 22 | | if (!result.Success) |
| | | 23 | | { |
| | 1 | 24 | | return _skillsCache ?? []; |
| | | 25 | | } |
| | | 26 | | |
| | | 27 | | try |
| | | 28 | | { |
| | 4 | 29 | | var metadataJson = JsonSerializer.Deserialize<List<Dictionary<string, string>>>(result.Result); |
| | 7 | 30 | | _skillsCache = metadataJson?.Select(m => new SkillMetadata |
| | 7 | 31 | | { |
| | 7 | 32 | | Name = m.GetValueOrDefault("name", ""), |
| | 7 | 33 | | Description = m.GetValueOrDefault("description", "") |
| | 7 | 34 | | }).ToList() ?? []; |
| | | 35 | | |
| | 3 | 36 | | _lastCacheUpdate = DateTime.UtcNow; |
| | 3 | 37 | | return _skillsCache; |
| | | 38 | | } |
| | 1 | 39 | | catch |
| | | 40 | | { |
| | 1 | 41 | | return _skillsCache ?? []; |
| | | 42 | | } |
| | 6 | 43 | | } |
| | | 44 | | |
| | | 45 | | /// <summary> |
| | | 46 | | /// Загрузить полное содержимое скилла (с кешированием) |
| | | 47 | | /// Вызывается только когда агент активирует скилл |
| | | 48 | | /// </summary> |
| | | 49 | | public async Task<SkillContent?> LoadSkillContentAsync(string skillName, CancellationToken cancellationToken) |
| | | 50 | | { |
| | | 51 | | // Проверяем кеш содержимого |
| | 18 | 52 | | if (_contentCache.TryGetValue(skillName, out var cachedContent)) |
| | | 53 | | { |
| | 2 | 54 | | return cachedContent; |
| | | 55 | | } |
| | | 56 | | |
| | 16 | 57 | | var args = new Dictionary<string, object> |
| | 16 | 58 | | { |
| | 16 | 59 | | { "param1", skillName } |
| | 16 | 60 | | }; |
| | | 61 | | |
| | 16 | 62 | | var result = await vsBridge.ExecuteToolAsync(BasicEnum.ReadSkillContent, args, cancellationToken); |
| | 16 | 63 | | if (!result.Success) |
| | | 64 | | { |
| | 1 | 65 | | return null; |
| | | 66 | | } |
| | | 67 | | |
| | | 68 | | try |
| | | 69 | | { |
| | 15 | 70 | | var skillContent = JsonUtils.Deserialize<SkillContent>(result.Result); |
| | 15 | 71 | | if (skillContent == null) |
| | 0 | 72 | | return null; |
| | | 73 | | |
| | | 74 | | // Кешируем содержимое (максимум 10 скиллов в кеше) |
| | 15 | 75 | | if (_contentCache.Count >= 10) |
| | | 76 | | { |
| | 2 | 77 | | var oldestKey = _contentCache.Keys.First(); |
| | 2 | 78 | | _contentCache.Remove(oldestKey); |
| | | 79 | | } |
| | 15 | 80 | | _contentCache[skillName] = skillContent; |
| | | 81 | | |
| | 15 | 82 | | return skillContent; |
| | | 83 | | } |
| | 0 | 84 | | catch |
| | | 85 | | { |
| | 0 | 86 | | return null; |
| | | 87 | | } |
| | 18 | 88 | | } |
| | | 89 | | |
| | | 90 | | /// <summary> |
| | | 91 | | /// Форматирует список скиллов для добавления в системный промпт |
| | | 92 | | /// Только название и описание (триггеры для активации) |
| | | 93 | | /// </summary> |
| | | 94 | | public string FormatSkillsForSystemPrompt(List<SkillMetadata> skills) |
| | | 95 | | { |
| | 2 | 96 | | if (skills.Count == 0) |
| | | 97 | | { |
| | 1 | 98 | | return string.Empty; |
| | | 99 | | } |
| | | 100 | | |
| | 1 | 101 | | var sb = new StringBuilder(); |
| | 1 | 102 | | sb.AppendLine("## Available Skills"); |
| | 1 | 103 | | sb.AppendLine(); |
| | 1 | 104 | | sb.AppendLine("You have access to the following skills. Skills are specialized instructions that you can activat |
| | 1 | 105 | | sb.AppendLine(); |
| | | 106 | | |
| | 4 | 107 | | foreach (var skill in skills) |
| | | 108 | | { |
| | 1 | 109 | | sb.AppendLine(); |
| | 1 | 110 | | sb.AppendLine($""" |
| | 1 | 111 | | - **{skill.Name}**: {skill.Description} |
| | 1 | 112 | | To read instructions: |
| | 1 | 113 | | <function name="{BasicEnum.ReadSkillContent}"> |
| | 1 | 114 | | {skill.Name} |
| | 1 | 115 | | </function> |
| | 1 | 116 | | """); |
| | 1 | 117 | | sb.AppendLine(); |
| | | 118 | | } |
| | | 119 | | |
| | 1 | 120 | | sb.AppendLine($"When you need detailed instructions from a skill, use `{BasicEnum.ReadSkillContent}` tool to loa |
| | | 121 | | |
| | 1 | 122 | | return sb.ToString(); |
| | | 123 | | } |
| | | 124 | | |
| | | 125 | | /// <summary> |
| | | 126 | | /// Принудительно обновить кеш скиллов |
| | | 127 | | /// При изменении файлов (через FileSystemWatcher) |
| | | 128 | | /// </summary> |
| | | 129 | | public async Task RefreshCacheAsync(CancellationToken cancellationToken) |
| | | 130 | | { |
| | 1 | 131 | | _lastCacheUpdate = DateTime.MinValue; // Сбрасываем кеш |
| | 1 | 132 | | _contentCache.Clear(); // Очищаем кеш содержимого |
| | 1 | 133 | | await GetSkillsMetadataAsync(cancellationToken); // Перезагружаем метаданные |
| | 1 | 134 | | } |
| | | 135 | | |
| | | 136 | | public async Task<VsToolResult> LoadSkillContentMarkDownAsync(IReadOnlyDictionary<string, object> args, Cancellation |
| | | 137 | | { |
| | 0 | 138 | | var skillName = args.GetString("param1"); |
| | 0 | 139 | | if (string.IsNullOrEmpty(skillName)) |
| | | 140 | | { |
| | 0 | 141 | | return new VsToolResult |
| | 0 | 142 | | { |
| | 0 | 143 | | Success = false, |
| | 0 | 144 | | Result = "Skill name is required parameter!" |
| | 0 | 145 | | }; |
| | | 146 | | } |
| | 0 | 147 | | var skillContent = await LoadSkillContentAsync(skillName, cancellationToken); |
| | | 148 | | |
| | 0 | 149 | | if (skillContent == null) |
| | | 150 | | { |
| | 0 | 151 | | return new VsToolResult |
| | 0 | 152 | | { |
| | 0 | 153 | | Success = false, |
| | 0 | 154 | | Result = "Skill content is empty..." |
| | 0 | 155 | | }; |
| | | 156 | | } |
| | 0 | 157 | | var sb = new StringBuilder(); |
| | 0 | 158 | | sb.AppendLine(); |
| | 0 | 159 | | sb.AppendLine($"## Skill {skillName}"); |
| | 0 | 160 | | sb.AppendLine(skillContent.Content); |
| | | 161 | | |
| | 0 | 162 | | return new VsToolResult { Result = sb.ToString() }; |
| | 0 | 163 | | } |
| | | 164 | | } |