lmrc_cli/generator/
mod.rs

1pub mod apps;
2pub mod documentation;
3pub mod gitlab_ci;
4pub mod pipeline;
5pub mod workspace;
6
7use colored::Colorize;
8use lmrc_config_validator::LmrcConfig;
9use std::fs;
10use std::path::{Path, PathBuf};
11
12use crate::error::Result;
13
14pub struct ProjectGenerator {
15    config: LmrcConfig,
16    project_path: PathBuf,
17}
18
19impl ProjectGenerator {
20    pub fn new(config: LmrcConfig, project_path: PathBuf) -> Self {
21        Self {
22            config,
23            project_path,
24        }
25    }
26
27    pub fn project_path(&self) -> &Path {
28        &self.project_path
29    }
30
31    pub async fn generate(&self) -> Result<()> {
32        println!("{}", "Generating project structure...".yellow());
33
34        // Create root directory
35        fs::create_dir_all(&self.project_path)?;
36
37        // Create directory structure
38        self.create_directory_structure()?;
39
40        // Generate configuration file
41        self.generate_config_file()?;
42
43        // Generate applications
44        self.generate_applications()?;
45
46        // Generate libs directory with placeholders
47        self.generate_libs()?;
48
49        // Generate pipeline application
50        pipeline::generate_pipeline_app(&self.project_path, &self.config)?;
51
52        // Generate workspace Cargo.toml AFTER apps and pipeline are created
53        workspace::generate_workspace_toml(&self.project_path, &self.config)?;
54
55        // Generate GitLab CI configuration
56        gitlab_ci::generate_gitlab_ci(&self.project_path, &self.config)?;
57
58        // Generate documentation
59        documentation::generate_docs(&self.project_path, &self.config)?;
60
61        // Generate Docker files
62        self.generate_docker_files()?;
63
64        // Generate .gitignore
65        self.generate_gitignore()?;
66
67        Ok(())
68    }
69
70    pub async fn generate_minimal(&self) -> Result<()> {
71        println!("{}", "Generating minimal project structure...".yellow());
72
73        // Create root directory
74        fs::create_dir_all(&self.project_path)?;
75
76        // Create directory structure
77        self.create_directory_structure()?;
78
79        // Don't generate Cargo.toml for minimal projects - it will be created when first component is added
80
81        // Generate configuration file
82        self.generate_config_file()?;
83
84        // Generate libs directory with placeholders
85        self.generate_libs()?;
86
87        // Generate Docker files
88        self.generate_docker_files()?;
89
90        // Generate .gitignore
91        self.generate_gitignore()?;
92
93        Ok(())
94    }
95
96    fn create_directory_structure(&self) -> Result<()> {
97        let dirs = vec!["apps", "libs", "docs", "docker", "infra"];
98
99        for dir in dirs {
100            let path = self.project_path.join(dir);
101            fs::create_dir_all(&path)?;
102            println!("  {} {}", "Created:".green(), path.display());
103        }
104
105        Ok(())
106    }
107
108    fn generate_config_file(&self) -> Result<()> {
109        let config_content = self.config.to_toml_string()?;
110        let config_path = self.project_path.join("lmrc.toml");
111        fs::write(&config_path, config_content)?;
112        println!("  {} {}", "Created:".green(), config_path.display());
113        Ok(())
114    }
115
116    fn generate_applications(&self) -> Result<()> {
117        // Use the new apps module to generate applications from templates
118        apps::generate_applications(&self.project_path, &self.config)?;
119        Ok(())
120    }
121
122    fn generate_libs(&self) -> Result<()> {
123        let libs_path = self.project_path.join("libs");
124        let readme = r#"# Shared Libraries
125
126This directory contains shared libraries used across multiple applications.
127
128Place reusable code, utilities, and shared logic here.
129"#;
130        fs::write(libs_path.join("README.md"), readme)?;
131        println!("  {} libs/README.md", "Created:".green());
132        Ok(())
133    }
134
135    fn generate_docker_files(&self) -> Result<()> {
136        let docker_path = self.project_path.join("docker");
137
138        // Generate Dockerfile template for applications
139        let dockerfile = r#"# Build stage
140FROM rust:1.75-slim as builder
141
142WORKDIR /app
143COPY . .
144
145RUN cargo build --release
146
147# Runtime stage
148FROM debian:bookworm-slim
149
150RUN apt-get update && \
151    apt-get install -y ca-certificates && \
152    rm -rf /var/lib/apt/lists/*
153
154WORKDIR /app
155
156COPY --from=builder /app/target/release/app /app/app
157
158ENTRYPOINT ["/app/app"]
159"#;
160
161        fs::write(docker_path.join("Dockerfile"), dockerfile)?;
162        println!("  {} docker/Dockerfile", "Created:".green());
163
164        // Generate docker-compose.yml for local development
165        let docker_compose = format!(
166            r#"version: '3.8'
167
168services:
169  postgres:
170    image: postgres:{}
171    environment:
172      POSTGRES_DB: {}
173      POSTGRES_USER: postgres
174      POSTGRES_PASSWORD: postgres
175    ports:
176      - "5432:5432"
177    volumes:
178      - postgres_data:/var/lib/postgresql/data
179
180volumes:
181  postgres_data:
182"#,
183            self.config
184                .infrastructure
185                .postgres
186                .as_ref()
187                .map(|p| p.version.as_str())
188                .unwrap_or("16"),
189            self.config
190                .infrastructure
191                .postgres
192                .as_ref()
193                .map(|p| p.database_name.as_str())
194                .unwrap_or("myapp")
195        );
196
197        fs::write(docker_path.join("docker-compose.yml"), docker_compose)?;
198        println!("  {} docker/docker-compose.yml", "Created:".green());
199
200        Ok(())
201    }
202
203    fn generate_gitignore(&self) -> Result<()> {
204        let gitignore = r#"# Rust
205/target/
206**/*.rs.bk
207*.pdb
208Cargo.lock
209
210# IDE
211.idea/
212.vscode/
213*.swp
214*.swo
215*~
216
217# OS
218.DS_Store
219Thumbs.db
220
221# Environment
222.env
223.env.local
224
225# Secrets
226secrets/
227*.pem
228*.key
229
230# SSH Keys (project-local)
231.ssh/
232*.pub
233"#;
234
235        let gitignore_path = self.project_path.join(".gitignore");
236        fs::write(&gitignore_path, gitignore)?;
237        println!("  {} .gitignore", "Created:".green());
238
239        Ok(())
240    }
241}