use core::fmt::{self, Write}; #[test] fn test_format_f64() { assert_eq!("1", format!("{:.0}", 1.0f64)); assert_eq!("9", format!("{:.0}", 9.4f64)); assert_eq!("10", format!("{:.0}", 9.9f64)); assert_eq!("9.8", format!("{:.1}", 9.849f64)); assert_eq!("9.9", format!("{:.1}", 9.851f64)); assert_eq!("0", format!("{:.0}", 0.5f64)); assert_eq!("1.23456789e6", format!("{:e}", 1234567.89f64)); assert_eq!("1.23456789e3", format!("{:e}", 1234.56789f64)); assert_eq!("1.23456789E6", format!("{:E}", 1234567.89f64)); assert_eq!("1.23456789E3", format!("{:E}", 1234.56789f64)); assert_eq!("0.0", format!("{:?}", 0.0f64)); assert_eq!("1.01", format!("{:?}", 1.01f64)); let high_cutoff = 1e16_f64; assert_eq!("1e16", format!("{:?}", high_cutoff)); assert_eq!("-1e16", format!("{:?}", -high_cutoff)); assert!(!is_exponential(&format!("{:?}", high_cutoff * (1.0 - 2.0 * f64::EPSILON)))); assert_eq!("-3.0", format!("{:?}", -3f64)); assert_eq!("0.0001", format!("{:?}", 0.0001f64)); assert_eq!("9e-5", format!("{:?}", 0.00009f64)); assert_eq!("1234567.9", format!("{:.1?}", 1234567.89f64)); assert_eq!("1234.6", format!("{:.1?}", 1234.56789f64)); } #[test] fn test_format_f64_rounds_ties_to_even() { assert_eq!("0", format!("{:.0}", 0.5f64)); assert_eq!("2", format!("{:.0}", 1.5f64)); assert_eq!("2", format!("{:.0}", 2.5f64)); assert_eq!("4", format!("{:.0}", 3.5f64)); assert_eq!("4", format!("{:.0}", 4.5f64)); assert_eq!("6", format!("{:.0}", 5.5f64)); assert_eq!("128", format!("{:.0}", 127.5f64)); assert_eq!("128", format!("{:.0}", 128.5f64)); assert_eq!("0.2", format!("{:.1}", 0.25f64)); assert_eq!("0.8", format!("{:.1}", 0.75f64)); assert_eq!("0.12", format!("{:.2}", 0.125f64)); assert_eq!("0.88", format!("{:.2}", 0.875f64)); assert_eq!("0.062", format!("{:.3}", 0.062f64)); assert_eq!("-0", format!("{:.0}", -0.5f64)); assert_eq!("-2", format!("{:.0}", -1.5f64)); assert_eq!("-2", format!("{:.0}", -2.5f64)); assert_eq!("-4", format!("{:.0}", -3.5f64)); assert_eq!("-4", format!("{:.0}", -4.5f64)); assert_eq!("-6", format!("{:.0}", -5.5f64)); assert_eq!("-128", format!("{:.0}", -127.5f64)); assert_eq!("-128", format!("{:.0}", -128.5f64)); assert_eq!("-0.2", format!("{:.1}", -0.25f64)); assert_eq!("-0.8", format!("{:.1}", -0.75f64)); assert_eq!("-0.12", format!("{:.2}", -0.125f64)); assert_eq!("-0.88", format!("{:.2}", -0.875f64)); assert_eq!("-0.062", format!("{:.3}", -0.062f64)); assert_eq!("2e0", format!("{:.0e}", 1.5f64)); assert_eq!("2e0", format!("{:.0e}", 2.5f64)); assert_eq!("4e0", format!("{:.0e}", 3.5f64)); assert_eq!("4e0", format!("{:.0e}", 4.5f64)); assert_eq!("6e0", format!("{:.0e}", 5.5f64)); assert_eq!("1.28e2", format!("{:.2e}", 127.5f64)); assert_eq!("1.28e2", format!("{:.2e}", 128.5f64)); assert_eq!("-2e0", format!("{:.0e}", -1.5f64)); assert_eq!("-2e0", format!("{:.0e}", -2.5f64)); assert_eq!("-4e0", format!("{:.0e}", -3.5f64)); assert_eq!("-4e0", format!("{:.0e}", -4.5f64)); assert_eq!("-6e0", format!("{:.0e}", -5.5f64)); assert_eq!("-1.28e2", format!("{:.2e}", -127.5f64)); assert_eq!("-1.28e2", format!("{:.2e}", -128.5f64)); assert_eq!("2E0", format!("{:.0E}", 1.5f64)); assert_eq!("2E0", format!("{:.0E}", 2.5f64)); assert_eq!("4E0", format!("{:.0E}", 3.5f64)); assert_eq!("4E0", format!("{:.0E}", 4.5f64)); assert_eq!("6E0", format!("{:.0E}", 5.5f64)); assert_eq!("1.28E2", format!("{:.2E}", 127.5f64)); assert_eq!("1.28E2", format!("{:.2E}", 128.5f64)); assert_eq!("-2E0", format!("{:.0E}", -1.5f64)); assert_eq!("-2E0", format!("{:.0E}", -2.5f64)); assert_eq!("-4E0", format!("{:.0E}", -3.5f64)); assert_eq!("-4E0", format!("{:.0E}", -4.5f64)); assert_eq!("-6E0", format!("{:.0E}", -5.5f64)); assert_eq!("-1.28E2", format!("{:.2E}", -127.5f64)); assert_eq!("-1.28E2", format!("{:.2E}", -128.5f64)); } #[test] fn test_format_f32() { assert_eq!("1", format!("{:.0}", 1.0f32)); assert_eq!("9", format!("{:.0}", 9.4f32)); assert_eq!("10", format!("{:.0}", 9.9f32)); assert_eq!("9.8", format!("{:.1}", 9.849f32)); assert_eq!("9.9", format!("{:.1}", 9.851f32)); assert_eq!("0", format!("{:.0}", 0.5f32)); assert_eq!("1.2345679e6", format!("{:e}", 1234567.89f32)); assert_eq!("1.2345679e3", format!("{:e}", 1234.56789f32)); assert_eq!("1.2345679E6", format!("{:E}", 1234567.89f32)); assert_eq!("1.2345679E3", format!("{:E}", 1234.56789f32)); assert_eq!("0.0", format!("{:?}", 0.0f32)); assert_eq!("1.01", format!("{:?}", 1.01f32)); let high_cutoff = 1e16_f32; assert_eq!("1e16", format!("{:?}", high_cutoff)); assert_eq!("-1e16", format!("{:?}", -high_cutoff)); assert!(!is_exponential(&format!("{:?}", high_cutoff * (1.0 - 2.0 * f32::EPSILON)))); assert_eq!("-3.0", format!("{:?}", -3f32)); assert_eq!("0.0001", format!("{:?}", 0.0001f32)); assert_eq!("9e-5", format!("{:?}", 0.00009f32)); assert_eq!("1234567.9", format!("{:.1?}", 1234567.89f32)); assert_eq!("1234.6", format!("{:.1?}", 1234.56789f32)); } #[test] fn test_format_f32_rounds_ties_to_even() { assert_eq!("0", format!("{:.0}", 0.5f32)); assert_eq!("2", format!("{:.0}", 1.5f32)); assert_eq!("2", format!("{:.0}", 2.5f32)); assert_eq!("4", format!("{:.0}", 3.5f32)); assert_eq!("4", format!("{:.0}", 4.5f32)); assert_eq!("6", format!("{:.0}", 5.5f32)); assert_eq!("128", format!("{:.0}", 127.5f32)); assert_eq!("128", format!("{:.0}", 128.5f32)); assert_eq!("0.2", format!("{:.1}", 0.25f32)); assert_eq!("0.8", format!("{:.1}", 0.75f32)); assert_eq!("0.12", format!("{:.2}", 0.125f32)); assert_eq!("0.88", format!("{:.2}", 0.875f32)); assert_eq!("0.062", format!("{:.3}", 0.062f32)); assert_eq!("-0", format!("{:.0}", -0.5f32)); assert_eq!("-2", format!("{:.0}", -1.5f32)); assert_eq!("-2", format!("{:.0}", -2.5f32)); assert_eq!("-4", format!("{:.0}", -3.5f32)); assert_eq!("-4", format!("{:.0}", -4.5f32)); assert_eq!("-6", format!("{:.0}", -5.5f32)); assert_eq!("-128", format!("{:.0}", -127.5f32)); assert_eq!("-128", format!("{:.0}", -128.5f32)); assert_eq!("-0.2", format!("{:.1}", -0.25f32)); assert_eq!("-0.8", format!("{:.1}", -0.75f32)); assert_eq!("-0.12", format!("{:.2}", -0.125f32)); assert_eq!("-0.88", format!("{:.2}", -0.875f32)); assert_eq!("-0.062", format!("{:.3}", -0.062f32)); assert_eq!("2e0", format!("{:.0e}", 1.5f32)); assert_eq!("2e0", format!("{:.0e}", 2.5f32)); assert_eq!("4e0", format!("{:.0e}", 3.5f32)); assert_eq!("4e0", format!("{:.0e}", 4.5f32)); assert_eq!("6e0", format!("{:.0e}", 5.5f32)); assert_eq!("1.28e2", format!("{:.2e}", 127.5f32)); assert_eq!("1.28e2", format!("{:.2e}", 128.5f32)); assert_eq!("-2e0", format!("{:.0e}", -1.5f32)); assert_eq!("-2e0", format!("{:.0e}", -2.5f32)); assert_eq!("-4e0", format!("{:.0e}", -3.5f32)); assert_eq!("-4e0", format!("{:.0e}", -4.5f32)); assert_eq!("-6e0", format!("{:.0e}", -5.5f32)); assert_eq!("-1.28e2", format!("{:.2e}", -127.5f32)); assert_eq!("-1.28e2", format!("{:.2e}", -128.5f32)); assert_eq!("2E0", format!("{:.0E}", 1.5f32)); assert_eq!("2E0", format!("{:.0E}", 2.5f32)); assert_eq!("4E0", format!("{:.0E}", 3.5f32)); assert_eq!("4E0", format!("{:.0E}", 4.5f32)); assert_eq!("6E0", format!("{:.0E}", 5.5f32)); assert_eq!("1.28E2", format!("{:.2E}", 127.5f32)); assert_eq!("1.28E2", format!("{:.2E}", 128.5f32)); assert_eq!("-2E0", format!("{:.0E}", -1.5f32)); assert_eq!("-2E0", format!("{:.0E}", -2.5f32)); assert_eq!("-4E0", format!("{:.0E}", -3.5f32)); assert_eq!("-4E0", format!("{:.0E}", -4.5f32)); assert_eq!("-6E0", format!("{:.0E}", -5.5f32)); assert_eq!("-1.28E2", format!("{:.2E}", -127.5f32)); assert_eq!("-1.28E2", format!("{:.2E}", -128.5f32)); } #[test] fn test_format_f64_max_precision_exponential() { struct ExactExpWriter { prefix: &'static [u8], zeroes_remaining: u32, suffix: &'static [u8], prefix_pos: usize, suffix_pos: usize, total_len: u32, } impl ExactExpWriter { fn new(prefix: &'static str, suffix: &'static str) -> Self { Self { prefix: prefix.as_bytes(), zeroes_remaining: u16::MAX.into(), suffix: suffix.as_bytes(), prefix_pos: 0, suffix_pos: 0, total_len: 0, } } fn finish(self) { assert_eq!(self.prefix_pos, self.prefix.len()); assert_eq!(self.zeroes_remaining, 0); assert_eq!(self.suffix_pos, self.suffix.len()); assert_eq!(self.total_len, u32::from(u16::MAX) + 4); } } impl Write for ExactExpWriter { fn write_str(&mut self, s: &str) -> fmt::Result { for byte in s.bytes() { self.total_len += 1; if self.prefix_pos < self.prefix.len() { assert_eq!(byte, self.prefix[self.prefix_pos]); self.prefix_pos += 1; } else if self.zeroes_remaining > 0 { assert_eq!(byte, b'0'); self.zeroes_remaining -= 1; } else { assert!(self.suffix_pos < self.suffix.len()); assert_eq!(byte, self.suffix[self.suffix_pos]); self.suffix_pos += 1; } } Ok(()) } } fn assert_exact_exp(args: fmt::Arguments<'_>, prefix: &'static str, suffix: &'static str) { let mut writer = ExactExpWriter::new(prefix, suffix); fmt::write(&mut writer, args).unwrap(); writer.finish(); } assert_exact_exp(format_args!("{:.65535e}", 0.0f64), "0.", "e0"); assert_exact_exp(format_args!("{:.65535e}", 1.0f64), "1.", "e0"); assert_exact_exp(format_args!("{:.65535E}", 0.0f64), "0.", "E0"); assert_exact_exp(format_args!("{:.65535E}", 1.0f64), "1.", "E0"); assert_exact_exp(format_args!("{:65535.65535e}", 1.0f64), "1.", "e0"); } fn is_exponential(s: &str) -> bool { s.contains("e") || s.contains("E") }