use std::path::{Path, PathBuf};
use debian_changelog::ChangeLog;
use debian_control::lossless::Control;
use debian_copyright::lossless::Copyright;
use debian_watch::parse::ParsedWatchFile;
use makefile_lossless::Makefile;
use toml_edit::DocumentMut;
use crate::{Error, Version};
pub trait Editor<T>: std::ops::Deref<Target = T> + std::ops::DerefMut<Target = T> {
fn commit(self: Box<Self>) -> Result<(), Error>;
}
pub trait Workspace {
fn package(&self) -> Option<&str>;
fn current_version(&self) -> Option<&Version>;
fn parsed_control(&self) -> Result<Control, Error>;
fn parsed_changelog(&self) -> Result<ChangeLog, Error>;
fn parsed_copyright(&self) -> Result<Copyright, Error>;
fn parsed_upstream_metadata(&self) -> Result<yaml_edit::YamlFile, Error>;
fn parsed_watch(&self) -> Result<ParsedWatchFile, Error>;
fn parsed_rules(&self) -> Result<Makefile, Error>;
fn source_format(&self) -> Result<Option<String>, Error>;
fn control(&self) -> Result<Box<dyn Editor<Control> + '_>, Error>;
fn changelog(&self) -> Result<Box<dyn Editor<ChangeLog> + '_>, Error>;
fn parsed_debcargo(&self) -> Result<Option<DocumentMut>, Error> {
let rel = Path::new("debian/debcargo.toml");
match self.read_file(rel)? {
None => Ok(None),
Some(bytes) => {
let text = String::from_utf8(bytes.into_owned()).map_err(|e| {
Error::Parse(format!("debcargo.toml is not valid UTF-8: {}", e))
})?;
let doc: DocumentMut = text
.parse()
.map_err(|e| Error::Parse(format!("Failed to parse debcargo.toml: {}", e)))?;
Ok(Some(doc))
}
}
}
fn debcargo(&self) -> Result<Option<Box<dyn Editor<DocumentMut> + '_>>, Error>;
fn read_file(&self, rel: &Path) -> Result<Option<std::borrow::Cow<'_, [u8]>>, Error>;
fn write_file(&self, rel: &Path, content: &[u8]) -> Result<(), Error>;
fn list_dir(&self, rel: &Path) -> Result<Option<Vec<String>>, Error>;
fn walk_dir(&self, rel: &Path) -> Result<Option<Vec<PathBuf>>, Error> {
let Some(top_entries) = self.list_dir(rel)? else {
return Ok(None);
};
let mut out = Vec::new();
let mut stack: Vec<(PathBuf, Vec<String>)> = vec![(rel.to_path_buf(), top_entries)];
while let Some((dir, entries)) = stack.pop() {
for name in entries {
let child = dir.join(&name);
match self.list_dir(&child)? {
Some(sub) => stack.push((child, sub)),
None => out.push(child),
}
}
}
Ok(Some(out))
}
fn file_mode(&self, rel: &Path) -> Result<Option<u32>, Error>;
fn base_path(&self) -> Option<&Path> {
None
}
}
pub fn compat_level(ws: &dyn Workspace) -> Result<Option<u8>, Error> {
if let Some(bytes) = ws.read_file(Path::new("debian/compat"))? {
if let Ok(text) = std::str::from_utf8(&bytes) {
let trimmed = text
.split_once('#')
.map_or(text, |(before, _)| before)
.trim();
if let Ok(level) = trimmed.parse::<u8>() {
return Ok(Some(level));
}
}
}
let control = match ws.parsed_control() {
Ok(c) => c,
Err(Error::NotFound) => return Ok(None),
Err(e) => return Err(e),
};
let Some(source) = control.source() else {
return Ok(None);
};
if let Some(dh_compat) = source.as_deb822().get("X-DH-Compat") {
let trimmed = dh_compat
.split_once('#')
.map_or(dh_compat.as_str(), |(before, _)| before)
.trim();
if let Ok(level) = trimmed.parse::<u8>() {
return Ok(Some(level));
}
}
let Some(build_depends) = source.build_depends() else {
return Ok(None);
};
let Some(rel) = build_depends
.entries()
.flat_map(|entry| entry.relations().collect::<Vec<_>>())
.find(|r| r.try_name().as_deref() == Some("debhelper-compat"))
else {
return Ok(None);
};
Ok(rel
.version()
.and_then(|(_op, v)| v.to_string().parse::<u8>().ok()))
}