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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
use std::convert::TryFrom;
use crate::dockerfile_parser::Instruction;
use crate::error::*;
use crate::parser::{Pair, Rule};
use crate::splicer::Span;
use enquote::unquote;
use snafu::ResultExt;
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct ArgInstruction {
pub span: Span,
pub name: String,
pub name_span: Span,
pub value: Option<String>,
pub value_span: Option<Span>,
}
impl ArgInstruction {
pub(crate) fn from_record(record: Pair) -> Result<ArgInstruction> {
let span = Span::from_pair(&record);
let mut name = None;
let mut value = None;
for field in record.into_inner() {
match field.as_rule() {
Rule::arg_name => name = Some((field.as_str(), Span::from_pair(&field))),
Rule::arg_quoted_value => {
let v = unquote(field.as_str()).context(UnescapeError)?;
value = Some((v, Span::from_pair(&field)));
}
Rule::arg_value => value = Some((
field.as_str().to_string(),
Span::from_pair(&field)
)),
_ => return Err(unexpected_token(field))
}
}
let (name, name_span) = match name {
Some((name, name_span)) => (name.to_string(), name_span),
_ => return Err(Error::GenericParseError {
message: "arg name is required".into()
})
};
let (value, value_span) = match value {
Some((value, value_span)) => (Some(value), Some(value_span)),
None => (None, None)
};
Ok(ArgInstruction {
span,
name,
name_span,
value,
value_span,
})
}
}
impl<'a> TryFrom<&'a Instruction> for &'a ArgInstruction {
type Error = Error;
fn try_from(instruction: &'a Instruction) -> std::result::Result<Self, Self::Error> {
if let Instruction::Arg(a) = instruction {
Ok(a)
} else {
Err(Error::ConversionError {
from: format!("{:?}", instruction),
to: "ArgInstruction".into()
})
}
}
}