Added rewrite_all for body
This commit is contained in:
parent
64f54e8e58
commit
8f86bac608
|
@ -1,4 +1,4 @@
|
||||||
use tokio::io::AsyncRead;
|
use tokio::io::{AsyncRead, AsyncReadExt};
|
||||||
#[cfg(feature="serve_dir")]
|
#[cfg(feature="serve_dir")]
|
||||||
use tokio::fs::File;
|
use tokio::fs::File;
|
||||||
|
|
||||||
|
@ -11,6 +11,51 @@ pub enum Body {
|
||||||
Reader(Box<dyn AsyncRead + Send + Sync + Unpin>),
|
Reader(Box<dyn AsyncRead + Send + Sync + Unpin>),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Body {
|
||||||
|
/// Called by [`Response::rewrite_all`]
|
||||||
|
pub (crate) async fn rewrite_all(&mut self, based_on: &crate::Request) -> std::io::Result<bool> {
|
||||||
|
let bytes = match self {
|
||||||
|
Self::Bytes(bytes) => {
|
||||||
|
let mut newbytes = Vec::new();
|
||||||
|
std::mem::swap(bytes, &mut newbytes);
|
||||||
|
newbytes
|
||||||
|
}
|
||||||
|
Self::Reader(reader) => {
|
||||||
|
let mut bytes = Vec::new();
|
||||||
|
reader.read_to_end(&mut bytes).await?;
|
||||||
|
bytes
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let mut body = String::from_utf8(bytes).expect("text/gemini wasn't UTF8");
|
||||||
|
|
||||||
|
let mut maybe_indx = if body.starts_with("=> ") {
|
||||||
|
Some(3)
|
||||||
|
} else {
|
||||||
|
body.find("\n=> ").map(|offset| offset + 4)
|
||||||
|
};
|
||||||
|
while let Some(indx) = maybe_indx {
|
||||||
|
|
||||||
|
// Find the end of the link part
|
||||||
|
let end = (&body[indx..]).find(&[' ', '\n', '\r'][..])
|
||||||
|
.map(|offset| indx + offset )
|
||||||
|
.unwrap_or(body.len());
|
||||||
|
|
||||||
|
// Perform replacement
|
||||||
|
if let Some(replacement) = based_on.rewrite_path(&body[indx..end]) {
|
||||||
|
body.replace_range(indx..end, replacement.as_str());
|
||||||
|
} else {
|
||||||
|
return Ok(false)
|
||||||
|
};
|
||||||
|
|
||||||
|
// Find next match
|
||||||
|
maybe_indx = (&body[indx..]).find("\n=> ").map(|offset| offset + 4 + indx);
|
||||||
|
}
|
||||||
|
|
||||||
|
*self = Self::Bytes(body.into_bytes());
|
||||||
|
Ok(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<D: Borrow<Document>> From<D> for Body {
|
impl<D: Borrow<Document>> From<D> for Body {
|
||||||
fn from(document: D) -> Self {
|
fn from(document: D) -> Self {
|
||||||
Self::from(document.borrow().to_string())
|
Self::from(document.borrow().to_string())
|
||||||
|
|
|
@ -144,6 +144,51 @@ impl Response {
|
||||||
pub const fn is_success(&self) -> bool {
|
pub const fn is_success(&self) -> bool {
|
||||||
self.status == 10
|
self.status == 10
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Rewrite any links in this response based on the path identified by a request
|
||||||
|
///
|
||||||
|
/// For more information about what rewriting a link means, see
|
||||||
|
/// [`Request::rewrite_path`].
|
||||||
|
///
|
||||||
|
/// Currently, this rewrites any links in:
|
||||||
|
/// * SUCCESS (10) requests with a `text/gemini` MIME
|
||||||
|
/// * REDIRECT (3X) requests
|
||||||
|
///
|
||||||
|
/// For all other responses, and for any responses without links, this method has no
|
||||||
|
/// effect.
|
||||||
|
///
|
||||||
|
/// If this response contains a reader-based body, this **MAY** load the reader's
|
||||||
|
/// contents into memory if the mime is "text/gemini". If an IO error occurs during
|
||||||
|
/// this process, this error will be raised
|
||||||
|
///
|
||||||
|
/// If the request does not contain enough information to rewrite a link (in other
|
||||||
|
/// words, if [`Requet::rewrite_path`] returns [`None`]), then false is returned. In
|
||||||
|
/// all other cases, this method returns true.
|
||||||
|
///
|
||||||
|
/// Panics if a "text/gemini" response is not UTF-8 formatted
|
||||||
|
pub async fn rewrite_all(&mut self, based_on: &crate::Request) -> std::io::Result<bool> {
|
||||||
|
#[cfg(feature = "scgi_srv")]
|
||||||
|
match self.status {
|
||||||
|
20 if self.meta == "text/gemini" => {
|
||||||
|
if let Some(body) = self.body.as_mut() {
|
||||||
|
body.rewrite_all(based_on).await
|
||||||
|
} else {
|
||||||
|
Ok(false)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
30 | 31 => {
|
||||||
|
if let Some(path) = based_on.rewrite_path(&self.meta) {
|
||||||
|
self.meta = path;
|
||||||
|
Ok(true)
|
||||||
|
} else {
|
||||||
|
Ok(false)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => Ok(true),
|
||||||
|
}
|
||||||
|
#[cfg(feature = "gemini_srv")]
|
||||||
|
Ok(true)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AsRef<Option<Body>> for Response {
|
impl AsRef<Option<Body>> for Response {
|
||||||
|
|
Loading…
Reference in a new issue