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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
|
#include "asl/format.hpp"
#include "asl/testing/testing.hpp"
#include "asl/allocator.hpp"
static_assert(asl::formattable<decltype("Hello")>);
class StringSink : public asl::Writer
{
// @Todo Use string, once we have it, or a buffer
isize_t m_current_len{};
char* m_data{};
public:
void write(asl::span<const asl::byte> str) override
{
m_data = reinterpret_cast<char*>(asl::GlobalHeap::realloc(
m_data,
asl::layout::array<char>(m_current_len),
asl::layout::array<char>(m_current_len + str.size())));
asl::memcpy(m_data + m_current_len, str.data(), str.size());
m_current_len += str.size();
}
constexpr asl::string_view str() const { return {m_data, m_current_len}; }
void reset()
{
m_current_len = 0;
asl::GlobalHeap::dealloc(m_data, asl::layout::array<char>(m_current_len));
m_data = nullptr;
}
};
ASL_TEST(format_args)
{
StringSink sink;
// @Todo Introduce ASL_TEST_EXPECT_EQ, or ASL_TEST_EXPECT_STREQ
asl::format(&sink, "Hello, world!");
ASL_TEST_EXPECT(sink.str() == "Hello, world!"_sv);
sink.reset();
asl::format(&sink, "");
ASL_TEST_EXPECT(sink.str() == ""_sv);
sink.reset();
asl::format(&sink, "Hello, {}!", "world");
ASL_TEST_EXPECT(sink.str() == "Hello, world!"_sv);
sink.reset();
asl::format(&sink, "Hello, {}! {}", "world");
ASL_TEST_EXPECT(sink.str() == "Hello, world! <ERROR>"_sv);
sink.reset();
asl::format(&sink, "Hello, pup!", "world");
ASL_TEST_EXPECT(sink.str() == "Hello, pup!"_sv);
sink.reset();
asl::format(&sink, "{}", "CHEESE");
ASL_TEST_EXPECT(sink.str() == "CHEESE"_sv);
sink.reset();
asl::format(&sink, "{ ", "CHEESE");
ASL_TEST_EXPECT(sink.str() == "<ERROR> "_sv);
sink.reset();
asl::format(&sink, "{", "CHEESE");
ASL_TEST_EXPECT(sink.str() == "<ERROR>"_sv);
sink.reset();
asl::format(&sink, "a{{b");
ASL_TEST_EXPECT(sink.str() == "a{b"_sv);
sink.reset();
asl::format(&sink, "{{{}}} }", "CHEESE");
ASL_TEST_EXPECT(sink.str() == "{CHEESE} }"_sv);
}
ASL_TEST(format_integers)
{
StringSink sink;
sink.reset();
asl::format(&sink, "{} {} {}", 0, 1, 2);
ASL_TEST_EXPECT(sink.str() == "0 1 2"_sv);
sink.reset();
asl::format(&sink, "{} {} {}", 10, 11, 12);
ASL_TEST_EXPECT(sink.str() == "10 11 12"_sv);
sink.reset();
asl::format(&sink, "{} {} {}", 100, 101, 102);
ASL_TEST_EXPECT(sink.str() == "100 101 102"_sv);
sink.reset();
asl::format(&sink, "{} {} {}", 1000, 1001, 1002);
ASL_TEST_EXPECT(sink.str() == "1000 1001 1002"_sv);
sink.reset();
asl::format(&sink, "{} {} {} {}", -1, -23, -456, -7890);
ASL_TEST_EXPECT(sink.str() == "-1 -23 -456 -7890"_sv);
}
ASL_TEST(format_boolean)
{
StringSink sink;
sink.reset();
asl::format(&sink, "{} {}", true, false);
ASL_TEST_EXPECT(sink.str() == "true false"_sv);
}
|