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
use nom::character::{is_alphanumeric, is_digit};
use serde::{Deserialize, Serialize};

use std::{fmt, net::Ipv4Addr};

use crate::parse::{parse_ip_address, parse_u16, slice_to_string};

/// Domain address for a URI. Currently Only Ipv4 and
/// domains are implemented.
#[derive(Debug, PartialEq, Clone, Serialize, Deserialize)]
pub enum Domain {
    Ipv4(Ipv4Addr, Option<u16>),
    Domain(String, Option<u16>),
}

impl fmt::Display for Domain {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match self {
            Domain::Ipv4(addr, port) => {
                if let Some(port) = port {
                    write!(f, "{}:{}", addr, port)
                } else {
                    write!(f, "{}", addr)
                }
            },
            Domain::Domain(domain, port) => {
                if let Some(port) = port {
                    write!(f, "{}:{}", domain, port)
                } else {
                    write!(f, "{}", domain)
                }
            },
        }
    }
}

use nom::{
    branch::alt,
    bytes::complete::take_while,
    character::complete::char,
    combinator::{map_res, opt},
    error::ParseError,
    IResult,
};

pub fn parse_port<'a, E: ParseError<&'a [u8]>>(
    input: &'a [u8],
) -> IResult<&'a [u8], Option<u16>, E> {
    let (input, port) = opt(map_res::<_, _, _, _, E, _, _>(
        take_while::<_, _, E>(is_digit),
        parse_u16::<E>,
    ))(input)?;
    Ok((input, port))
}

pub fn parse_domain<'a, E: ParseError<&'a [u8]>>(input: &'a [u8]) -> IResult<&'a [u8], Domain, E> {
    alt((parse_ip_domain::<E>, parse_domain_domain::<E>))(input)
}

pub fn parse_ip_domain<'a, E: ParseError<&'a [u8]>>(
    input: &'a [u8],
) -> IResult<&'a [u8], Domain, E> {
    let (input, addr) = parse_ip_address::<E>(input)?;
    let (input, _) = opt::<_, _, E, _>(char::<_, E>(':'))(input)?;
    let (input, port) = parse_port::<E>(input)?;
    Ok((input, Domain::Ipv4(addr, port)))
}

pub fn parse_domain_domain<'a, E: ParseError<&'a [u8]>>(
    input: &'a [u8],
) -> IResult<&'a [u8], Domain, E> {
    let (input, domain) = map_res::<_, _, _, _, E, _, _>(
        take_while::<_, _, E>(|item| is_alphanumeric(item) || item == b'.'),
        slice_to_string::<E>,
    )(input)?;
    let (input, _) = opt::<_, _, E, _>(char::<_, E>(':'))(input)?;
    let (input, port) = opt::<_, _, E, _>(map_res::<_, _, _, _, E, _, _>(
        take_while::<_, _, E>(is_digit),
        parse_u16::<E>,
    ))(input)?;
    Ok((input, Domain::Domain(domain, port)))
}