natter/src/intermediate/regular_link.rs

111 lines
3.4 KiB
Rust
Raw Normal View History

2023-12-21 14:56:58 -05:00
use organic::types::StandardProperties;
2023-12-23 20:45:20 -05:00
use url::Url;
2023-12-21 14:56:58 -05:00
use super::get_web_path;
use super::macros::intermediate;
2023-10-29 22:31:29 -04:00
2023-10-27 20:43:57 -04:00
use super::IObject;
use crate::context::RenderContext;
use crate::error::CustomError;
2023-10-27 17:48:19 -04:00
2023-10-29 15:36:15 -04:00
#[derive(Debug, Clone)]
2023-10-27 20:43:57 -04:00
pub(crate) struct IRegularLink {
pub(crate) raw_link: String,
2023-10-27 20:43:57 -04:00
pub(crate) children: Vec<IObject>,
pub(crate) target: LinkTarget,
2023-12-21 14:56:58 -05:00
pub(crate) post_blank: organic::types::PostBlank,
2023-10-27 20:43:57 -04:00
}
2023-10-27 17:48:19 -04:00
intermediate!(
IRegularLink,
&'orig organic::types::RegularLink<'parse>,
original,
intermediate_context,
{
let children = {
let mut ret = Vec::new();
for obj in original.children.iter() {
ret.push(IObject::new(intermediate_context.clone(), obj).await?);
}
ret
};
let raw_link = original.get_raw_link();
let target = LinkTarget::from_string(&raw_link)?;
Ok(IRegularLink {
raw_link: raw_link.into_owned(),
children,
target,
2023-12-21 14:56:58 -05:00
post_blank: original.get_post_blank(),
})
}
);
#[derive(Debug, Clone)]
pub(crate) enum LinkTarget {
Raw(String),
2023-12-23 21:29:59 -05:00
Post {
post_id: Option<String>,
subpath: String,
},
}
impl LinkTarget {
pub(crate) fn from_string<S: AsRef<str>>(input: S) -> Result<LinkTarget, CustomError> {
fn impl_from_string(input: &str) -> Result<LinkTarget, CustomError> {
let parsed = Url::parse(input);
if let Err(url::ParseError::RelativeUrlWithoutBase) = parsed {
// For URLs to targets.
// TODO: This shouldn't be raw but instead a variant for targets.
return Ok(LinkTarget::Raw(input.to_owned()));
}
let parsed = parsed?;
match parsed.scheme() {
"post" => {
let post_id = parsed.host_str().map(str::to_owned);
2023-12-23 21:29:59 -05:00
let subpath = {
let subpath = parsed.path();
2023-12-23 21:36:24 -05:00
if let Some(subpath) = subpath.strip_prefix('/') {
subpath
2023-12-23 21:29:59 -05:00
} else {
subpath
}
};
Ok(LinkTarget::Post {
post_id,
subpath: subpath.to_owned(),
})
}
_ => Ok(LinkTarget::Raw(input.to_owned())),
}
}
impl_from_string(input.as_ref())
}
pub(crate) fn generate_final_target(
&self,
render_context: RenderContext<'_>,
) -> Result<Option<String>, CustomError> {
match self {
LinkTarget::Raw(raw_link) => Ok(Some(raw_link.clone())),
2023-12-23 21:29:59 -05:00
LinkTarget::Post { post_id, subpath } => {
let path = post_id
.as_ref()
.map(|post_id| {
2023-12-23 21:29:59 -05:00
let path_to_post = render_context
.config
.get_relative_path_to_post(post_id)
.join(subpath);
get_web_path(
render_context.config,
render_context.output_root_directory,
render_context.output_file,
2023-12-23 21:29:59 -05:00
path_to_post,
)
})
.map_or(Ok(None), |r| r.map(Some))?;
Ok(path)
}
}
}
}