/* * Copyright (c) 2020, the SerenityOS developers. * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include TEST_CASE(is_integral_works_properly) { EXPECT(!IsIntegral); EXPECT(IsIntegral); } TEST_CASE(format_string_literals) { EXPECT_EQ(DeprecatedString::formatted("prefix-{}-suffix", "abc"), "prefix-abc-suffix"); EXPECT_EQ(DeprecatedString::formatted("{}{}{}", "a", "b", "c"), "abc"); } TEST_CASE(format_integers) { EXPECT_EQ(DeprecatedString::formatted("{}", 42u), "42"); EXPECT_EQ(DeprecatedString::formatted("{:4}", 42u), " 42"); EXPECT_EQ(DeprecatedString::formatted("{:08}", 42u), "00000042"); EXPECT_EQ(DeprecatedString::formatted("{:7}", -17), " -17"); EXPECT_EQ(DeprecatedString::formatted("{}", -17), "-17"); EXPECT_EQ(DeprecatedString::formatted("{:04}", 13), "0013"); EXPECT_EQ(DeprecatedString::formatted("{:08x}", 4096), "00001000"); EXPECT_EQ(DeprecatedString::formatted("{:x}", 0x1111222233334444ull), "1111222233334444"); EXPECT_EQ(DeprecatedString::formatted("{:4}", 12345678), "12345678"); EXPECT_EQ(DeprecatedString::formatted("{:'}", 0), "0"); EXPECT_EQ(DeprecatedString::formatted("{:'}", 4096), "4,096"); EXPECT_EQ(DeprecatedString::formatted("{:'}", 16777216), "16,777,216"); EXPECT_EQ(DeprecatedString::formatted("{:'}", AK::NumericLimits::max()), "18,446,744,073,709,551,615"); EXPECT_EQ(DeprecatedString::formatted("{:'}", AK::NumericLimits::max()), "18,446,744,073,709,551,615"); EXPECT_EQ(DeprecatedString::formatted("{:'}", AK::NumericLimits::min() + 1), "-9,223,372,036,854,775,807"); EXPECT_EQ(DeprecatedString::formatted("{:'x}", 0), "0"); EXPECT_EQ(DeprecatedString::formatted("{:'x}", 16777216), "1,000,000"); EXPECT_EQ(DeprecatedString::formatted("{:'x}", AK::NumericLimits::max()), "f,fff,fff,fff,fff,fff"); EXPECT_EQ(DeprecatedString::formatted("{:'x}", AK::NumericLimits::min() + 1), "-7,fff,fff,fff,fff,fff"); EXPECT_EQ(DeprecatedString::formatted("{:'b}", AK::NumericLimits::max()), "1,111,111,111,111,111,111,111,111,111,111,111,111,111,111,111,111,111,111,111,111,111"); } TEST_CASE(reorder_format_arguments) { EXPECT_EQ(DeprecatedString::formatted("{1}{0}", "a", "b"), "ba"); EXPECT_EQ(DeprecatedString::formatted("{0}{1}", "a", "b"), "ab"); // Compiletime check bypass: ignoring a passed argument. EXPECT_EQ(DeprecatedString::formatted("{0}{0}{0}"sv, "a", "b"), "aaa"); // Compiletime check bypass: ignoring a passed argument. EXPECT_EQ(DeprecatedString::formatted("{1}{}{0}"sv, "a", "b", "c"), "baa"); } TEST_CASE(escape_braces) { EXPECT_EQ(DeprecatedString::formatted("{{{}", "foo"), "{foo"); EXPECT_EQ(DeprecatedString::formatted("{}}}", "bar"), "bar}"); } TEST_CASE(everything) { EXPECT_EQ(DeprecatedString::formatted("{{{:04}/{}/{0:8}/{1}", 42u, "foo"), "{0042/foo/ 42/foo"); } TEST_CASE(string_builder) { StringBuilder builder; builder.appendff(" {} ", 42); builder.appendff("{1}{0} ", 1, 2); EXPECT_EQ(builder.to_deprecated_string(), " 42 21 "); } TEST_CASE(format_without_arguments) { EXPECT_EQ(DeprecatedString::formatted("foo"), "foo"); } TEST_CASE(format_upper_case_integer) { EXPECT_EQ(DeprecatedString::formatted("{:4X}", 0xff), " FF"); EXPECT_EQ(DeprecatedString::formatted("{:#4X}", 0xff), "0XFF"); EXPECT_EQ(DeprecatedString::formatted("{:b}", 0xff), "11111111"); EXPECT_EQ(DeprecatedString::formatted("{:B}", 0xff), "11111111"); EXPECT_EQ(DeprecatedString::formatted("{:#b}", 0xff), "0b11111111"); } TEST_CASE(format_aligned) { EXPECT_EQ(DeprecatedString::formatted("{:*<8}", 13), "13******"); EXPECT_EQ(DeprecatedString::formatted("{:*^8}", 13), "***13***"); EXPECT_EQ(DeprecatedString::formatted("{:*>8}", 13), "******13"); EXPECT_EQ(DeprecatedString::formatted("{:*>+8}", 13), "*****+13"); EXPECT_EQ(DeprecatedString::formatted("{:*^ 8}", 13), "** 13***"); } TEST_CASE(format_octal) { EXPECT_EQ(DeprecatedString::formatted("{:o}", 0744), "744"); EXPECT_EQ(DeprecatedString::formatted("{:#o}", 0744), "0744"); EXPECT_EQ(DeprecatedString::formatted("{:'o}", 054321), "54,321"); EXPECT_EQ(DeprecatedString::formatted("{:'o}", 0567012340), "567,012,340"); } TEST_CASE(zero_pad) { EXPECT_EQ(DeprecatedString::formatted("{: <010}", 42), "42 "); EXPECT_EQ(DeprecatedString::formatted("{:010}", 42), "0000000042"); EXPECT_EQ(DeprecatedString::formatted("{:/^010}", 42), "////42////"); EXPECT_EQ(DeprecatedString::formatted("{:04x}", -32), "-0020"); EXPECT_EQ(DeprecatedString::formatted("{:#06x}", -64), "-0x000040"); } TEST_CASE(replacement_field) { EXPECT_EQ(DeprecatedString::formatted("{:*>{1}}", 13, static_cast(10)), "********13"); EXPECT_EQ(DeprecatedString::formatted("{:*<{1}}", 7, 4), "7***"); // Compiletime check bypass: intentionally ignoring extra arguments EXPECT_EQ(DeprecatedString::formatted("{:{2}}"sv, -5, 8, 16), " -5"); EXPECT_EQ(DeprecatedString::formatted("{{{:*^{1}}}}", 1, 3), "{*1*}"); EXPECT_EQ(DeprecatedString::formatted("{:0{}}", 1, 3), "001"); } TEST_CASE(replacement_field_regression) { // FIXME: Compiletime check bypass: cannot parse '}}' correctly. EXPECT_EQ(DeprecatedString::formatted("{:{}}"sv, "", static_cast(6)), " "); } TEST_CASE(complex_string_specifiers) { EXPECT_EQ(DeprecatedString::formatted("{:.8}", "123456789"), "12345678"); EXPECT_EQ(DeprecatedString::formatted("{:9}", "abcd"), "abcd "); EXPECT_EQ(DeprecatedString::formatted("{:>9}", "abcd"), " abcd"); EXPECT_EQ(DeprecatedString::formatted("{:^9}", "abcd"), " abcd "); EXPECT_EQ(DeprecatedString::formatted("{:4.6}", "a"), "a "); EXPECT_EQ(DeprecatedString::formatted("{:4.6}", "abcdef"), "abcdef"); EXPECT_EQ(DeprecatedString::formatted("{:4.6}", "abcdefghi"), "abcdef"); } TEST_CASE(cast_integer_to_character) { EXPECT_EQ(DeprecatedString::formatted("{:c}", static_cast('a')), "a"); EXPECT_EQ(DeprecatedString::formatted("{:c}", static_cast('f')), "f"); } TEST_CASE(boolean_values) { EXPECT_EQ(DeprecatedString::formatted("{}", true), "true"); EXPECT_EQ(DeprecatedString::formatted("{}", false), "false"); EXPECT_EQ(DeprecatedString::formatted("{:6}", true), "true "); EXPECT_EQ(DeprecatedString::formatted("{:>4}", false), "false"); EXPECT_EQ(DeprecatedString::formatted("{:d}", false), "0"); EXPECT_EQ(DeprecatedString::formatted("{:d}", true), "1"); EXPECT_EQ(DeprecatedString::formatted("{:#08x}", true), "0x00000001"); } TEST_CASE(pointers) { void* ptr = reinterpret_cast(0x4000); if (sizeof(void*) == 4) { EXPECT_EQ(DeprecatedString::formatted("{:p}", 32), "0x00000020"); EXPECT_EQ(DeprecatedString::formatted("{:p}", ptr), "0x00004000"); EXPECT_EQ(DeprecatedString::formatted("{}", ptr), "0x00004000"); } else if (sizeof(void*) == 8) { EXPECT_EQ(DeprecatedString::formatted("{:p}", 32), "0x0000000000000020"); EXPECT_EQ(DeprecatedString::formatted("{:p}", ptr), "0x0000000000004000"); EXPECT_EQ(DeprecatedString::formatted("{}", ptr), "0x0000000000004000"); } else { VERIFY_NOT_REACHED(); } } // If the format implementation did absolutely nothing, all tests would pass. This // is because when a test fails we only write "FAIL" to stdout using format. // // This is a bit scary, thus this test. At least this test should fail in this case. TEST_CASE(ensure_that_format_works) { if (DeprecatedString::formatted("FAIL") != "FAIL") { fprintf(stderr, "FAIL\n"); exit(1); } if (DeprecatedString::formatted("{} FAIL {}", 1, 2) != "1 FAIL 2") { fprintf(stderr, "FAIL\n"); exit(1); } } TEST_CASE(format_string_literal_as_pointer) { char const* literal = "abc"; EXPECT_EQ(DeprecatedString::formatted("{:p}", literal), DeprecatedString::formatted("{:p}", reinterpret_cast(literal))); } TEST_CASE(format_character) { char a = 'a'; EXPECT_EQ(DeprecatedString::formatted("{}", true ? a : 'b'), "a"); } struct A { }; struct B { }; template<> struct AK::Formatter : Formatter { ErrorOr format(FormatBuilder& builder, B) { return Formatter::format(builder, "B"sv); } }; TEST_CASE(format_if_supported) { EXPECT_EQ(DeprecatedString::formatted("{}", FormatIfSupported { A {} }), "?"); EXPECT_EQ(DeprecatedString::formatted("{}", FormatIfSupported { B {} }), "B"); } TEST_CASE(file_descriptor) { char filename[] = "/tmp/test-file-descriptor-XXXXXX"; int fd = mkstemp(filename); FILE* file = fdopen(fd, "w+"); outln(file, "{}", "Hello, World!"); out(file, "foo"); outln(file, "bar"); rewind(file); Array buffer; auto const nread = fread(buffer.data(), 1, buffer.size(), file); EXPECT_EQ("Hello, World!\nfoobar\n"sv, StringView { buffer.span().trim(nread) }); fclose(file); unlink(filename); } TEST_CASE(floating_point_numbers) { EXPECT_EQ(DeprecatedString::formatted("{}", 1.12), "1.12"); EXPECT_EQ(DeprecatedString::formatted("{}", 1.), "1"); EXPECT_EQ(DeprecatedString::formatted("{:.3}", 1.12), "1.12"); EXPECT_EQ(DeprecatedString::formatted("{:.1}", 1.12), "1.1"); EXPECT_EQ(DeprecatedString::formatted("{}", -1.12), "-1.12"); EXPECT_EQ(DeprecatedString::formatted("{:'.4}", 1234.5678), "1,234.5678"); EXPECT_EQ(DeprecatedString::formatted("{:'.4}", -1234.5678), "-1,234.5678"); EXPECT_EQ(DeprecatedString::formatted("{}", NAN), "nan"); EXPECT_EQ(DeprecatedString::formatted("{}", INFINITY), "inf"); EXPECT_EQ(DeprecatedString::formatted("{}", -INFINITY), "-inf"); // FIXME: There is always the question what we mean with the width field. Do we mean significant digits? // Do we mean the whole width? This is what was the simplest to implement: EXPECT_EQ(DeprecatedString::formatted("{:x>5.1}", 1.12), "xx1.1"); } TEST_CASE(no_precision_no_trailing_number) { EXPECT_EQ(DeprecatedString::formatted("{:.0}", 0.1), "0"); } TEST_CASE(yay_this_implementation_sucks) { EXPECT_EQ(DeprecatedString::formatted("{:.0}", .99999999999), "0"); } TEST_CASE(precision_with_trailing_zeros) { EXPECT_EQ(DeprecatedString::formatted("{:0.3}", 1.12), "1.120"); EXPECT_EQ(DeprecatedString::formatted("{:0.1}", 1.12), "1.1"); } TEST_CASE(magnitude_less_than_zero) { EXPECT_EQ(DeprecatedString::formatted("{}", -0.654), "-0.654"); EXPECT_EQ(DeprecatedString::formatted("{}", 0.654), "0.654"); } TEST_CASE(format_nullptr) { EXPECT_EQ(DeprecatedString::formatted("{}", nullptr), DeprecatedString::formatted("{:p}", static_cast(0))); } struct C { int i; }; template<> struct AK::Formatter : AK::Formatter { ErrorOr format(FormatBuilder& builder, C c) { return AK::Formatter::format(builder, "C(i={})"sv, c.i); } }; TEST_CASE(use_format_string_formatter) { EXPECT_EQ(DeprecatedString::formatted("{:*<10}", C { 42 }), "C(i=42)***"); } TEST_CASE(long_long_regression) { EXPECT_EQ(DeprecatedString::formatted("{}", 0x0123456789abcdefLL), "81985529216486895"); StringBuilder builder; AK::FormatBuilder fmtbuilder { builder }; MUST(fmtbuilder.put_i64(0x0123456789abcdefLL)); EXPECT_EQ(builder.string_view(), "81985529216486895"); } TEST_CASE(hex_dump) { EXPECT_EQ(DeprecatedString::formatted("{:hex-dump}", "0000"), "30303030"); EXPECT_EQ(DeprecatedString::formatted("{:>4hex-dump}", "0000"), "30303030 0000"); EXPECT_EQ(DeprecatedString::formatted("{:>2hex-dump}", "0000"), "3030 00\n3030 00"); EXPECT_EQ(DeprecatedString::formatted("{:*>4hex-dump}", "0000"), "30303030****0000"); } TEST_CASE(span_format) { { Vector v { 1, 2, 3, 4 }; EXPECT_EQ(DeprecatedString::formatted("{}", v.span()), "[ 1, 2, 3, 4 ]"); EXPECT_EQ(DeprecatedString::formatted("{}", const_cast&>(v).span()), "[ 1, 2, 3, 4 ]"); } { Vector v { "1"sv, "2"sv, "3"sv, "4"sv }; EXPECT_EQ(DeprecatedString::formatted("{}", v.span()), "[ 1, 2, 3, 4 ]"); EXPECT_EQ(DeprecatedString::formatted("{}", const_cast&>(v).span()), "[ 1, 2, 3, 4 ]"); } { Vector> v { { "1"sv, "2"sv }, { "3"sv, "4"sv } }; EXPECT_EQ(DeprecatedString::formatted("{}", v.span()), "[ [ 1, 2 ], [ 3, 4 ] ]"); EXPECT_EQ(DeprecatedString::formatted("{}", const_cast&>(v).span()), "[ [ 1, 2 ], [ 3, 4 ] ]"); } } TEST_CASE(vector_format) { { Vector v { 1, 2, 3, 4 }; EXPECT_EQ(DeprecatedString::formatted("{}", v), "[ 1, 2, 3, 4 ]"); } { Vector v { "1"sv, "2"sv, "3"sv, "4"sv }; EXPECT_EQ(DeprecatedString::formatted("{}", v), "[ 1, 2, 3, 4 ]"); } { Vector> v { { "1"sv, "2"sv }, { "3"sv, "4"sv } }; EXPECT_EQ(DeprecatedString::formatted("{}", v), "[ [ 1, 2 ], [ 3, 4 ] ]"); } } TEST_CASE(format_wchar) { EXPECT_EQ(DeprecatedString::formatted("{}", L'a'), "a"); EXPECT_EQ(DeprecatedString::formatted("{}", L'\U0001F41E'), "\xF0\x9F\x90\x9E"); EXPECT_EQ(DeprecatedString::formatted("{:x}", L'a'), "61"); EXPECT_EQ(DeprecatedString::formatted("{:x}", L'\U0001F41E'), "1f41e"); EXPECT_EQ(DeprecatedString::formatted("{:d}", L'a'), "97"); EXPECT_EQ(DeprecatedString::formatted("{:d}", L'\U0001F41E'), "128030"); EXPECT_EQ(DeprecatedString::formatted("{:6}", L'a'), "a "); EXPECT_EQ(DeprecatedString::formatted("{:6d}", L'a'), " 97"); EXPECT_EQ(DeprecatedString::formatted("{:#x}", L'\U0001F41E'), "0x1f41e"); }