use std::sync::Arc; use anyhow::Result; use client::telemetry::Telemetry; use collections::HashMap; use command_palette_hooks::CommandPaletteFilter; use gpui::{ prelude::*, AppContext, EntityId, Global, Model, ModelContext, Subscription, Task, View, }; use project::Fs; use settings::{Settings, SettingsStore}; use crate::kernels::kernel_specifications; use crate::{JupyterSettings, KernelSpecification, Session}; struct GlobalReplStore(Model); impl Global for GlobalReplStore {} pub struct ReplStore { fs: Arc, enabled: bool, sessions: HashMap>, kernel_specifications: Vec, telemetry: Arc, _subscriptions: Vec, } impl ReplStore { const NAMESPACE: &'static str = "repl"; pub(crate) fn init(fs: Arc, telemetry: Arc, cx: &mut AppContext) { let store = cx.new_model(move |cx| Self::new(fs, telemetry, cx)); store .update(cx, |store, cx| store.refresh_kernelspecs(cx)) .detach_and_log_err(cx); cx.set_global(GlobalReplStore(store)) } pub fn global(cx: &AppContext) -> Model { cx.global::().0.clone() } pub fn new(fs: Arc, telemetry: Arc, cx: &mut ModelContext) -> Self { let subscriptions = vec![cx.observe_global::(move |this, cx| { this.set_enabled(JupyterSettings::enabled(cx), cx); })]; let this = Self { fs, telemetry, enabled: JupyterSettings::enabled(cx), sessions: HashMap::default(), kernel_specifications: Vec::new(), _subscriptions: subscriptions, }; this.on_enabled_changed(cx); this } pub fn fs(&self) -> &Arc { &self.fs } pub fn telemetry(&self) -> &Arc { &self.telemetry } pub fn is_enabled(&self) -> bool { self.enabled } pub fn kernel_specifications(&self) -> impl Iterator { self.kernel_specifications.iter() } pub fn sessions(&self) -> impl Iterator> { self.sessions.values() } fn set_enabled(&mut self, enabled: bool, cx: &mut ModelContext) { if self.enabled == enabled { return; } self.enabled = enabled; self.on_enabled_changed(cx); } fn on_enabled_changed(&self, cx: &mut ModelContext) { if !self.enabled { CommandPaletteFilter::update_global(cx, |filter, _cx| { filter.hide_namespace(Self::NAMESPACE); }); return; } CommandPaletteFilter::update_global(cx, |filter, _cx| { filter.show_namespace(Self::NAMESPACE); }); cx.notify(); } pub fn refresh_kernelspecs(&mut self, cx: &mut ModelContext) -> Task> { let kernel_specifications = kernel_specifications(self.fs.clone()); cx.spawn(|this, mut cx| async move { let kernel_specifications = kernel_specifications.await?; this.update(&mut cx, |this, cx| { this.kernel_specifications = kernel_specifications; cx.notify(); }) }) } pub fn kernelspec(&self, language_name: &str, cx: &AppContext) -> Option { let settings = JupyterSettings::get_global(cx); let selected_kernel = settings.kernel_selections.get(language_name); let found_by_name = self .kernel_specifications .iter() .find(|runtime_specification| { if let Some(selected) = selected_kernel { // Top priority is the selected kernel return runtime_specification.name.to_lowercase() == selected.to_lowercase(); } return false; }) .cloned(); if let Some(found_by_name) = found_by_name { return Some(found_by_name); } self.kernel_specifications .iter() .find(|runtime_specification| { runtime_specification.kernelspec.language.to_lowercase() == language_name.to_lowercase() }) .cloned() } pub fn get_session(&self, entity_id: EntityId) -> Option<&View> { self.sessions.get(&entity_id) } pub fn insert_session(&mut self, entity_id: EntityId, session: View) { self.sessions.insert(entity_id, session); } pub fn remove_session(&mut self, entity_id: EntityId) { self.sessions.remove(&entity_id); } }