Skip to content

Commit 74874a6

Browse files
committed
Auto merge of #83652 - xu-cheng:ipv4-octal, r=sfackler
Disallow octal format in Ipv4 string In its original specification, leading zero in Ipv4 string is interpreted as octal literals. So a IP address 0127.0.0.1 actually means 87.0.0.1. This confusion can lead to many security vulnerabilities. Therefore, in [IETF RFC 6943], it suggests to disallow octal/hexadecimal format in Ipv4 string all together. Existing implementation already disallows hexadecimal numbers. This commit makes Parser reject octal numbers. Fixes #83648. [IETF RFC 6943]: https://tools.ietf.org/html/rfc6943#section-3.1.1
2 parents 926ec1c + 974192c commit 74874a6

File tree

3 files changed

+23
-1
lines changed

3 files changed

+23
-1
lines changed

library/std/src/net/ip.rs

+2
Original file line numberDiff line numberDiff line change
@@ -67,7 +67,9 @@ pub enum IpAddr {
6767
///
6868
/// `Ipv4Addr` provides a [`FromStr`] implementation. The four octets are in decimal
6969
/// notation, divided by `.` (this is called "dot-decimal notation").
70+
/// Notably, octal numbers and hexadecimal numbers are not allowed per [IETF RFC 6943].
7071
///
72+
/// [IETF RFC 6943]: https://tools.ietf.org/html/rfc6943#section-3.1.1
7173
/// [`FromStr`]: crate::str::FromStr
7274
///
7375
/// # Examples

library/std/src/net/parser.rs

+13-1
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,11 @@ impl<'a> Parser<'a> {
6767
if self.state.is_empty() { result } else { None }.ok_or(AddrParseError(()))
6868
}
6969

70+
/// Peek the next character from the input
71+
fn peek_char(&self) -> Option<char> {
72+
self.state.first().map(|&b| char::from(b))
73+
}
74+
7075
/// Read the next character from the input
7176
fn read_char(&mut self) -> Option<char> {
7277
self.state.split_first().map(|(&b, tail)| {
@@ -132,7 +137,14 @@ impl<'a> Parser<'a> {
132137
let mut groups = [0; 4];
133138

134139
for (i, slot) in groups.iter_mut().enumerate() {
135-
*slot = p.read_separator('.', i, |p| p.read_number(10, None))?;
140+
*slot = p.read_separator('.', i, |p| {
141+
// Disallow octal number in IP string.
142+
// https://tools.ietf.org/html/rfc6943#section-3.1.1
143+
match (p.peek_char(), p.read_number(10, None)) {
144+
(Some('0'), Some(number)) if number != 0 => None,
145+
(_, number) => number,
146+
}
147+
})?;
136148
}
137149

138150
Some(groups.into())

library/std/src/net/parser/tests.rs

+8
Original file line numberDiff line numberDiff line change
@@ -8,11 +8,15 @@ const SCOPE_ID: u32 = 1337;
88
const IPV4: Ipv4Addr = Ipv4Addr::new(192, 168, 0, 1);
99
const IPV4_STR: &str = "192.168.0.1";
1010
const IPV4_STR_PORT: &str = "192.168.0.1:8080";
11+
const IPV4_STR_WITH_OCTAL: &str = "0127.0.0.1";
12+
const IPV4_STR_WITH_HEX: &str = "0x10.0.0.1";
1113

1214
const IPV6: Ipv6Addr = Ipv6Addr::new(0x2001, 0xdb8, 0, 0, 0, 0, 0xc0a8, 0x1);
1315
const IPV6_STR_FULL: &str = "2001:db8:0:0:0:0:c0a8:1";
1416
const IPV6_STR_COMPRESS: &str = "2001:db8::c0a8:1";
1517
const IPV6_STR_V4: &str = "2001:db8::192.168.0.1";
18+
const IPV6_STR_V4_WITH_OCTAL: &str = "2001:db8::0127.0.0.1";
19+
const IPV6_STR_V4_WITH_HEX: &str = "2001:db8::0x10.0.0.1";
1620
const IPV6_STR_PORT: &str = "[2001:db8::c0a8:1]:8080";
1721
const IPV6_STR_PORT_SCOPE_ID: &str = "[2001:db8::c0a8:1%1337]:8080";
1822

@@ -22,6 +26,8 @@ fn parse_ipv4() {
2226
assert_eq!(result, IPV4);
2327

2428
assert!(Ipv4Addr::from_str(IPV4_STR_PORT).is_err());
29+
assert!(Ipv4Addr::from_str(IPV4_STR_WITH_OCTAL).is_err());
30+
assert!(Ipv4Addr::from_str(IPV4_STR_WITH_HEX).is_err());
2531
assert!(Ipv4Addr::from_str(IPV6_STR_FULL).is_err());
2632
assert!(Ipv4Addr::from_str(IPV6_STR_COMPRESS).is_err());
2733
assert!(Ipv4Addr::from_str(IPV6_STR_V4).is_err());
@@ -39,6 +45,8 @@ fn parse_ipv6() {
3945
let result: Ipv6Addr = IPV6_STR_V4.parse().unwrap();
4046
assert_eq!(result, IPV6);
4147

48+
assert!(Ipv6Addr::from_str(IPV6_STR_V4_WITH_OCTAL).is_err());
49+
assert!(Ipv6Addr::from_str(IPV6_STR_V4_WITH_HEX).is_err());
4250
assert!(Ipv6Addr::from_str(IPV4_STR).is_err());
4351
assert!(Ipv6Addr::from_str(IPV4_STR_PORT).is_err());
4452
assert!(Ipv6Addr::from_str(IPV6_STR_PORT).is_err());

0 commit comments

Comments
 (0)