registry.rs

  1use anyhow::Result;
  2use async_trait::async_trait;
  3use collections::FxHashMap;
  4use gpui::{App, Global, SharedString};
  5use language::LanguageName;
  6use parking_lot::RwLock;
  7use task::{
  8    AdapterSchema, AdapterSchemas, DebugRequest, DebugScenario, SpawnInTerminal, TaskTemplate,
  9};
 10
 11use crate::{
 12    adapters::{DebugAdapter, DebugAdapterName},
 13    inline_value::InlineValueProvider,
 14};
 15use std::{collections::BTreeMap, sync::Arc};
 16
 17/// Given a user build configuration, locator creates a fill-in debug target ([DebugScenario]) on behalf of the user.
 18#[async_trait]
 19pub trait DapLocator: Send + Sync {
 20    fn name(&self) -> SharedString;
 21    /// Determines whether this locator can generate debug target for given task.
 22    async fn create_scenario(
 23        &self,
 24        build_config: &TaskTemplate,
 25        resolved_label: &str,
 26        adapter: &DebugAdapterName,
 27    ) -> Option<DebugScenario>;
 28
 29    async fn run(&self, build_config: SpawnInTerminal) -> Result<DebugRequest>;
 30}
 31
 32#[derive(Default)]
 33struct DapRegistryState {
 34    adapters: BTreeMap<DebugAdapterName, Arc<dyn DebugAdapter>>,
 35    locators: FxHashMap<SharedString, Arc<dyn DapLocator>>,
 36    inline_value_providers: FxHashMap<String, Arc<dyn InlineValueProvider>>,
 37}
 38
 39#[derive(Clone, Default)]
 40/// Stores available debug adapters.
 41pub struct DapRegistry(Arc<RwLock<DapRegistryState>>);
 42impl Global for DapRegistry {}
 43
 44impl DapRegistry {
 45    pub fn global(cx: &mut App) -> &mut Self {
 46        cx.default_global::<Self>()
 47    }
 48
 49    pub fn add_adapter(&self, adapter: Arc<dyn DebugAdapter>) {
 50        let name = adapter.name();
 51        let _previous_value = self.0.write().adapters.insert(name, adapter);
 52    }
 53    pub fn add_locator(&self, locator: Arc<dyn DapLocator>) {
 54        self.0.write().locators.insert(locator.name(), locator);
 55    }
 56
 57    pub fn remove_adapter(&self, name: &str) {
 58        self.0.write().adapters.remove(name);
 59    }
 60
 61    pub fn remove_locator(&self, locator: &str) {
 62        self.0.write().locators.remove(locator);
 63    }
 64
 65    pub fn adapter_language(&self, adapter_name: &str) -> Option<LanguageName> {
 66        self.adapter(adapter_name)
 67            .and_then(|adapter| adapter.adapter_language_name())
 68    }
 69
 70    pub async fn adapters_schema(&self) -> task::AdapterSchemas {
 71        let mut schemas = AdapterSchemas(vec![]);
 72
 73        let adapters = self.0.read().adapters.clone();
 74
 75        for (name, adapter) in adapters.into_iter() {
 76            schemas.0.push(AdapterSchema {
 77                adapter: name.into(),
 78                schema: adapter.dap_schema(),
 79            });
 80        }
 81
 82        schemas
 83    }
 84
 85    pub fn add_inline_value_provider(
 86        &self,
 87        language: String,
 88        provider: Arc<dyn InlineValueProvider>,
 89    ) {
 90        let _previous_value = self
 91            .0
 92            .write()
 93            .inline_value_providers
 94            .insert(language, provider);
 95        debug_assert!(
 96            _previous_value.is_none(),
 97            "Attempted to insert a new inline value provider when one is already registered"
 98        );
 99    }
100
101    pub fn locators(&self) -> FxHashMap<SharedString, Arc<dyn DapLocator>> {
102        self.0.read().locators.clone()
103    }
104
105    pub fn adapter(&self, name: &str) -> Option<Arc<dyn DebugAdapter>> {
106        self.0.read().adapters.get(name).cloned()
107    }
108
109    pub fn inline_value_provider(&self, language: &str) -> Option<Arc<dyn InlineValueProvider>> {
110        self.0.read().inline_value_providers.get(language).cloned()
111    }
112
113    pub fn enumerate_adapters(&self) -> Vec<DebugAdapterName> {
114        self.0.read().adapters.keys().cloned().collect()
115    }
116}