fuchsia_url/
pinned_absolute_package_url.rs

1// Copyright 2022 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5use crate::errors::ParseError;
6use crate::parse::{PackageName, PackageVariant};
7use crate::{AbsolutePackageUrl, RepositoryUrl, UnpinnedAbsolutePackageUrl};
8use fuchsia_hash::Hash;
9
10/// A URL locating a Fuchsia package. Must have a hash.
11/// Has the form "fuchsia-pkg://<repository>/<name>[/variant]?hash=<hash>" where:
12///   * "repository" is a valid hostname
13///   * "name" is a valid package name
14///   * "variant" is an optional valid package variant
15///   * "hash" is a valid package hash
16/// https://fuchsia.dev/fuchsia-src/concepts/packages/package_url
17#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
18pub struct PinnedAbsolutePackageUrl {
19    unpinned: UnpinnedAbsolutePackageUrl,
20    hash: Hash,
21}
22
23impl PinnedAbsolutePackageUrl {
24    /// Create a `PinnedAbsolutePackageUrl` from its component parts.
25    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    /// Create a PinnedAbsolutePackageUrl from its component parts and a &str `path` that will be
35    /// validated.
36    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    /// Parse a "fuchsia-pkg://" URL that locates a pinned (has a hash query parameter) package.
41    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    /// Create a `PinnedAbsolutePackageUrl` from an unpinned url and a hash.
49    pub fn from_unpinned(unpinned: UnpinnedAbsolutePackageUrl, hash: Hash) -> Self {
50        Self { unpinned, hash }
51    }
52
53    /// Split this URL into an unpinned URL and hash.
54    pub fn into_unpinned_and_hash(self) -> (UnpinnedAbsolutePackageUrl, Hash) {
55        let Self { unpinned, hash } = self;
56        (unpinned, hash)
57    }
58
59    /// The URL without the hash.
60    pub fn as_unpinned(&self) -> &UnpinnedAbsolutePackageUrl {
61        &self.unpinned
62    }
63
64    /// The URL's hash.
65    pub fn hash(&self) -> Hash {
66        self.hash
67    }
68
69    /// Change the repository to `repository`.
70    pub fn set_repository(&mut self, repository: RepositoryUrl) -> &mut Self {
71        self.unpinned.set_repository(repository);
72        self
73    }
74}
75
76// PinnedAbsolutePackageUrl does not maintain any invariants on its `unpinned` field in addition to
77// those already maintained by UnpinnedAbsolutePackageUrl so this is safe.
78impl std::ops::Deref for PinnedAbsolutePackageUrl {
79    type Target = UnpinnedAbsolutePackageUrl;
80
81    fn deref(&self) -> &Self::Target {
82        &self.unpinned
83    }
84}
85
86// PinnedAbsolutePackageUrl does not maintain any invariants on its `unpinned` field in addition to
87// those already maintained by UnpinnedAbsolutePackageUrl so this is safe.
88impl 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            // Creation
194            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            // Stringification
209            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}