1use std::path::Path;
5
6use super::metadata::ParserMetadata;
7use crate::models::{DatasourceId, Dependency, PackageData, PackageType};
8use crate::parser_warn as warn;
9use crate::parsers::utils::{MAX_ITERATION_COUNT, read_file_to_string, truncate_field};
10use packageurl::PackageUrl;
11
12use super::PackageParser;
13
14pub struct CarthageCartfileParser;
15
16impl PackageParser for CarthageCartfileParser {
17 const PACKAGE_TYPE: PackageType = PackageType::Carthage;
18
19 fn metadata() -> Vec<ParserMetadata> {
20 vec![ParserMetadata {
21 description: "Carthage Cartfile dependency manifest",
22 file_patterns: &["**/Cartfile", "**/Cartfile.private"],
23 package_type: "carthage",
24 primary_language: "Objective-C",
25 documentation_url: Some(
26 "https://github.com/Carthage/Carthage/blob/master/Documentation/Artifacts.md",
27 ),
28 }]
29 }
30
31 fn extract_packages(path: &Path) -> Vec<PackageData> {
32 let is_private = is_private_cartfile_path(path);
33 let content = match read_file_to_string(path, None) {
34 Ok(c) => c,
35 Err(e) => {
36 warn!("Failed to read Cartfile at {:?}: {}", path, e);
37 return vec![default_cartfile_package_data(is_private)];
38 }
39 };
40
41 let dependencies = parse_cartfile_lines(&content, false);
42
43 vec![PackageData {
44 package_type: Some(Self::PACKAGE_TYPE),
45 primary_language: Some("Objective-C".to_string()),
46 is_private,
47 dependencies,
48 datasource_id: Some(DatasourceId::CarthageCartfile),
49 ..Default::default()
50 }]
51 }
52
53 fn is_match(path: &Path) -> bool {
54 path.file_name()
55 .is_some_and(|name| name == "Cartfile" || name == "Cartfile.private")
56 }
57}
58
59pub struct CarthageCartfileResolvedParser;
60
61impl PackageParser for CarthageCartfileResolvedParser {
62 const PACKAGE_TYPE: PackageType = PackageType::Carthage;
63
64 fn metadata() -> Vec<ParserMetadata> {
65 vec![ParserMetadata {
66 description: "Carthage Cartfile.resolved pinned dependencies",
67 file_patterns: &["**/Cartfile.resolved"],
68 package_type: "carthage",
69 primary_language: "Objective-C",
70 documentation_url: Some(
71 "https://github.com/Carthage/Carthage/blob/master/Documentation/Artifacts.md",
72 ),
73 }]
74 }
75
76 fn extract_packages(path: &Path) -> Vec<PackageData> {
77 let content = match read_file_to_string(path, None) {
78 Ok(c) => c,
79 Err(e) => {
80 warn!("Failed to read Cartfile.resolved at {:?}: {}", path, e);
81 return vec![default_cartfile_resolved_package_data()];
82 }
83 };
84
85 let dependencies = parse_cartfile_lines(&content, true);
86
87 vec![PackageData {
88 package_type: Some(Self::PACKAGE_TYPE),
89 primary_language: Some("Objective-C".to_string()),
90 dependencies,
91 datasource_id: Some(DatasourceId::CarthageCartfileResolved),
92 ..Default::default()
93 }]
94 }
95
96 fn is_match(path: &Path) -> bool {
97 path.file_name()
98 .is_some_and(|name| name == "Cartfile.resolved")
99 }
100}
101
102#[derive(Debug, PartialEq)]
103enum OriginType {
104 Github,
105 Git,
106 Binary,
107}
108
109struct ParsedLine {
110 origin: OriginType,
111 source: String,
112 version_spec: Option<String>,
113}
114
115fn parse_cartfile_lines(content: &str, is_resolved: bool) -> Vec<Dependency> {
116 let mut dependencies = Vec::new();
117
118 for line in content.lines().take(MAX_ITERATION_COUNT) {
119 let line = line.trim();
120 if line.is_empty() || line.starts_with('#') {
121 continue;
122 }
123
124 let Some(parsed) = parse_line(line) else {
125 warn!("Failed to parse Cartfile line: {}", line);
126 continue;
127 };
128
129 let purl_version = if is_resolved {
130 parsed.version_spec.as_deref()
131 } else {
132 None
133 };
134
135 let (purl, name) = match parsed.origin {
136 OriginType::Github => make_github_purl(&parsed.source, purl_version),
137 OriginType::Git => make_git_dep_info(&parsed.source),
138 OriginType::Binary => make_binary_dep_info(&parsed.source),
139 };
140
141 let extracted_requirement = parsed.version_spec.map(truncate_field);
142
143 let is_pinned = if is_resolved { Some(true) } else { None };
144
145 dependencies.push(Dependency {
146 purl: purl.map(truncate_field),
147 extracted_requirement,
148 scope: Some("dependencies".to_string()),
149 is_runtime: None,
150 is_optional: None,
151 is_pinned,
152 is_direct: Some(true),
153 resolved_package: None,
154 extra_data: name.map(|n| {
155 let mut map = std::collections::HashMap::new();
156 map.insert("name".to_string(), serde_json::json!(n));
157 map
158 }),
159 });
160 }
161
162 dependencies
163}
164
165fn parse_line(line: &str) -> Option<ParsedLine> {
166 let (origin, rest) = if let Some(rest) = line.strip_prefix("github") {
167 (OriginType::Github, rest.trim())
168 } else if let Some(rest) = line.strip_prefix("git") {
169 (OriginType::Git, rest.trim())
170 } else if let Some(rest) = line.strip_prefix("binary") {
171 (OriginType::Binary, rest.trim())
172 } else {
173 return None;
174 };
175
176 let (source, remaining) = extract_quoted_string(rest)?;
177
178 let version_spec = extract_version_spec(remaining.trim());
179
180 Some(ParsedLine {
181 origin,
182 source,
183 version_spec,
184 })
185}
186
187fn extract_quoted_string(s: &str) -> Option<(String, &str)> {
188 let s = s.trim();
189 if !s.starts_with('"') {
190 return None;
191 }
192 let rest = &s[1..];
193 let end = rest.find('"')?;
194 Some((rest[..end].to_string(), &rest[end + 1..]))
195}
196
197fn extract_version_spec(s: &str) -> Option<String> {
198 let s = strip_inline_comment(s.trim());
199 if s.is_empty() || s.starts_with('#') {
200 return None;
201 }
202
203 let spec = if let Some(rest) = s.strip_prefix("~>") {
204 format!("~> {}", rest.trim())
205 } else if let Some(rest) = s.strip_prefix(">=") {
206 format!(">= {}", rest.trim())
207 } else if let Some(rest) = s.strip_prefix("==") {
208 format!("== {}", rest.trim())
209 } else if s.starts_with('"') {
210 let (version, _) = extract_quoted_string(s)?;
211 version
212 } else {
213 s.to_string()
214 };
215
216 if spec.is_empty() { None } else { Some(spec) }
217}
218
219fn strip_inline_comment(s: &str) -> &str {
220 s.find('#').map_or(s, |i| s[..i].trim_end())
221}
222
223fn make_github_purl(source: &str, version: Option<&str>) -> (Option<String>, Option<String>) {
224 let parts: Vec<&str> = source.splitn(2, '/').collect();
225 if parts.len() != 2 {
226 warn!("Invalid GitHub source in Cartfile: {}", source);
227 return (None, Some(source.to_string()));
228 }
229
230 let namespace = parts[0];
231 let name = parts[1];
232
233 let purl = match PackageUrl::new("github", name) {
234 Ok(mut p) => {
235 if let Err(e) = p.with_namespace(namespace) {
236 warn!(
237 "Failed to set namespace for github purl '{}': {}",
238 source, e
239 );
240 return (None, Some(name.to_string()));
241 }
242 if let Some(v) = version
243 && let Err(e) = p.with_version(v)
244 {
245 warn!(
246 "Failed to set version '{}' for github purl '{}': {}",
247 v, source, e
248 );
249 }
250 Some(p.to_string())
251 }
252 Err(e) => {
253 warn!("Failed to create PackageUrl for github '{}': {}", source, e);
254 None
255 }
256 };
257
258 (purl, Some(name.to_string()))
259}
260
261fn make_git_dep_info(source: &str) -> (Option<String>, Option<String>) {
262 let name = source
263 .rsplit('/')
264 .next()
265 .map(|s| s.strip_suffix(".git").unwrap_or(s))
266 .filter(|s| !s.is_empty())
267 .map(String::from);
268
269 (None, name)
270}
271
272fn make_binary_dep_info(source: &str) -> (Option<String>, Option<String>) {
273 let name = source
274 .rsplit('/')
275 .next()
276 .and_then(|s| s.strip_suffix(".json"))
277 .filter(|s| !s.is_empty())
278 .map(String::from);
279
280 (None, name)
281}
282
283fn is_private_cartfile_path(path: &Path) -> bool {
284 path.file_name()
285 .is_some_and(|name| name == "Cartfile.private")
286}
287
288fn default_cartfile_package_data(is_private: bool) -> PackageData {
289 PackageData {
290 package_type: Some(PackageType::Carthage),
291 primary_language: Some("Objective-C".to_string()),
292 is_private,
293 datasource_id: Some(DatasourceId::CarthageCartfile),
294 ..Default::default()
295 }
296}
297
298fn default_cartfile_resolved_package_data() -> PackageData {
299 PackageData {
300 package_type: Some(PackageType::Carthage),
301 primary_language: Some("Objective-C".to_string()),
302 datasource_id: Some(DatasourceId::CarthageCartfileResolved),
303 ..Default::default()
304 }
305}