use std::{
fs::read_to_string,
io::{self, Write as _},
path::{Path, PathBuf},
process::{Command, Stdio},
sync::{Arc, RwLock},
};
use ghrepo::GHRepo;
use reqwest::header::CONTENT_TYPE;
use url::Url;
use crate::{OK_END, VERSION, document::Document, error::Error};
#[derive(Clone, Debug, PartialEq)]
pub enum DocumentSource {
File {
path: PathBuf,
basepath: Option<PathBuf>,
},
Stdin {
text: Option<String>,
},
Github {
repo: GHRepo,
branch: String,
},
HyperText {
url: Url,
},
BuiltIn(&'static str),
}
impl DocumentSource {
pub fn return_text(self, returned_text: String) -> Option<Self> {
match self {
DocumentSource::Stdin { .. } => Some(DocumentSource::Stdin {
text: Some(returned_text),
}),
_ => None,
}
}
}
#[derive(Clone)]
pub struct SharedDocumentSource(pub Arc<RwLock<DocumentSource>>);
impl SharedDocumentSource {
#[cfg(test)]
pub fn test() -> SharedDocumentSource {
SharedDocumentSource(Arc::new(RwLock::new(DocumentSource::Stdin { text: None })))
}
pub fn read(&self) -> Result<DocumentSource, Error> {
Ok(self.0.read().map(|g| g.clone())?)
}
pub fn write(&self, source: DocumentSource) -> Result<(), Error> {
let mut inner = self.0.write()?;
*inner = source;
Ok(())
}
}
impl std::ops::Deref for SharedDocumentSource {
type Target = RwLock<DocumentSource>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
pub fn open_source(
source: &str,
url_transform_command: Option<String>,
) -> Result<(String, DocumentSource), Error> {
use ghrepo::GHRepo;
use url::Url;
use core::str::FromStr as _;
if let Some(handle) = source.strip_prefix("github:")
&& let Ok(repo) = GHRepo::from_str(handle)
{
let owner = repo.owner();
let name = repo.name();
let client = reqwest::blocking::Client::builder()
.user_agent(format!(
"mdfried/{}",
VERSION.get().unwrap_or(&"unknown".to_owned())
))
.build()?;
for branch in ["master", "main"] {
let url = format!(
"https://raw.githubusercontent.com/{owner}/{name}/refs/heads/{branch}/README.md"
);
log::info!("trying github URL: {url}");
print!("Fetching URL {url}...");
let response = client.get(&url).send()?;
if response.status().is_success() {
println!("{OK_END}");
return Ok((
response.text()?,
DocumentSource::Github {
repo,
branch: branch.to_owned(),
},
));
} else {
println!("error.");
}
}
return Err(Error::Io(io::Error::other(format!(
"failed to request https://raw.githubusercontent.com/{owner}/{name}/refs/heads/[master|main]/README.md"
))));
} else if let Ok(url) = Url::parse(source) {
log::info!("requesting URL: {url}");
print!("Fetching URL {url}...");
let client = reqwest::blocking::Client::builder()
.user_agent(format!(
"mdfried/{}",
VERSION.get().unwrap_or(&"unknown".to_owned())
))
.build()?;
let response = client.get(url.as_ref()).send()?;
if response.status().is_success() {
println!("{OK_END}");
log::debug!(
"have url_transform_command? {}, content_type: {:?}",
url_transform_command.is_some(),
response.headers().get(CONTENT_TYPE)
);
if let Some(url_transform_command) = url_transform_command
&& let Some(content_type) = response.headers().get(CONTENT_TYPE)
&& content_type
.to_str()
.map_err(|err| {
Error::Io(io::Error::other(format!("content-type error: {err}")))
})?
.starts_with("text/html")
{
let mut child = Command::new("sh")
.arg("-c")
.arg(url_transform_command)
.env("URL", url.as_ref())
.stdin(Stdio::piped())
.stdout(Stdio::piped())
.spawn()?;
let Some(stdin) = child.stdin.as_mut() else {
return Err(Error::Io(io::Error::other(
"url_transform_command pipe error",
)));
};
stdin.write_all(response.text()?.as_bytes())?;
let output = child.wait_with_output()?;
return Ok((
String::from_utf8(output.stdout)
.map_err(|_err| Error::Io(io::Error::other("response not utf-8")))?,
DocumentSource::HyperText { url },
));
}
return Ok((response.text()?, DocumentSource::HyperText { url }));
} else {
println!("error.");
return Err(Error::Io(io::Error::other(format!(
"failed to request {url}"
))));
}
}
let path = PathBuf::from(source);
let basepath = path.parent().map(Path::to_path_buf);
Ok((
read_to_string(&path)?,
DocumentSource::File { path, basepath },
))
}
pub fn github_usercontent_url(repo: &GHRepo, branch: &str, path: &str) -> Result<String, Error> {
let path_url = Url::parse(&format!("https://dummy.com/{}", path))?;
let mut url = Url::parse("https://raw.githubusercontent.com")?;
let segs = path_url.path_segments().ok_or(Error::UrlParse(None))?;
url.path_segments_mut()
.or(Err(Error::UrlParse(None)))?
.extend(&[repo.owner(), repo.name(), branch])
.extend(segs);
url.set_query(path_url.query());
url.set_fragment(path_url.fragment());
Ok(url.to_string())
}
pub fn extend_url(mut url: Url, path: &str) -> Result<String, Error> {
let path_url = Url::parse(&format!("https://dummy.com/{}", path))?;
let segs = path_url.path_segments().ok_or(Error::UrlParse(None))?;
url.path_segments_mut()
.or(Err(Error::UrlParse(None)))?
.extend(segs);
url.set_query(path_url.query());
url.set_fragment(path_url.fragment());
Ok(url.to_string())
}
pub struct DocumentHistoryEntry {
pub source: DocumentSource,
pub document: Document,
pub scroll: u16,
}