1use crate::errors::ParseError;
6use crate::parse::{PackageName, PackageVariant};
7use crate::{AbsolutePackageUrl, RepositoryUrl, UnpinnedAbsolutePackageUrl};
8use fuchsia_hash::Hash;
9
10#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
18pub struct PinnedAbsolutePackageUrl {
19 unpinned: UnpinnedAbsolutePackageUrl,
20 hash: Hash,
21}
22
23impl PinnedAbsolutePackageUrl {
24 pub fn new(
26 repo: RepositoryUrl,
27 name: PackageName,
28 variant: Option<PackageVariant>,
29 hash: Hash,
30 ) -> Self {
31 Self { unpinned: UnpinnedAbsolutePackageUrl::new(repo, name, variant), hash }
32 }
33
34 pub fn new_with_path(repo: RepositoryUrl, path: &str, hash: Hash) -> Result<Self, ParseError> {
37 Ok(Self { unpinned: UnpinnedAbsolutePackageUrl::new_with_path(repo, path)?, hash })
38 }
39
40 pub fn parse(url: &str) -> Result<Self, ParseError> {
42 match AbsolutePackageUrl::parse(url)? {
43 AbsolutePackageUrl::Unpinned(_) => Err(ParseError::MissingHash),
44 AbsolutePackageUrl::Pinned(pinned) => Ok(pinned),
45 }
46 }
47
48 pub fn from_unpinned(unpinned: UnpinnedAbsolutePackageUrl, hash: Hash) -> Self {
50 Self { unpinned, hash }
51 }
52
53 pub fn into_unpinned_and_hash(self) -> (UnpinnedAbsolutePackageUrl, Hash) {
55 let Self { unpinned, hash } = self;
56 (unpinned, hash)
57 }
58
59 pub fn as_unpinned(&self) -> &UnpinnedAbsolutePackageUrl {
61 &self.unpinned
62 }
63
64 pub fn hash(&self) -> Hash {
66 self.hash
67 }
68
69 pub fn set_repository(&mut self, repository: RepositoryUrl) -> &mut Self {
71 self.unpinned.set_repository(repository);
72 self
73 }
74}
75
76impl std::ops::Deref for PinnedAbsolutePackageUrl {
79 type Target = UnpinnedAbsolutePackageUrl;
80
81 fn deref(&self) -> &Self::Target {
82 &self.unpinned
83 }
84}
85
86impl std::ops::DerefMut for PinnedAbsolutePackageUrl {
89 fn deref_mut(&mut self) -> &mut Self::Target {
90 &mut self.unpinned
91 }
92}
93
94impl std::str::FromStr for PinnedAbsolutePackageUrl {
95 type Err = ParseError;
96
97 fn from_str(url: &str) -> Result<Self, Self::Err> {
98 Self::parse(url)
99 }
100}
101
102impl std::convert::TryFrom<&str> for PinnedAbsolutePackageUrl {
103 type Error = ParseError;
104
105 fn try_from(value: &str) -> Result<Self, Self::Error> {
106 Self::parse(value)
107 }
108}
109
110impl std::fmt::Display for PinnedAbsolutePackageUrl {
111 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
112 write!(f, "{}?hash={}", self.unpinned, self.hash)
113 }
114}
115
116impl serde::Serialize for PinnedAbsolutePackageUrl {
117 fn serialize<S: serde::Serializer>(&self, ser: S) -> Result<S::Ok, S::Error> {
118 self.to_string().serialize(ser)
119 }
120}
121
122impl<'de> serde::Deserialize<'de> for PinnedAbsolutePackageUrl {
123 fn deserialize<D>(de: D) -> Result<Self, D::Error>
124 where
125 D: serde::Deserializer<'de>,
126 {
127 let url = String::deserialize(de)?;
128 Ok(Self::parse(&url).map_err(|err| serde::de::Error::custom(err))?)
129 }
130}
131
132#[cfg(test)]
133mod tests {
134 use super::*;
135 use crate::errors::PackagePathSegmentError;
136 use assert_matches::assert_matches;
137 use std::convert::TryFrom as _;
138
139 #[test]
140 fn parse_err() {
141 for (url, err) in [
142 ("fuchsia-boot://example.org/name?hash=0000000000000000000000000000000000000000000000000000000000000000", ParseError::InvalidScheme),
143 ("fuchsia-pkg://?hash=0000000000000000000000000000000000000000000000000000000000000000", ParseError::MissingHost),
144 ("fuchsia-pkg://exaMple.org?hash=0000000000000000000000000000000000000000000000000000000000000000", ParseError::InvalidHost),
145 ("fuchsia-pkg://example.org/?hash=0000000000000000000000000000000000000000000000000000000000000000", ParseError::MissingName),
146 (
147 "fuchsia-pkg://example.org//?hash=0000000000000000000000000000000000000000000000000000000000000000",
148 ParseError::InvalidPathSegment(PackagePathSegmentError::Empty),
149 ),
150 ("fuchsia-pkg://example.org/name/variant/extra?hash=0000000000000000000000000000000000000000000000000000000000000000", ParseError::ExtraPathSegments),
151 ("fuchsia-pkg://example.org/name?hash=0000000000000000000000000000000000000000000000000000000000000000#resource", ParseError::CannotContainResource),
152
153 ] {
154 assert_matches!(
155 PinnedAbsolutePackageUrl::parse(url),
156 Err(e) if e == err,
157 "the url {:?}", url
158 );
159 assert_matches!(
160 url.parse::<PinnedAbsolutePackageUrl>(),
161 Err(e) if e == err,
162 "the url {:?}", url
163 );
164 assert_matches!(
165 PinnedAbsolutePackageUrl::try_from(url),
166 Err(e) if e == err,
167 "the url {:?}", url
168 );
169 assert_matches!(
170 serde_json::from_str::<PinnedAbsolutePackageUrl>(url),
171 Err(_),
172 "the url {:?}",
173 url
174 );
175 }
176 }
177
178 #[test]
179 fn parse_ok() {
180 for (url, variant, path) in [
181 ("fuchsia-pkg://example.org/name?hash=0000000000000000000000000000000000000000000000000000000000000000", None, "/name"),
182 (
183 "fuchsia-pkg://example.org/name/variant?hash=0000000000000000000000000000000000000000000000000000000000000000",
184 Some("variant"),
185 "/name/variant",
186 ),
187 ] {
188 let json_url = format!("\"{url}\"");
189 let host = "example.org";
190 let name = "name";
191 let hash = "0000000000000000000000000000000000000000000000000000000000000000".parse::<Hash>().unwrap();
192
193 let name = name.parse::<crate::PackageName>().unwrap();
195 let variant = variant.map(|v| v.parse::<crate::PackageVariant>().unwrap());
196 let validate = |parsed: &PinnedAbsolutePackageUrl| {
197 assert_eq!(parsed.host(), host);
198 assert_eq!(parsed.name(), &name);
199 assert_eq!(parsed.variant(), variant.as_ref());
200 assert_eq!(parsed.path(), path);
201 assert_eq!(parsed.hash(), hash);
202 };
203 validate(&PinnedAbsolutePackageUrl::parse(url).unwrap());
204 validate(&url.parse::<PinnedAbsolutePackageUrl>().unwrap());
205 validate(&PinnedAbsolutePackageUrl::try_from(url).unwrap());
206 validate(&serde_json::from_str::<PinnedAbsolutePackageUrl>(&json_url).unwrap());
207
208 assert_eq!(
210 PinnedAbsolutePackageUrl::parse(url).unwrap().to_string(),
211 url,
212 "the url {:?}",
213 url
214 );
215 assert_eq!(
216 serde_json::to_string(&PinnedAbsolutePackageUrl::parse(url).unwrap()).unwrap(),
217 json_url,
218 "the url {:?}",
219 url
220 );
221 }
222 }
223}