1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
use std::io;
use bstr::{BStr, BString};
use quick_error::quick_error;
use crate::mutable::{encode, NL};
quick_error! {
#[derive(Debug)]
pub enum Error {
StartsWithDash {
display("Tags must not start with a dash: '-'")
}
InvalidRefName(err: git_validate::tag::name::Error) {
display("The tag name was no valid reference name")
from()
source(err)
}
}
}
impl From<Error> for io::Error {
fn from(err: Error) -> Self {
io::Error::new(io::ErrorKind::Other, err)
}
}
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
pub struct Tag {
pub target: git_hash::ObjectId,
pub target_kind: crate::Kind,
pub name: BString,
pub message: BString,
pub signature: Option<git_actor::Signature>,
pub pgp_signature: Option<BString>,
}
impl Tag {
pub fn write_to(&self, mut out: impl io::Write) -> io::Result<()> {
encode::trusted_header_id(b"object", &self.target, &mut out)?;
encode::trusted_header_field(b"type", self.target_kind.as_bytes(), &mut out)?;
encode::header_field(b"tag", validated_name(self.name.as_ref())?, &mut out)?;
if let Some(tagger) = &self.signature {
encode::trusted_header_signature(b"tagger", tagger, &mut out)?;
}
if !self.message.is_empty() {
out.write_all(NL)?;
out.write_all(&self.message)?;
}
if let Some(ref message) = self.pgp_signature {
out.write_all(NL)?;
out.write_all(message)?;
}
Ok(())
}
}
fn validated_name(name: &BStr) -> Result<&BStr, Error> {
git_validate::tag::name(name)?;
if name[0] == b'-' {
return Err(Error::StartsWithDash);
}
Ok(name)
}
#[cfg(test)]
mod tests;