parsers.rs 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674
  1. #![cfg_attr(not(all(feature = "proto-ipv6", feature = "proto-ipv4")), allow(dead_code))]
  2. use core::str::FromStr;
  3. use core::result;
  4. #[cfg(feature = "ethernet")]
  5. use crate::wire::EthernetAddress;
  6. use crate::wire::{IpAddress, IpCidr, IpEndpoint};
  7. #[cfg(feature = "proto-ipv4")]
  8. use crate::wire::{Ipv4Address, Ipv4Cidr};
  9. #[cfg(feature = "proto-ipv6")]
  10. use crate::wire::{Ipv6Address, Ipv6Cidr};
  11. type Result<T> = result::Result<T, ()>;
  12. struct Parser<'a> {
  13. data: &'a [u8],
  14. pos: usize
  15. }
  16. impl<'a> Parser<'a> {
  17. fn new(data: &'a str) -> Parser<'a> {
  18. Parser {
  19. data: data.as_bytes(),
  20. pos: 0
  21. }
  22. }
  23. fn lookahead_char(&self, ch: u8) -> bool {
  24. if self.pos < self.data.len() {
  25. self.data[self.pos] == ch
  26. } else {
  27. false
  28. }
  29. }
  30. fn advance(&mut self) -> Result<u8> {
  31. match self.data.get(self.pos) {
  32. Some(&chr) => {
  33. self.pos += 1;
  34. Ok(chr)
  35. }
  36. None => Err(())
  37. }
  38. }
  39. fn try_do<F, T>(&mut self, f: F) -> Option<T>
  40. where F: FnOnce(&mut Parser<'a>) -> Result<T> {
  41. let pos = self.pos;
  42. match f(self) {
  43. Ok(res) => Some(res),
  44. Err(()) => {
  45. self.pos = pos;
  46. None
  47. }
  48. }
  49. }
  50. fn accept_eof(&mut self) -> Result<()> {
  51. if self.data.len() == self.pos {
  52. Ok(())
  53. } else {
  54. Err(())
  55. }
  56. }
  57. fn until_eof<F, T>(&mut self, f: F) -> Result<T>
  58. where F: FnOnce(&mut Parser<'a>) -> Result<T> {
  59. let res = f(self)?;
  60. self.accept_eof()?;
  61. Ok(res)
  62. }
  63. fn accept_char(&mut self, chr: u8) -> Result<()> {
  64. if self.advance()? == chr {
  65. Ok(())
  66. } else {
  67. Err(())
  68. }
  69. }
  70. fn accept_str(&mut self, string: &[u8]) -> Result<()> {
  71. for byte in string.iter() {
  72. self.accept_char(*byte)?;
  73. }
  74. Ok(())
  75. }
  76. fn accept_digit(&mut self, hex: bool) -> Result<u8> {
  77. let digit = self.advance()?;
  78. if (b'0'..=b'9').contains(&digit) {
  79. Ok(digit - b'0')
  80. } else if hex && (b'a'..=b'f').contains(&digit) {
  81. Ok(digit - b'a' + 10)
  82. } else if hex && (b'A'..=b'F').contains(&digit) {
  83. Ok(digit - b'A' + 10)
  84. } else {
  85. Err(())
  86. }
  87. }
  88. fn accept_number(&mut self, max_digits: usize, max_value: u32,
  89. hex: bool) -> Result<u32> {
  90. let mut value = self.accept_digit(hex)? as u32;
  91. for _ in 1..max_digits {
  92. match self.try_do(|p| p.accept_digit(hex)) {
  93. Some(digit) => {
  94. value *= if hex { 16 } else { 10 };
  95. value += digit as u32;
  96. }
  97. None => break
  98. }
  99. }
  100. if value < max_value {
  101. Ok(value)
  102. } else {
  103. Err(())
  104. }
  105. }
  106. #[cfg(feature = "ethernet")]
  107. fn accept_mac_joined_with(&mut self, separator: u8) -> Result<EthernetAddress> {
  108. let mut octets = [0u8; 6];
  109. for (n, octet) in octets.iter_mut().enumerate() {
  110. *octet = self.accept_number(2, 0x100, true)? as u8;
  111. if n != 5 {
  112. self.accept_char(separator)?;
  113. }
  114. }
  115. Ok(EthernetAddress(octets))
  116. }
  117. #[cfg(feature = "ethernet")]
  118. fn accept_mac(&mut self) -> Result<EthernetAddress> {
  119. if let Some(mac) = self.try_do(|p| p.accept_mac_joined_with(b'-')) {
  120. return Ok(mac)
  121. }
  122. if let Some(mac) = self.try_do(|p| p.accept_mac_joined_with(b':')) {
  123. return Ok(mac)
  124. }
  125. Err(())
  126. }
  127. #[cfg(feature = "proto-ipv6")]
  128. fn accept_ipv4_mapped_ipv6_part(&mut self, parts: &mut [u16], idx: &mut usize) -> Result<()> {
  129. let octets = self.accept_ipv4_octets()?;
  130. parts[*idx] = ((octets[0] as u16) << 8) | (octets[1] as u16);
  131. *idx += 1;
  132. parts[*idx] = ((octets[2] as u16) << 8) | (octets[3] as u16);
  133. *idx += 1;
  134. Ok(())
  135. }
  136. #[cfg(feature = "proto-ipv6")]
  137. fn accept_ipv6_part(&mut self, (head, tail): (&mut [u16; 8], &mut [u16; 6]),
  138. (head_idx, tail_idx): (&mut usize, &mut usize),
  139. mut use_tail: bool, is_cidr: bool) -> Result<()> {
  140. let double_colon = match self.try_do(|p| p.accept_str(b"::")) {
  141. Some(_) if !use_tail && *head_idx < 7 => {
  142. // Found a double colon. Start filling out the
  143. // tail and set the double colon flag in case
  144. // this is the last character we can parse.
  145. use_tail = true;
  146. true
  147. },
  148. Some(_) => {
  149. // This is a bad address. Only one double colon is
  150. // allowed and an address is only 128 bits.
  151. return Err(());
  152. }
  153. None => {
  154. if *head_idx != 0 || use_tail && *tail_idx != 0 {
  155. // If this is not the first number or the position following
  156. // a double colon, we expect there to be a single colon.
  157. self.accept_char(b':')?;
  158. }
  159. false
  160. }
  161. };
  162. match self.try_do(|p| p.accept_number(4, 0x10000, true)) {
  163. Some(part) if !use_tail && *head_idx < 8 => {
  164. // Valid u16 to be added to the address
  165. head[*head_idx] = part as u16;
  166. *head_idx += 1;
  167. if *head_idx == 6 && head[0..*head_idx] == [0, 0, 0, 0, 0, 0xffff] {
  168. self.try_do(|p| {
  169. p.accept_char(b':')?;
  170. p.accept_ipv4_mapped_ipv6_part(head, head_idx)
  171. });
  172. }
  173. Ok(())
  174. },
  175. Some(part) if *tail_idx < 6 => {
  176. // Valid u16 to be added to the address
  177. tail[*tail_idx] = part as u16;
  178. *tail_idx += 1;
  179. if *tail_idx == 1 && tail[0] == 0xffff
  180. && head[0..8] == [0, 0, 0, 0, 0, 0, 0, 0] {
  181. self.try_do(|p| {
  182. p.accept_char(b':')?;
  183. p.accept_ipv4_mapped_ipv6_part(tail, tail_idx)
  184. });
  185. }
  186. Ok(())
  187. },
  188. Some(_) => {
  189. // Tail or head section is too long
  190. Err(())
  191. }
  192. None if double_colon && (is_cidr || self.pos == self.data.len()) => {
  193. // The address ends with "::". E.g. 1234:: or ::
  194. Ok(())
  195. }
  196. None => {
  197. // Invalid address
  198. Err(())
  199. }
  200. }?;
  201. if *head_idx + *tail_idx > 8 {
  202. // The head and tail indexes add up to a bad address length.
  203. Err(())
  204. } else if !self.lookahead_char(b':') {
  205. if *head_idx < 8 && !use_tail {
  206. // There was no double colon found, and the head is too short
  207. return Err(());
  208. }
  209. Ok(())
  210. } else {
  211. // Continue recursing
  212. self.accept_ipv6_part((head, tail), (head_idx, tail_idx), use_tail, is_cidr)
  213. }
  214. }
  215. #[cfg(feature = "proto-ipv6")]
  216. fn accept_ipv6(&mut self, is_cidr: bool) -> Result<Ipv6Address> {
  217. // IPv6 addresses may contain a "::" to indicate a series of
  218. // 16 bit sections that evaluate to 0. E.g.
  219. //
  220. // fe80:0000:0000:0000:0000:0000:0000:0001
  221. //
  222. // May be written as
  223. //
  224. // fe80::1
  225. //
  226. // As a result, we need to find the first section of colon
  227. // delimited u16's before a possible "::", then the
  228. // possible second section after the "::", and finally
  229. // combine the second optional section to the end of the
  230. // final address.
  231. //
  232. // See https://tools.ietf.org/html/rfc4291#section-2.2
  233. // for details.
  234. let (mut addr, mut tail) = ([0u16; 8], [0u16; 6]);
  235. let (mut head_idx, mut tail_idx) = (0, 0);
  236. self.accept_ipv6_part((&mut addr, &mut tail), (&mut head_idx, &mut tail_idx), false, is_cidr)?;
  237. // We need to copy the tail portion (the portion following the "::") to the
  238. // end of the address.
  239. addr[8 - tail_idx..].copy_from_slice(&tail[..tail_idx]);
  240. Ok(Ipv6Address::from_parts(&addr))
  241. }
  242. fn accept_ipv4_octets(&mut self) -> Result<[u8; 4]> {
  243. let mut octets = [0u8; 4];
  244. for (n, octet) in octets.iter_mut().enumerate() {
  245. *octet = self.accept_number(3, 0x100, false)? as u8;
  246. if n != 3 {
  247. self.accept_char(b'.')?;
  248. }
  249. }
  250. Ok(octets)
  251. }
  252. #[cfg(feature = "proto-ipv4")]
  253. fn accept_ipv4(&mut self) -> Result<Ipv4Address> {
  254. let octets = self.accept_ipv4_octets()?;
  255. Ok(Ipv4Address(octets))
  256. }
  257. fn accept_ip(&mut self) -> Result<IpAddress> {
  258. #[cfg(feature = "proto-ipv4")]
  259. #[allow(clippy::single_match)]
  260. match self.try_do(|p| p.accept_ipv4()) {
  261. Some(ipv4) => return Ok(IpAddress::Ipv4(ipv4)),
  262. None => ()
  263. }
  264. #[cfg(feature = "proto-ipv6")]
  265. #[allow(clippy::single_match)]
  266. match self.try_do(|p| p.accept_ipv6(false)) {
  267. Some(ipv6) => return Ok(IpAddress::Ipv6(ipv6)),
  268. None => ()
  269. }
  270. Err(())
  271. }
  272. #[cfg(feature = "proto-ipv4")]
  273. fn accept_ipv4_endpoint(&mut self) -> Result<IpEndpoint> {
  274. let ip = self.accept_ipv4()?;
  275. let port = if self.accept_eof().is_ok() {
  276. 0
  277. } else {
  278. self.accept_char(b':')?;
  279. self.accept_number(5, 65535, false)?
  280. };
  281. Ok(IpEndpoint { addr: IpAddress::Ipv4(ip), port: port as u16 })
  282. }
  283. #[cfg(feature = "proto-ipv6")]
  284. fn accept_ipv6_endpoint(&mut self) -> Result<IpEndpoint> {
  285. if self.lookahead_char(b'[') {
  286. self.accept_char(b'[')?;
  287. let ip = self.accept_ipv6(false)?;
  288. self.accept_char(b']')?;
  289. self.accept_char(b':')?;
  290. let port = self.accept_number(5, 65535, false)?;
  291. Ok(IpEndpoint { addr: IpAddress::Ipv6(ip), port: port as u16 })
  292. } else {
  293. let ip = self.accept_ipv6(false)?;
  294. Ok(IpEndpoint { addr: IpAddress::Ipv6(ip), port: 0 })
  295. }
  296. }
  297. fn accept_ip_endpoint(&mut self) -> Result<IpEndpoint> {
  298. #[cfg(feature = "proto-ipv4")]
  299. #[allow(clippy::single_match)]
  300. match self.try_do(|p| p.accept_ipv4_endpoint()) {
  301. Some(ipv4) => return Ok(ipv4),
  302. None => ()
  303. }
  304. #[cfg(feature = "proto-ipv6")]
  305. #[allow(clippy::single_match)]
  306. match self.try_do(|p| p.accept_ipv6_endpoint()) {
  307. Some(ipv6) => return Ok(ipv6),
  308. None => ()
  309. }
  310. Err(())
  311. }
  312. }
  313. #[cfg(feature = "ethernet")]
  314. impl FromStr for EthernetAddress {
  315. type Err = ();
  316. /// Parse a string representation of an Ethernet address.
  317. fn from_str(s: &str) -> Result<EthernetAddress> {
  318. Parser::new(s).until_eof(|p| p.accept_mac())
  319. }
  320. }
  321. #[cfg(feature = "proto-ipv4")]
  322. impl FromStr for Ipv4Address {
  323. type Err = ();
  324. /// Parse a string representation of an IPv4 address.
  325. fn from_str(s: &str) -> Result<Ipv4Address> {
  326. Parser::new(s).until_eof(|p| p.accept_ipv4())
  327. }
  328. }
  329. #[cfg(feature = "proto-ipv6")]
  330. impl FromStr for Ipv6Address {
  331. type Err = ();
  332. /// Parse a string representation of an IPv6 address.
  333. fn from_str(s: &str) -> Result<Ipv6Address> {
  334. Parser::new(s).until_eof(|p| p.accept_ipv6(false))
  335. }
  336. }
  337. impl FromStr for IpAddress {
  338. type Err = ();
  339. /// Parse a string representation of an IP address.
  340. fn from_str(s: &str) -> Result<IpAddress> {
  341. Parser::new(s).until_eof(|p| p.accept_ip())
  342. }
  343. }
  344. #[cfg(feature = "proto-ipv4")]
  345. impl FromStr for Ipv4Cidr {
  346. type Err = ();
  347. /// Parse a string representation of an IPv4 CIDR.
  348. fn from_str(s: &str) -> Result<Ipv4Cidr> {
  349. Parser::new(s).until_eof(|p| {
  350. let ip = p.accept_ipv4()?;
  351. p.accept_char(b'/')?;
  352. let prefix_len = p.accept_number(2, 33, false)? as u8;
  353. Ok(Ipv4Cidr::new(ip, prefix_len))
  354. })
  355. }
  356. }
  357. #[cfg(feature = "proto-ipv6")]
  358. impl FromStr for Ipv6Cidr {
  359. type Err = ();
  360. /// Parse a string representation of an IPv6 CIDR.
  361. fn from_str(s: &str) -> Result<Ipv6Cidr> {
  362. // https://tools.ietf.org/html/rfc4291#section-2.3
  363. Parser::new(s).until_eof(|p| {
  364. let ip = p.accept_ipv6(true)?;
  365. p.accept_char(b'/')?;
  366. let prefix_len = p.accept_number(3, 129, false)? as u8;
  367. Ok(Ipv6Cidr::new(ip, prefix_len))
  368. })
  369. }
  370. }
  371. impl FromStr for IpCidr {
  372. type Err = ();
  373. /// Parse a string representation of an IP CIDR.
  374. fn from_str(s: &str) -> Result<IpCidr> {
  375. #[cfg(feature = "proto-ipv4")]
  376. #[allow(clippy::single_match)]
  377. match Ipv4Cidr::from_str(s) {
  378. Ok(cidr) => return Ok(IpCidr::Ipv4(cidr)),
  379. Err(_) => ()
  380. }
  381. #[cfg(feature = "proto-ipv6")]
  382. #[allow(clippy::single_match)]
  383. match Ipv6Cidr::from_str(s) {
  384. Ok(cidr) => return Ok(IpCidr::Ipv6(cidr)),
  385. Err(_) => ()
  386. }
  387. Err(())
  388. }
  389. }
  390. impl FromStr for IpEndpoint {
  391. type Err = ();
  392. fn from_str(s: &str) -> Result<IpEndpoint> {
  393. Parser::new(s).until_eof(|p| Ok(p.accept_ip_endpoint()?))
  394. }
  395. }
  396. #[cfg(test)]
  397. mod test {
  398. use super::*;
  399. macro_rules! check_cidr_test_array {
  400. ($tests:expr, $from_str:path, $variant:path) => {
  401. for &(s, cidr) in &$tests {
  402. assert_eq!($from_str(s), cidr);
  403. assert_eq!(IpCidr::from_str(s), cidr.map($variant));
  404. if let Ok(cidr) = cidr {
  405. assert_eq!($from_str(&format!("{}", cidr)), Ok(cidr));
  406. assert_eq!(IpCidr::from_str(&format!("{}", cidr)),
  407. Ok($variant(cidr)));
  408. }
  409. }
  410. }
  411. }
  412. #[test]
  413. #[cfg(all(feature = "proto-ipv4", feature = "ethernet"))]
  414. fn test_mac() {
  415. assert_eq!(EthernetAddress::from_str(""), Err(()));
  416. assert_eq!(EthernetAddress::from_str("02:00:00:00:00:00"),
  417. Ok(EthernetAddress([0x02, 0x00, 0x00, 0x00, 0x00, 0x00])));
  418. assert_eq!(EthernetAddress::from_str("01:23:45:67:89:ab"),
  419. Ok(EthernetAddress([0x01, 0x23, 0x45, 0x67, 0x89, 0xab])));
  420. assert_eq!(EthernetAddress::from_str("cd:ef:10:00:00:00"),
  421. Ok(EthernetAddress([0xcd, 0xef, 0x10, 0x00, 0x00, 0x00])));
  422. assert_eq!(EthernetAddress::from_str("00:00:00:ab:cd:ef"),
  423. Ok(EthernetAddress([0x00, 0x00, 0x00, 0xab, 0xcd, 0xef])));
  424. assert_eq!(EthernetAddress::from_str("00-00-00-ab-cd-ef"),
  425. Ok(EthernetAddress([0x00, 0x00, 0x00, 0xab, 0xcd, 0xef])));
  426. assert_eq!(EthernetAddress::from_str("AB-CD-EF-00-00-00"),
  427. Ok(EthernetAddress([0xab, 0xcd, 0xef, 0x00, 0x00, 0x00])));
  428. assert_eq!(EthernetAddress::from_str("100:00:00:00:00:00"), Err(()));
  429. assert_eq!(EthernetAddress::from_str("002:00:00:00:00:00"), Err(()));
  430. assert_eq!(EthernetAddress::from_str("02:00:00:00:00:000"), Err(()));
  431. assert_eq!(EthernetAddress::from_str("02:00:00:00:00:0x"), Err(()));
  432. }
  433. #[test]
  434. #[cfg(feature = "proto-ipv4")]
  435. fn test_ipv4() {
  436. assert_eq!(Ipv4Address::from_str(""), Err(()));
  437. assert_eq!(Ipv4Address::from_str("1.2.3.4"),
  438. Ok(Ipv4Address([1, 2, 3, 4])));
  439. assert_eq!(Ipv4Address::from_str("001.2.3.4"),
  440. Ok(Ipv4Address([1, 2, 3, 4])));
  441. assert_eq!(Ipv4Address::from_str("0001.2.3.4"), Err(()));
  442. assert_eq!(Ipv4Address::from_str("999.2.3.4"), Err(()));
  443. assert_eq!(Ipv4Address::from_str("1.2.3.4.5"), Err(()));
  444. assert_eq!(Ipv4Address::from_str("1.2.3"), Err(()));
  445. assert_eq!(Ipv4Address::from_str("1.2.3."), Err(()));
  446. assert_eq!(Ipv4Address::from_str("1.2.3.4."), Err(()));
  447. }
  448. #[test]
  449. #[cfg(feature = "proto-ipv6")]
  450. fn test_ipv6() {
  451. // Obviously not valid
  452. assert_eq!(Ipv6Address::from_str(""), Err(()));
  453. assert_eq!(Ipv6Address::from_str("fe80:0:0:0:0:0:0:1"),
  454. Ok(Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1)));
  455. assert_eq!(Ipv6Address::from_str("::1"),
  456. Ok(Ipv6Address::LOOPBACK));
  457. assert_eq!(Ipv6Address::from_str("::"),
  458. Ok(Ipv6Address::UNSPECIFIED));
  459. assert_eq!(Ipv6Address::from_str("fe80::1"),
  460. Ok(Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1)));
  461. assert_eq!(Ipv6Address::from_str("1234:5678::"),
  462. Ok(Ipv6Address::new(0x1234, 0x5678, 0, 0, 0, 0, 0, 0)));
  463. assert_eq!(Ipv6Address::from_str("1234:5678::8765:4321"),
  464. Ok(Ipv6Address::new(0x1234, 0x5678, 0, 0, 0, 0, 0x8765, 0x4321)));
  465. // Two double colons in address
  466. assert_eq!(Ipv6Address::from_str("1234:5678::1::1"),
  467. Err(()));
  468. assert_eq!(Ipv6Address::from_str("4444:333:22:1::4"),
  469. Ok(Ipv6Address::new(0x4444, 0x0333, 0x0022, 0x0001, 0, 0, 0, 4)));
  470. assert_eq!(Ipv6Address::from_str("1:1:1:1:1:1::"),
  471. Ok(Ipv6Address::new(1, 1, 1, 1, 1, 1, 0, 0)));
  472. assert_eq!(Ipv6Address::from_str("::1:1:1:1:1:1"),
  473. Ok(Ipv6Address::new(0, 0, 1, 1, 1, 1, 1, 1)));
  474. assert_eq!(Ipv6Address::from_str("::1:1:1:1:1:1:1"),
  475. Err(()));
  476. // Double colon appears too late indicating an address that is too long
  477. assert_eq!(Ipv6Address::from_str("1:1:1:1:1:1:1::"),
  478. Err(()));
  479. // Section after double colon is too long for a valid address
  480. assert_eq!(Ipv6Address::from_str("::1:1:1:1:1:1:1"),
  481. Err(()));
  482. // Obviously too long
  483. assert_eq!(Ipv6Address::from_str("1:1:1:1:1:1:1:1:1"),
  484. Err(()));
  485. // Address is too short
  486. assert_eq!(Ipv6Address::from_str("1:1:1:1:1:1:1"),
  487. Err(()));
  488. // Long number
  489. assert_eq!(Ipv6Address::from_str("::000001"),
  490. Err(()));
  491. // IPv4-Mapped address
  492. assert_eq!(Ipv6Address::from_str("::ffff:192.168.1.1"),
  493. Ok(Ipv6Address([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 192, 168, 1, 1])));
  494. assert_eq!(Ipv6Address::from_str("0:0:0:0:0:ffff:192.168.1.1"),
  495. Ok(Ipv6Address([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 192, 168, 1, 1])));
  496. assert_eq!(Ipv6Address::from_str("0::ffff:192.168.1.1"),
  497. Ok(Ipv6Address([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 192, 168, 1, 1])));
  498. // Only ffff is allowed in position 6 when IPv4 mapped
  499. assert_eq!(Ipv6Address::from_str("0:0:0:0:0:eeee:192.168.1.1"),
  500. Err(()));
  501. // Positions 1-5 must be 0 when IPv4 mapped
  502. assert_eq!(Ipv6Address::from_str("0:0:0:0:1:ffff:192.168.1.1"),
  503. Err(()));
  504. assert_eq!(Ipv6Address::from_str("1::ffff:192.168.1.1"),
  505. Err(()));
  506. // Out of range ipv4 octet
  507. assert_eq!(Ipv6Address::from_str("0:0:0:0:0:ffff:256.168.1.1"),
  508. Err(()));
  509. // Invalid hex in ipv4 octet
  510. assert_eq!(Ipv6Address::from_str("0:0:0:0:0:ffff:c0.168.1.1"),
  511. Err(()));
  512. }
  513. #[test]
  514. #[cfg(feature = "proto-ipv4")]
  515. fn test_ip_ipv4() {
  516. assert_eq!(IpAddress::from_str(""), Err(()));
  517. assert_eq!(IpAddress::from_str("1.2.3.4"),
  518. Ok(IpAddress::Ipv4(Ipv4Address([1, 2, 3, 4]))));
  519. assert_eq!(IpAddress::from_str("x"), Err(()));
  520. }
  521. #[test]
  522. #[cfg(feature = "proto-ipv6")]
  523. fn test_ip_ipv6() {
  524. assert_eq!(IpAddress::from_str(""), Err(()));
  525. assert_eq!(IpAddress::from_str("fe80::1"),
  526. Ok(IpAddress::Ipv6(Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1))));
  527. assert_eq!(IpAddress::from_str("x"), Err(()));
  528. }
  529. #[test]
  530. #[cfg(feature = "proto-ipv4")]
  531. fn test_cidr_ipv4() {
  532. let tests = [
  533. ("127.0.0.1/8",
  534. Ok(Ipv4Cidr::new(Ipv4Address([127, 0, 0, 1]), 8u8))),
  535. ("192.168.1.1/24",
  536. Ok(Ipv4Cidr::new(Ipv4Address([192, 168, 1, 1]), 24u8))),
  537. ("8.8.8.8/32",
  538. Ok(Ipv4Cidr::new(Ipv4Address([8, 8, 8, 8]), 32u8))),
  539. ("8.8.8.8/0",
  540. Ok(Ipv4Cidr::new(Ipv4Address([8, 8, 8, 8]), 0u8))),
  541. ("", Err(())),
  542. ("1", Err(())),
  543. ("127.0.0.1", Err(())),
  544. ("127.0.0.1/", Err(())),
  545. ("127.0.0.1/33", Err(())),
  546. ("127.0.0.1/111", Err(())),
  547. ("/32", Err(())),
  548. ];
  549. check_cidr_test_array!(tests, Ipv4Cidr::from_str, IpCidr::Ipv4);
  550. }
  551. #[test]
  552. #[cfg(feature = "proto-ipv6")]
  553. fn test_cidr_ipv6() {
  554. let tests = [
  555. ("fe80::1/64",
  556. Ok(Ipv6Cidr::new(Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1), 64u8))),
  557. ("fe80::/64",
  558. Ok(Ipv6Cidr::new(Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 0), 64u8))),
  559. ("::1/128",
  560. Ok(Ipv6Cidr::new(Ipv6Address::LOOPBACK, 128u8))),
  561. ("::/128",
  562. Ok(Ipv6Cidr::new(Ipv6Address::UNSPECIFIED, 128u8))),
  563. ("fe80:0:0:0:0:0:0:1/64",
  564. Ok(Ipv6Cidr::new(Ipv6Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1), 64u8))),
  565. ("fe80:0:0:0:0:0:0:1|64",
  566. Err(())),
  567. ("fe80::|64",
  568. Err(())),
  569. ("fe80::1::/64",
  570. Err(()))
  571. ];
  572. check_cidr_test_array!(tests, Ipv6Cidr::from_str, IpCidr::Ipv6);
  573. }
  574. #[test]
  575. #[cfg(feature = "proto-ipv4")]
  576. fn test_endpoint_ipv4() {
  577. assert_eq!(IpEndpoint::from_str(""), Err(()));
  578. assert_eq!(IpEndpoint::from_str("x"), Err(()));
  579. assert_eq!(
  580. IpEndpoint::from_str("127.0.0.1"),
  581. Ok(IpEndpoint { addr: IpAddress::v4(127, 0, 0, 1), port: 0 })
  582. );
  583. assert_eq!(
  584. IpEndpoint::from_str("127.0.0.1:12345"),
  585. Ok(IpEndpoint { addr: IpAddress::v4(127, 0, 0, 1), port: 12345 })
  586. );
  587. }
  588. #[test]
  589. #[cfg(feature = "proto-ipv6")]
  590. fn test_endpoint_ipv6() {
  591. assert_eq!(IpEndpoint::from_str(""), Err(()));
  592. assert_eq!(IpEndpoint::from_str("x"), Err(()));
  593. assert_eq!(
  594. IpEndpoint::from_str("fe80::1"),
  595. Ok(IpEndpoint { addr: IpAddress::v6(0xfe80, 0, 0, 0, 0, 0, 0, 1), port: 0 })
  596. );
  597. assert_eq!(
  598. IpEndpoint::from_str("[fe80::1]:12345"),
  599. Ok(IpEndpoint { addr: IpAddress::v6(0xfe80, 0, 0, 0, 0, 0, 0, 1), port: 12345 })
  600. );
  601. }
  602. }