Skip to main content

ggen_cli_lib/
pack_install.rs

1//! Improved pack installation with better UX and performance
2//!
3//! Features:
4//! - Real-time progress reporting
5//! - Intelligent caching
6//! - Better error handling
7//! - Dependency resolution visualization
8//! - Installation planning and preview
9
10use async_trait::async_trait;
11use futures::future::join_all;
12use std::sync::Arc;
13use tokio::sync::Semaphore;
14
15use crate::error::GgenError;
16use crate::progress::{CacheStatus, InstallationPlan, PlanStep, ProgressReporter};
17use ggen_core::marketplace::{
18    cache::{CacheConfig, CachedPack, PackCache},
19    error::Error as MarketplaceError,
20    AsyncRepository, Package, PackageId, PackageVersion,
21};
22
23/// Improved pack installer with better UX
24pub struct PackInstaller {
25    repository: Box<dyn AsyncRepository<PackageIterator = std::vec::IntoIter<Package>>>,
26    cache: PackCache,
27    progress: ProgressReporter,
28    max_concurrent_downloads: usize,
29}
30
31impl PackInstaller {
32    /// Create a new pack installer
33    pub async fn new() -> Result<Self, GgenError> {
34        // Initialize cache with default configuration
35        let cache_config = CacheConfig::default();
36        let cache = PackCache::new(cache_config)
37            .map_err(|e| GgenError::FileError(format!("Failed to initialize cache: {}", e)))?;
38
39        // Create a simple repository for testing
40        // In production, this would be configured based on user settings
41        let repository = Self::create_default_repository().await?;
42
43        Ok(Self {
44            repository,
45            cache,
46            progress: ProgressReporter::new(),
47            max_concurrent_downloads: 4,
48        })
49    }
50
51    /// Create a repository with default configuration
52    async fn create_default_repository(
53    ) -> Result<Box<dyn AsyncRepository<PackageIterator = std::vec::IntoIter<Package>>>, GgenError>
54    {
55        // For now, use a mock repository
56        // In production, this would connect to the actual marketplace
57        Ok(Box::new(TestRepository {}))
58    }
59
60    /// Install a pack with improved UX
61    pub async fn install_pack(
62        &self, pack_id: &str, force_reinstall: bool, dry_run: bool,
63    ) -> Result<InstallResult, GgenError> {
64        let progress = self.progress.clone();
65        progress.start_operation(&format!("Installing pack: {}", pack_id));
66
67        // Create installation plan first
68        let plan = self.create_installation_plan(pack_id).await?;
69
70        if dry_run {
71            progress.complete();
72            return Ok(InstallResult::DryRun(plan));
73        }
74
75        // Validate plan
76        self.validate_installation_plan(&plan).await?;
77
78        // Execute installation with progress reporting
79        let result = self.execute_installation(&plan, force_reinstall).await;
80
81        progress.complete();
82
83        result
84    }
85
86    /// Create installation plan for user preview
87    async fn create_installation_plan(&self, pack_id: &str) -> Result<InstallationPlan, GgenError> {
88        let progress = self.progress.clone();
89        progress.start_step("Resolving package", 1);
90        progress.set_total_steps(4);
91
92        // Parse package ID
93        let package_id = PackageId::new(pack_id)
94            .map_err(|e| GgenError::ValidationError(format!("Invalid pack ID: {}", e)))?;
95
96        // Get package from repository
97        let package = self
98            .repository
99            .get_package(&package_id)
100            .await
101            .map_err(|e| self.map_repository_error(e, "get package"))?;
102
103        let latest_version = package.latest_version.clone();
104
105        progress.complete_step("Resolving package");
106        progress.start_step("Resolving dependencies", 2);
107
108        // Resolve dependencies
109        let dependencies = self
110            .resolve_dependencies_tree(&package_id, &latest_version)
111            .await?;
112
113        progress.complete_step("Resolving dependencies");
114        progress.start_step("Checking cache status", 3);
115
116        // Check cache status
117        let cache_status = self
118            .check_cache_status(&package_id, &latest_version, &dependencies)
119            .await?;
120
121        progress.complete_step("Checking cache status");
122        progress.start_step("Calculating installation size", 4);
123
124        // Calculate total size
125        let total_size_mb = self.calculate_total_size(&package, &dependencies).await?;
126
127        progress.complete_step("Calculating installation size");
128
129        // Create installation steps
130        let steps = self.create_installation_steps(&package, &dependencies, total_size_mb);
131
132        // Estimate duration (rough estimate: 10MB per second)
133        let estimated_duration_seconds = (total_size_mb / 10.0).max(5.0) as u64;
134
135        Ok(InstallationPlan {
136            pack_id: pack_id.to_string(),
137            total_size_mb,
138            estimated_duration_seconds,
139            total_dependencies: dependencies.len(),
140            steps,
141            cache_status,
142        })
143    }
144
145    /// Resolve dependencies tree
146    async fn resolve_dependencies_tree(
147        &self, package_id: &PackageId, _version: &PackageVersion,
148    ) -> Result<Vec<PackageId>, GgenError> {
149        let mut resolved = Vec::new();
150        let mut visited = std::collections::HashSet::new();
151        let mut to_resolve = vec![package_id.clone()];
152
153        while let Some(id) = to_resolve.pop() {
154            if visited.contains(&id) {
155                continue;
156            }
157
158            visited.insert(id.clone());
159            resolved.push(id.clone());
160
161            // Get package dependencies (mock implementation)
162            // In production, this would call the repository
163            let dependencies = self.get_mock_dependencies(&id).await?;
164
165            for dep_id in dependencies {
166                if !visited.contains(&dep_id) {
167                    to_resolve.push(dep_id);
168                }
169            }
170        }
171
172        Ok(resolved)
173    }
174
175    /// Check cache status for packages
176    async fn check_cache_status(
177        &self, package_id: &PackageId, version: &PackageVersion, dependencies: &[PackageId],
178    ) -> Result<CacheStatus, GgenError> {
179        let mut cached_size_mb = 0.0;
180        let mut cache_hits = 0;
181        let mut total_packages = 1; // Main package
182
183        // Check main package cache
184        if let Some(cached) = self.cache.get(package_id, version) {
185            cached_size_mb += cached.size_bytes as f64 / 1_048_576.0;
186            cache_hits += 1;
187        }
188
189        // Check dependency caches
190        for dep_id in dependencies {
191            // Use latest version for dependencies
192            if let Ok(dep_version) = self.get_latest_version(dep_id).await {
193                if let Some(cached) = self.cache.get(dep_id, &dep_version) {
194                    cached_size_mb += cached.size_bytes as f64 / 1_048_576.0;
195                    cache_hits += 1;
196                }
197            }
198            total_packages += 1;
199        }
200
201        Ok(CacheStatus {
202            is_cached: cache_hits == total_packages,
203            cached_size_mb: Some(cached_size_mb),
204            cache_hit: cache_hits > 0,
205        })
206    }
207
208    /// Calculate total installation size
209    async fn calculate_total_size(
210        &self, _package: &Package, dependencies: &[PackageId],
211    ) -> Result<f64, GgenError> {
212        let mut total_size = 0.0;
213
214        // Main package size
215        total_size += 1.0; // Mock main package size
216
217        // Dependency sizes (mock estimates)
218        for dep_id in dependencies {
219            total_size += self.get_mock_package_size(dep_id).await?;
220        }
221
222        Ok(total_size)
223    }
224
225    /// Create installation steps plan
226    fn create_installation_steps(
227        &self, _package: &Package, dependencies: &[PackageId], total_size_mb: f64,
228    ) -> Vec<PlanStep> {
229        let mut steps = Vec::new();
230        let step_size_mb = total_size_mb / 6.0; // Distribute size across steps
231
232        steps.push(PlanStep {
233            step_number: 1,
234            name: "Validate package".to_string(),
235            description: "Checking package integrity and security".to_string(),
236            estimated_duration_ms: 1000,
237            size_mb: 0.1,
238        });
239
240        steps.push(PlanStep {
241            step_number: 2,
242            name: "Download main package".to_string(),
243            description: "Downloading primary package files".to_string(),
244            estimated_duration_ms: (step_size_mb * 1000.0) as u64,
245            size_mb: step_size_mb,
246        });
247
248        if !dependencies.is_empty() {
249            steps.push(PlanStep {
250                step_number: 3,
251                name: "Download dependencies".to_string(),
252                description: format!("Downloading {} dependency packages", dependencies.len()),
253                estimated_duration_ms: (step_size_mb * dependencies.len() as f64 * 1000.0) as u64,
254                size_mb: step_size_mb * dependencies.len() as f64,
255            });
256        }
257
258        steps.push(PlanStep {
259            step_number: 4,
260            name: "Verify signatures".to_string(),
261            description: "Cryptographic signature verification".to_string(),
262            estimated_duration_ms: 2000,
263            size_mb: 0.1,
264        });
265
266        steps.push(PlanStep {
267            step_number: 5,
268            name: "Extract files".to_string(),
269            description: "Extracting package contents to cache".to_string(),
270            estimated_duration_ms: 1500,
271            size_mb: 0.1,
272        });
273
274        steps.push(PlanStep {
275            step_number: 6,
276            name: "Update lockfile".to_string(),
277            description: "Recording installed packages in lockfile".to_string(),
278            estimated_duration_ms: 500,
279            size_mb: 0.1,
280        });
281
282        steps
283    }
284
285    /// Validate installation plan
286    async fn validate_installation_plan(&self, plan: &InstallationPlan) -> Result<(), GgenError> {
287        // Check if we have enough disk space (require 2x the size)
288        let required_space_mb = plan.total_size_mb * 2.0;
289        let available_space = self.get_available_disk_space().await?;
290
291        if available_space < required_space_mb {
292            return Err(GgenError::ValidationError(format!(
293                "Insufficient disk space: need {:.1} MB, have {:.1} MB",
294                required_space_mb, available_space
295            )));
296        }
297
298        // Check if pack is already installed (unless force_reinstall)
299        // This would be checked during actual execution
300
301        Ok(())
302    }
303
304    /// Execute installation with progress reporting
305    async fn execute_installation(
306        &self, plan: &InstallationPlan, _force_reinstall: bool,
307    ) -> Result<InstallResult, GgenError> {
308        let progress = self.progress.clone();
309        progress.set_total_steps(plan.steps.len());
310
311        let semaphore = Arc::new(Semaphore::new(self.max_concurrent_downloads));
312
313        // Step 1: Validate package
314        progress.start_step(&plan.steps[0].name, 1);
315        progress.update_step_progress(0.0, "Validating package...");
316        self.validate_package(&plan.pack_id).await?;
317        progress.complete_step(&plan.steps[0].name);
318
319        // Step 2: Download main package
320        progress.start_step(&plan.steps[1].name, 2);
321        progress.update_step_progress(0.0, "Downloading main package...");
322
323        let package_id = PackageId::new(&plan.pack_id)
324            .map_err(|e| GgenError::ValidationError(format!("Invalid pack ID: {}", e)))?;
325
326        let _package = self
327            .download_and_cache_package(&package_id, &plan.steps[1].name)
328            .await?;
329        progress.complete_step(&plan.steps[1].name);
330
331        // Step 3: Download dependencies
332        if !plan.steps[2].name.contains("empty") {
333            progress.start_step(&plan.steps[2].name, 3);
334            progress.update_step_progress(0.0, "Downloading dependencies...");
335
336            let dependency_ids = self.get_mock_dependencies(&package_id).await?;
337            let total_deps = dependency_ids.len();
338            let download_tasks = dependency_ids.into_iter().map(|dep_id| {
339                let semaphore = semaphore.clone();
340                let progress = progress.clone();
341                async move {
342                    let _permit = semaphore.acquire().await.unwrap();
343                    let dep_name = format!("Dependency: {}", dep_id);
344                    progress.update_item_progress(&dep_name, 0, total_deps);
345
346                    match self.download_and_cache_package(&dep_id, &dep_name).await {
347                        Ok(_dep_package) => {
348                            progress.update_item_progress(&dep_name, 1, total_deps);
349                            Ok(())
350                        }
351                        Err(e) => {
352                            progress.report_error(
353                                &format!("Failed to download {}: {}", dep_id, e),
354                                &dep_name,
355                            );
356                            Err(e)
357                        }
358                    }
359                }
360            });
361
362            // Run downloads in parallel with limited concurrency
363            let results = join_all(download_tasks).await;
364
365            for result in results {
366                if let Err(e) = result {
367                    return Err(e);
368                }
369            }
370
371            progress.complete_step(&plan.steps[2].name);
372        }
373
374        // Step 4: Verify signatures
375        progress.start_step(&plan.steps[3].name, 4);
376        progress.update_step_progress(0.0, "Verifying signatures...");
377        self.verify_package_signatures(&package_id).await?;
378        progress.complete_step(&plan.steps[3].name);
379
380        // Step 5: Extract files
381        progress.start_step(&plan.steps[4].name, 5);
382        progress.update_step_progress(0.0, "Extracting package files...");
383        self.extract_package_files(&package_id).await?;
384        progress.complete_step(&plan.steps[4].name);
385
386        // Step 6: Update lockfile
387        progress.start_step(&plan.steps[5].name, 6);
388        progress.update_step_progress(0.0, "Updating lockfile...");
389        self.update_installation_lockfile(&plan.pack_id).await?;
390        progress.complete_step(&plan.steps[5].name);
391
392        Ok(InstallResult::Success(InstallationResult {
393            pack_id: plan.pack_id.clone(),
394            installed_packages: vec![package_id],
395            cache_status: plan.cache_status.clone(),
396            total_size_mb: plan.total_size_mb,
397            duration_ms: progress.get_state().elapsed().as_millis() as u64,
398        }))
399    }
400
401    /// Download and cache a package
402    async fn download_and_cache_package(
403        &self, package_id: &PackageId, _step_name: &str,
404    ) -> Result<CachedPack, GgenError> {
405        let progress = self.progress.clone();
406        progress.update_step_progress(0.0, &format!("Checking cache for {}", package_id));
407
408        // Check cache first
409        let latest_version = self.get_latest_version(package_id).await.map_err(|e| {
410            GgenError::ValidationError(format!("No version found for {}: {}", package_id, e))
411        })?;
412
413        if let Some(cached) = self.cache.get(package_id, &latest_version) {
414            progress
415                .update_step_progress(100.0, &format!("Using cached version of {}", package_id));
416            return Ok(cached);
417        }
418
419        // Download from repository
420        progress.update_step_progress(0.0, &format!("Downloading {}", package_id));
421
422        let package = self
423            .repository
424            .get_package_version(package_id, &latest_version)
425            .await
426            .map_err(|e| self.map_repository_error(e, "download package"))?;
427
428        // Mock download simulation (1MB default)
429        let total_size = 1.0 * 1024.0 * 1024.0;
430        let mut downloaded = 0.0;
431        let progress_step = 100.0 / 10.0; // 10 progress steps
432
433        for i in 0..10 {
434            tokio::time::sleep(tokio::time::Duration::from_millis(200)).await;
435            downloaded += total_size / 10.0;
436            progress.update_data_progress(downloaded as u64, total_size as u64);
437            progress.update_step_progress(
438                (i + 1) as f64 * progress_step,
439                &format!("Downloading {}...", i + 1),
440            );
441        }
442
443        // Cache the package
444        let cached_package = self
445            .cache_package(package_id, &latest_version, package)
446            .await?;
447
448        progress.update_step_progress(100.0, &format!("Cached {}", package_id));
449        Ok(cached_package)
450    }
451
452    /// Mock helper methods (replace with real implementations)
453    async fn get_mock_dependencies(
454        &self, _package_id: &PackageId,
455    ) -> Result<Vec<PackageId>, GgenError> {
456        // Return mock dependencies based on package ID
457        Ok(vec![])
458    }
459
460    async fn get_latest_version(
461        &self, _package_id: &PackageId,
462    ) -> Result<PackageVersion, GgenError> {
463        Ok(PackageVersion::new("1.0.0")
464            .map_err(|e| GgenError::ValidationError(format!("Invalid version: {}", e)))?)
465    }
466
467    async fn get_mock_package_size(&self, _package_id: &PackageId) -> Result<f64, GgenError> {
468        Ok(1.0) // 1MB per package
469    }
470
471    async fn get_available_disk_space(&self) -> Result<f64, GgenError> {
472        // Mock implementation - return 1GB available
473        Ok(1024.0)
474    }
475
476    async fn validate_package(&self, _package_id: &str) -> Result<(), GgenError> {
477        // Mock validation
478        Ok(())
479    }
480
481    async fn verify_package_signatures(&self, _package_id: &PackageId) -> Result<(), GgenError> {
482        // Mock signature verification
483        Ok(())
484    }
485
486    async fn extract_package_files(&self, _package_id: &PackageId) -> Result<(), GgenError> {
487        // Mock extraction
488        Ok(())
489    }
490
491    async fn update_installation_lockfile(&self, _pack_id: &str) -> Result<(), GgenError> {
492        // Mock lockfile update
493        Ok(())
494    }
495
496    async fn cache_package(
497        &self, _package_id: &PackageId, _version: &PackageVersion, _package: Package,
498    ) -> Result<CachedPack, GgenError> {
499        // Mock caching
500        Err(GgenError::ValidationError(
501            "Mock implementation - needs real caching".to_string(),
502        ))
503    }
504
505    fn map_repository_error(&self, error: MarketplaceError, operation: &str) -> GgenError {
506        match error {
507            MarketplaceError::PackageNotFound { package_id } => GgenError::ValidationError(
508                format!("Package '{}' not found in marketplace", package_id),
509            ),
510            MarketplaceError::IoError(e) => {
511                GgenError::NetworkError(format!("Network/IO error while {}: {}", operation, e))
512            }
513            MarketplaceError::ValidationFailed { reason } => GgenError::ValidationError(format!(
514                "Validation error while {}: {}",
515                operation, reason
516            )),
517            _ => GgenError::FileError(format!("Failed to {}: {}", operation, error)),
518        }
519    }
520}
521
522/// Mock repository for testing
523struct TestRepository {}
524
525#[async_trait]
526impl AsyncRepository for TestRepository {
527    type PackageIterator = std::vec::IntoIter<Package>;
528
529    async fn get_package(&self, package_id: &PackageId) -> Result<Package, MarketplaceError> {
530        let version = PackageVersion::new("1.0.0").unwrap();
531        let metadata = ggen_core::marketplace::PackageMetadata::new(
532            package_id.clone(),
533            format!("Pack {}", package_id),
534            "Mock description",
535            "MIT",
536        );
537        Ok(Package {
538            metadata,
539            latest_version: version.clone(),
540            versions: vec![version],
541            releases: indexmap::IndexMap::new(),
542        })
543    }
544
545    async fn get_package_version(
546        &self, package_id: &PackageId, version: &PackageVersion,
547    ) -> Result<Package, MarketplaceError> {
548        let metadata = ggen_core::marketplace::PackageMetadata::new(
549            package_id.clone(),
550            format!("Pack {}", package_id),
551            "Mock description",
552            "MIT",
553        );
554        Ok(Package {
555            metadata,
556            latest_version: version.clone(),
557            versions: vec![version.clone()],
558            releases: indexmap::IndexMap::new(),
559        })
560    }
561
562    async fn all_packages(&self) -> Result<Vec<Package>, MarketplaceError> {
563        let id = PackageId::new("test-pack").unwrap();
564        Ok(vec![self.get_package(&id).await?])
565    }
566
567    async fn list_versions(
568        &self, _id: &PackageId,
569    ) -> Result<Vec<PackageVersion>, MarketplaceError> {
570        Ok(vec![PackageVersion::new("1.0.0").unwrap()])
571    }
572
573    async fn package_exists(&self, _id: &PackageId) -> Result<bool, MarketplaceError> {
574        Ok(true)
575    }
576}
577
578/// Installation result
579#[derive(Debug, Clone, serde::Serialize)]
580pub enum InstallResult {
581    Success(InstallationResult),
582    DryRun(InstallationPlan),
583}
584
585#[derive(Debug, Clone, serde::Serialize)]
586pub struct InstallationResult {
587    pub pack_id: String,
588    pub installed_packages: Vec<PackageId>,
589    pub cache_status: CacheStatus,
590    pub total_size_mb: f64,
591    pub duration_ms: u64,
592}
593
594impl From<InstallResult> for crate::cmds::pack::InstallOutput {
595    fn from(result: InstallResult) -> Self {
596        match result {
597            InstallResult::Success(success) => crate::cmds::pack::InstallOutput {
598                pack_id: success.pack_id.clone(),
599                pack_name: success.pack_id.clone(),
600                status: "installed".to_string(),
601                message: format!(
602                    "Pack installed successfully. Size: {:.1} MB, Duration: {}ms",
603                    success.total_size_mb, success.duration_ms
604                ),
605            },
606            InstallResult::DryRun(plan) => crate::cmds::pack::InstallOutput {
607                pack_id: plan.pack_id.clone(),
608                pack_name: plan.pack_id.clone(),
609                status: "dry_run".to_string(),
610                message: format!(
611                    "Dry run: Would install {:.1} MB with {} dependencies. Estimated: {}s",
612                    plan.total_size_mb, plan.total_dependencies, plan.estimated_duration_seconds
613                ),
614            },
615        }
616    }
617}
618
619#[cfg(test)]
620mod tests {
621    use super::*;
622
623    #[tokio::test]
624    async fn test_pack_installation_plan_creation() {
625        let installer = PackInstaller::new().await.unwrap();
626        let plan = installer
627            .create_installation_plan("test-pack")
628            .await
629            .unwrap();
630
631        assert_eq!(plan.pack_id, "test-pack");
632        assert!(plan.total_size_mb > 0.0);
633        assert!(!plan.steps.is_empty());
634        assert!(plan.estimated_duration_seconds > 0);
635    }
636
637    #[tokio::test]
638    async fn test_dry_run_installation() {
639        let installer = PackInstaller::new().await.unwrap();
640        let result = installer
641            .install_pack("test-pack", false, true)
642            .await
643            .unwrap();
644
645        match result {
646            InstallResult::DryRun(plan) => {
647                assert_eq!(plan.pack_id, "test-pack");
648            }
649            InstallResult::Success(_) => panic!("Expected dry run result"),
650        }
651    }
652}