Skip to content

Commit 864bdf9

Browse files
authored
Report error on duplicate named arg names (#4367)
1 parent b776cf6 commit 864bdf9

File tree

2 files changed

+18
-0
lines changed

2 files changed

+18
-0
lines changed

include/fmt/base.h

+16
Original file line numberDiff line numberDiff line change
@@ -1064,13 +1064,28 @@ template <typename Char> struct named_arg_info {
10641064
int id;
10651065
};
10661066

1067+
template <typename Char>
1068+
FMT_CONSTEXPR void check_for_duplicate(named_arg_info<Char>* named_args,
1069+
int named_arg_index,
1070+
basic_string_view<Char> arg_name) {
1071+
if (named_arg_index <= 0) return;
1072+
1073+
for (auto i = 0; i < named_arg_index; ++i) {
1074+
if (basic_string_view<Char>(named_args[i].name) == arg_name) {
1075+
report_error("duplicate named args found");
1076+
}
1077+
}
1078+
}
1079+
10671080
template <typename Char, typename T, FMT_ENABLE_IF(!is_named_arg<T>::value)>
10681081
void init_named_arg(named_arg_info<Char>*, int& arg_index, int&, const T&) {
10691082
++arg_index;
10701083
}
1084+
10711085
template <typename Char, typename T, FMT_ENABLE_IF(is_named_arg<T>::value)>
10721086
void init_named_arg(named_arg_info<Char>* named_args, int& arg_index,
10731087
int& named_arg_index, const T& arg) {
1088+
check_for_duplicate<Char>(named_args, named_arg_index, arg.name);
10741089
named_args[named_arg_index++] = {arg.name, arg_index++};
10751090
}
10761091

@@ -1084,6 +1099,7 @@ template <typename T, typename Char,
10841099
FMT_ENABLE_IF(is_static_named_arg<T>::value)>
10851100
FMT_CONSTEXPR void init_static_named_arg(named_arg_info<Char>* named_args,
10861101
int& arg_index, int& named_arg_index) {
1102+
check_for_duplicate<Char>(named_args, named_arg_index, T::name);
10871103
named_args[named_arg_index++] = {T::name, arg_index++};
10881104
}
10891105

test/format-test.cc

+2
Original file line numberDiff line numberDiff line change
@@ -599,6 +599,8 @@ TEST(format_test, named_arg) {
599599
EXPECT_THROW_MSG((void)fmt::format(runtime("{a} {}"), fmt::arg("a", 2), 42),
600600
format_error,
601601
"cannot switch from manual to automatic argument indexing");
602+
EXPECT_THROW_MSG((void)fmt::format("{a}", fmt::arg("a", 1),
603+
fmt::arg("a", 10)), format_error, "duplicate named args found");
602604
}
603605

604606
TEST(format_test, auto_arg_index) {

0 commit comments

Comments
 (0)