1use super::cache;
2use super::config::GitLabConfig;
3use super::provider_trait::{ContextProvider, ProviderParams};
4use super::{ProviderItem, ProviderResult};
5
6const DEFAULT_PER_PAGE: usize = 20;
7const CACHE_TTL_SECS: u64 = 120;
8
9pub fn list_issues(
10 config: &GitLabConfig,
11 state: Option<&str>,
12 labels: Option<&str>,
13 limit: Option<usize>,
14) -> Result<ProviderResult, String> {
15 let project = config
16 .project_path
17 .as_deref()
18 .ok_or("No project path configured. Set CI_PROJECT_PATH or configure git remote.")?;
19 let encoded = urlencoding::encode(project);
20 let per_page = limit.unwrap_or(DEFAULT_PER_PAGE).min(100);
21
22 let mut url =
23 format!("/projects/{encoded}/issues?per_page={per_page}&order_by=updated_at&sort=desc");
24 if let Some(s) = state {
25 url.push_str(&format!("&state={s}"));
26 }
27 if let Some(l) = labels {
28 url.push_str(&format!("&labels={l}"));
29 }
30
31 let cache_key = format!("gitlab:issues:{project}:{state:?}:{labels:?}:{per_page}");
32 if let Some(cached) = cache::get_cached(&cache_key) {
33 if let Ok(result) = serde_json::from_str::<ProviderResult>(&cached) {
34 return Ok(result);
35 }
36 }
37
38 let body = api_get(config, &url)?;
39 let items: Vec<serde_json::Value> =
40 serde_json::from_str(&body).map_err(|e| format!("JSON parse error: {e}"))?;
41
42 let result = ProviderResult {
43 provider: "gitlab".to_string(),
44 resource_type: "issues".to_string(),
45 total_count: None,
46 truncated: items.len() >= per_page,
47 items: items.iter().map(parse_issue).collect(),
48 };
49
50 if let Ok(json) = serde_json::to_string(&result) {
51 cache::set_cached(&cache_key, &json, CACHE_TTL_SECS);
52 }
53 Ok(result)
54}
55
56pub fn show_issue(config: &GitLabConfig, iid: u64) -> Result<ProviderResult, String> {
57 let project = config
58 .project_path
59 .as_deref()
60 .ok_or("No project path configured.")?;
61 let encoded = urlencoding::encode(project);
62 let url = format!("/projects/{encoded}/issues/{iid}");
63
64 let body = api_get(config, &url)?;
65 let issue: serde_json::Value =
66 serde_json::from_str(&body).map_err(|e| format!("JSON parse error: {e}"))?;
67
68 Ok(ProviderResult {
69 provider: "gitlab".to_string(),
70 resource_type: "issue".to_string(),
71 total_count: Some(1),
72 truncated: false,
73 items: vec![parse_issue(&issue)],
74 })
75}
76
77pub fn list_mrs(
78 config: &GitLabConfig,
79 state: Option<&str>,
80 limit: Option<usize>,
81) -> Result<ProviderResult, String> {
82 let project = config
83 .project_path
84 .as_deref()
85 .ok_or("No project path configured.")?;
86 let encoded = urlencoding::encode(project);
87 let per_page = limit.unwrap_or(DEFAULT_PER_PAGE).min(100);
88
89 let mut url = format!(
90 "/projects/{encoded}/merge_requests?per_page={per_page}&order_by=updated_at&sort=desc"
91 );
92 if let Some(s) = state {
93 url.push_str(&format!("&state={s}"));
94 }
95
96 let cache_key = format!("gitlab:mrs:{project}:{state:?}:{per_page}");
97 if let Some(cached) = cache::get_cached(&cache_key) {
98 if let Ok(result) = serde_json::from_str::<ProviderResult>(&cached) {
99 return Ok(result);
100 }
101 }
102
103 let body = api_get(config, &url)?;
104 let items: Vec<serde_json::Value> =
105 serde_json::from_str(&body).map_err(|e| format!("JSON parse error: {e}"))?;
106
107 let result = ProviderResult {
108 provider: "gitlab".to_string(),
109 resource_type: "merge_requests".to_string(),
110 total_count: None,
111 truncated: items.len() >= per_page,
112 items: items.iter().map(parse_mr).collect(),
113 };
114
115 if let Ok(json) = serde_json::to_string(&result) {
116 cache::set_cached(&cache_key, &json, CACHE_TTL_SECS);
117 }
118 Ok(result)
119}
120
121pub fn list_pipelines(
122 config: &GitLabConfig,
123 status: Option<&str>,
124 limit: Option<usize>,
125) -> Result<ProviderResult, String> {
126 let project = config
127 .project_path
128 .as_deref()
129 .ok_or("No project path configured.")?;
130 let encoded = urlencoding::encode(project);
131 let per_page = limit.unwrap_or(DEFAULT_PER_PAGE).min(100);
132
133 let mut url =
134 format!("/projects/{encoded}/pipelines?per_page={per_page}&order_by=updated_at&sort=desc");
135 if let Some(s) = status {
136 url.push_str(&format!("&status={s}"));
137 }
138
139 let body = api_get(config, &url)?;
140 let items: Vec<serde_json::Value> =
141 serde_json::from_str(&body).map_err(|e| format!("JSON parse error: {e}"))?;
142
143 Ok(ProviderResult {
144 provider: "gitlab".to_string(),
145 resource_type: "pipelines".to_string(),
146 total_count: None,
147 truncated: items.len() >= per_page,
148 items: items
149 .iter()
150 .map(|p| ProviderItem {
151 id: p["id"].as_u64().unwrap_or(0).to_string(),
152 title: p["ref"].as_str().unwrap_or("").to_string(),
153 state: p["status"].as_str().map(std::string::ToString::to_string),
154 author: None,
155 created_at: p["created_at"]
156 .as_str()
157 .map(std::string::ToString::to_string),
158 updated_at: p["updated_at"]
159 .as_str()
160 .map(std::string::ToString::to_string),
161 url: p["web_url"].as_str().map(std::string::ToString::to_string),
162 labels: Vec::new(),
163 body: None,
164 ..Default::default()
165 })
166 .collect(),
167 })
168}
169
170pub struct GitLabProvider {
171 config: Result<GitLabConfig, String>,
172}
173
174impl GitLabProvider {
175 pub fn new() -> Self {
176 Self {
177 config: GitLabConfig::from_env(),
178 }
179 }
180}
181
182impl Default for GitLabProvider {
183 fn default() -> Self {
184 Self::new()
185 }
186}
187
188impl ContextProvider for GitLabProvider {
189 fn id(&self) -> &'static str {
190 "gitlab"
191 }
192
193 fn display_name(&self) -> &'static str {
194 "GitLab"
195 }
196
197 fn supported_actions(&self) -> &[&str] {
198 &["issues", "merge_requests", "pipelines"]
199 }
200
201 fn execute(&self, action: &str, params: &ProviderParams) -> Result<ProviderResult, String> {
202 let config = self.config.as_ref().map_err(std::clone::Clone::clone)?;
203 match action {
204 "issues" => list_issues(config, params.state.as_deref(), None, params.limit),
205 "merge_requests" | "mrs" => list_mrs(config, params.state.as_deref(), params.limit),
206 "pipelines" => list_pipelines(config, params.state.as_deref(), params.limit),
207 _ => Err(format!("Unknown GitLab action: {action}")),
208 }
209 }
210
211 fn cache_ttl_secs(&self) -> u64 {
212 CACHE_TTL_SECS
213 }
214
215 fn is_available(&self) -> bool {
216 self.config.is_ok()
217 }
218}
219
220fn api_get(config: &GitLabConfig, endpoint: &str) -> Result<String, String> {
221 let url = config.api_url(endpoint);
222 let response = ureq::get(&url)
223 .header("PRIVATE-TOKEN", &config.token)
224 .call()
225 .map_err(|e| format!("GitLab API error: {e}"))?;
226
227 if response.status() != 200 {
228 return Err(format!("GitLab API returned status {}", response.status()));
229 }
230
231 response
232 .into_body()
233 .read_to_string()
234 .map_err(|e| format!("Failed to read response: {e}"))
235}
236
237fn parse_issue(v: &serde_json::Value) -> ProviderItem {
238 ProviderItem {
239 id: v["iid"].as_u64().unwrap_or(0).to_string(),
240 title: v["title"].as_str().unwrap_or("").to_string(),
241 state: v["state"].as_str().map(std::string::ToString::to_string),
242 author: v["author"]["username"]
243 .as_str()
244 .map(std::string::ToString::to_string),
245 created_at: v["created_at"]
246 .as_str()
247 .map(std::string::ToString::to_string),
248 updated_at: v["updated_at"]
249 .as_str()
250 .map(std::string::ToString::to_string),
251 url: v["web_url"].as_str().map(std::string::ToString::to_string),
252 labels: v["labels"]
253 .as_array()
254 .map(|arr| {
255 arr.iter()
256 .filter_map(|l| l.as_str().map(std::string::ToString::to_string))
257 .collect()
258 })
259 .unwrap_or_default(),
260 body: v["description"]
261 .as_str()
262 .map(std::string::ToString::to_string),
263 ..Default::default()
264 }
265}
266
267fn parse_mr(v: &serde_json::Value) -> ProviderItem {
268 ProviderItem {
269 id: v["iid"].as_u64().unwrap_or(0).to_string(),
270 title: v["title"].as_str().unwrap_or("").to_string(),
271 state: v["state"].as_str().map(std::string::ToString::to_string),
272 author: v["author"]["username"]
273 .as_str()
274 .map(std::string::ToString::to_string),
275 created_at: v["created_at"]
276 .as_str()
277 .map(std::string::ToString::to_string),
278 updated_at: v["updated_at"]
279 .as_str()
280 .map(std::string::ToString::to_string),
281 url: v["web_url"].as_str().map(std::string::ToString::to_string),
282 labels: v["labels"]
283 .as_array()
284 .map(|arr| {
285 arr.iter()
286 .filter_map(|l| l.as_str().map(std::string::ToString::to_string))
287 .collect()
288 })
289 .unwrap_or_default(),
290 body: v["description"]
291 .as_str()
292 .map(std::string::ToString::to_string),
293 ..Default::default()
294 }
295}