use anyhow::Result; use assistant_slash_command::{ ArgumentCompletion, SlashCommand, SlashCommandOutput, SlashCommandOutputSection, SlashCommandResult, }; use feature_flags::FeatureFlag; use gpui::{App, Task, WeakEntity}; use language::{CodeLabel, LspAdapterDelegate}; use semantic_index::{LoadedSearchResult, SemanticDb}; use std::{ fmt::Write, sync::{atomic::AtomicBool, Arc}, }; use ui::{prelude::*, IconName}; use workspace::Workspace; use crate::create_label_for_command; use crate::file_command::{build_entry_output_section, codeblock_fence_for_path}; pub struct SearchSlashCommandFeatureFlag; impl FeatureFlag for SearchSlashCommandFeatureFlag { const NAME: &'static str = "search-slash-command"; fn enabled_for_staff() -> bool { false } } pub struct SearchSlashCommand; impl SlashCommand for SearchSlashCommand { fn name(&self) -> String { "search".into() } fn label(&self, cx: &App) -> CodeLabel { create_label_for_command("search", &["--n"], cx) } fn description(&self) -> String { "Search your project semantically".into() } fn icon(&self) -> IconName { IconName::SearchCode } fn menu_text(&self) -> String { self.description() } fn requires_argument(&self) -> bool { true } fn complete_argument( self: Arc, _arguments: &[String], _cancel: Arc, _workspace: Option>, _window: &mut Window, _cx: &mut App, ) -> Task>> { Task::ready(Ok(Vec::new())) } fn run( self: Arc, arguments: &[String], _context_slash_command_output_sections: &[SlashCommandOutputSection], _context_buffer: language::BufferSnapshot, workspace: WeakEntity, _delegate: Option>, window: &mut Window, cx: &mut App, ) -> Task { let Some(workspace) = workspace.upgrade() else { return Task::ready(Err(anyhow::anyhow!("workspace was dropped"))); }; if arguments.is_empty() { return Task::ready(Err(anyhow::anyhow!("missing search query"))); }; let mut limit = None; let mut query = String::new(); for part in arguments { if let Some(parameter) = part.strip_prefix("--") { if let Ok(count) = parameter.parse::() { limit = Some(count); continue; } } query.push_str(part); query.push(' '); } query.pop(); if query.is_empty() { return Task::ready(Err(anyhow::anyhow!("missing search query"))); } let project = workspace.read(cx).project().clone(); let fs = project.read(cx).fs().clone(); let Some(project_index) = cx.update_global(|index: &mut SemanticDb, cx| index.project_index(project, cx)) else { return Task::ready(Err(anyhow::anyhow!("no project indexer"))); }; window.spawn(cx, |cx| async move { let results = project_index .read_with(&cx, |project_index, cx| { project_index.search(vec![query.clone()], limit.unwrap_or(5), cx) })? .await?; let loaded_results = SemanticDb::load_results(results, &fs, &cx).await?; let output = cx .background_spawn(async move { let mut text = format!("Search results for {query}:\n"); let mut sections = Vec::new(); for loaded_result in &loaded_results { add_search_result_section(loaded_result, &mut text, &mut sections); } let query = SharedString::from(query); sections.push(SlashCommandOutputSection { range: 0..text.len(), icon: IconName::MagnifyingGlass, label: query, metadata: None, }); SlashCommandOutput { text, sections, run_commands_in_text: false, } .to_event_stream() }) .await; Ok(output) }) } } pub fn add_search_result_section( loaded_result: &LoadedSearchResult, text: &mut String, sections: &mut Vec>, ) { let LoadedSearchResult { path, full_path, excerpt_content, row_range, .. } = loaded_result; let section_start_ix = text.len(); text.push_str(&codeblock_fence_for_path( Some(&path), Some(row_range.clone()), )); text.push_str(&excerpt_content); if !text.ends_with('\n') { text.push('\n'); } writeln!(text, "```\n").unwrap(); let section_end_ix = text.len() - 1; sections.push(build_entry_output_section( section_start_ix..section_end_ix, Some(&full_path), false, Some(row_range.start() + 1..row_range.end() + 1), )); }